roojs-ui.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         try {
4671            
4672             if(this.compiled){
4673                 return this.compiled(values);
4674             }
4675             var useF = this.disableFormats !== true;
4676             var fm = Roo.util.Format, tpl = this;
4677             var fn = function(m, name, format, args){
4678                 if(format && useF){
4679                     if(format.substr(0, 5) == "this."){
4680                         return tpl.call(format.substr(5), values[name], values);
4681                     }else{
4682                         if(args){
4683                             // quoted values are required for strings in compiled templates, 
4684                             // but for non compiled we need to strip them
4685                             // quoted reversed for jsmin
4686                             var re = /^\s*['"](.*)["']\s*$/;
4687                             args = args.split(',');
4688                             for(var i = 0, len = args.length; i < len; i++){
4689                                 args[i] = args[i].replace(re, "$1");
4690                             }
4691                             args = [values[name]].concat(args);
4692                         }else{
4693                             args = [values[name]];
4694                         }
4695                         return fm[format].apply(fm, args);
4696                     }
4697                 }else{
4698                     return values[name] !== undefined ? values[name] : "";
4699                 }
4700             };
4701             return this.html.replace(this.re, fn);
4702         } catch (e) {
4703             Roo.log(e);
4704             throw e;
4705         }
4706          
4707     },
4708     
4709     loading : false,
4710       
4711     load : function ()
4712     {
4713          
4714         if (this.loading) {
4715             return;
4716         }
4717         var _t = this;
4718         
4719         this.loading = true;
4720         this.compiled = false;
4721         
4722         var cx = new Roo.data.Connection();
4723         cx.request({
4724             url : this.url,
4725             method : 'GET',
4726             success : function (response) {
4727                 _t.loading = false;
4728                 _t.html = response.responseText;
4729                 _t.url = false;
4730                 _t.compile();
4731              },
4732             failure : function(response) {
4733                 Roo.log("Template failed to load from " + _t.url);
4734                 _t.loading = false;
4735             }
4736         });
4737     },
4738
4739     /**
4740      * Sets the HTML used as the template and optionally compiles it.
4741      * @param {String} html
4742      * @param {Boolean} compile (optional) True to compile the template (defaults to undefined)
4743      * @return {Roo.Template} this
4744      */
4745     set : function(html, compile){
4746         this.html = html;
4747         this.compiled = null;
4748         if(compile){
4749             this.compile();
4750         }
4751         return this;
4752     },
4753     
4754     /**
4755      * True to disable format functions (defaults to false)
4756      * @type Boolean
4757      */
4758     disableFormats : false,
4759     
4760     /**
4761     * The regular expression used to match template variables 
4762     * @type RegExp
4763     * @property 
4764     */
4765     re : /\{([\w-]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
4766     
4767     /**
4768      * Compiles the template into an internal function, eliminating the RegEx overhead.
4769      * @return {Roo.Template} this
4770      */
4771     compile : function(){
4772         var fm = Roo.util.Format;
4773         var useF = this.disableFormats !== true;
4774         var sep = Roo.isGecko ? "+" : ",";
4775         var fn = function(m, name, format, args){
4776             if(format && useF){
4777                 args = args ? ',' + args : "";
4778                 if(format.substr(0, 5) != "this."){
4779                     format = "fm." + format + '(';
4780                 }else{
4781                     format = 'this.call("'+ format.substr(5) + '", ';
4782                     args = ", values";
4783                 }
4784             }else{
4785                 args= ''; format = "(values['" + name + "'] == undefined ? '' : ";
4786             }
4787             return "'"+ sep + format + "values['" + name + "']" + args + ")"+sep+"'";
4788         };
4789         var body;
4790         // branched to use + in gecko and [].join() in others
4791         if(Roo.isGecko){
4792             body = "this.compiled = function(values){ return '" +
4793                    this.html.replace(/\\/g, '\\\\').replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
4794                     "';};";
4795         }else{
4796             body = ["this.compiled = function(values){ return ['"];
4797             body.push(this.html.replace(/\\/g, '\\\\').replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn));
4798             body.push("'].join('');};");
4799             body = body.join('');
4800         }
4801         /**
4802          * eval:var:values
4803          * eval:var:fm
4804          */
4805         eval(body);
4806         return this;
4807     },
4808     
4809     // private function used to call members
4810     call : function(fnName, value, allValues){
4811         return this[fnName](value, allValues);
4812     },
4813     
4814     /**
4815      * Applies the supplied values to the template and inserts the new node(s) as the first child of el.
4816      * @param {String/HTMLElement/Roo.Element} el The context element
4817      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4818      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4819      * @return {HTMLElement/Roo.Element} The new node or Element
4820      */
4821     insertFirst: function(el, values, returnElement){
4822         return this.doInsert('afterBegin', el, values, returnElement);
4823     },
4824
4825     /**
4826      * Applies the supplied values to the template and inserts the new node(s) before el.
4827      * @param {String/HTMLElement/Roo.Element} el The context element
4828      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4829      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4830      * @return {HTMLElement/Roo.Element} The new node or Element
4831      */
4832     insertBefore: function(el, values, returnElement){
4833         return this.doInsert('beforeBegin', el, values, returnElement);
4834     },
4835
4836     /**
4837      * Applies the supplied values to the template and inserts the new node(s) after el.
4838      * @param {String/HTMLElement/Roo.Element} el The context element
4839      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4840      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4841      * @return {HTMLElement/Roo.Element} The new node or Element
4842      */
4843     insertAfter : function(el, values, returnElement){
4844         return this.doInsert('afterEnd', el, values, returnElement);
4845     },
4846     
4847     /**
4848      * Applies the supplied values to the template and appends the new node(s) to el.
4849      * @param {String/HTMLElement/Roo.Element} el The context element
4850      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4851      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4852      * @return {HTMLElement/Roo.Element} The new node or Element
4853      */
4854     append : function(el, values, returnElement){
4855         return this.doInsert('beforeEnd', el, values, returnElement);
4856     },
4857
4858     doInsert : function(where, el, values, returnEl){
4859         el = Roo.getDom(el);
4860         var newNode = Roo.DomHelper.insertHtml(where, el, this.applyTemplate(values));
4861         return returnEl ? Roo.get(newNode, true) : newNode;
4862     },
4863
4864     /**
4865      * Applies the supplied values to the template and overwrites the content of el with the new node(s).
4866      * @param {String/HTMLElement/Roo.Element} el The context element
4867      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4868      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4869      * @return {HTMLElement/Roo.Element} The new node or Element
4870      */
4871     overwrite : function(el, values, returnElement){
4872         el = Roo.getDom(el);
4873         el.innerHTML = this.applyTemplate(values);
4874         return returnElement ? Roo.get(el.firstChild, true) : el.firstChild;
4875     }
4876 };
4877 /**
4878  * Alias for {@link #applyTemplate}
4879  * @method
4880  */
4881 Roo.Template.prototype.apply = Roo.Template.prototype.applyTemplate;
4882
4883 // backwards compat
4884 Roo.DomHelper.Template = Roo.Template;
4885
4886 /**
4887  * Creates a template from the passed element's value (<i>display:none</i> textarea, preferred) or innerHTML.
4888  * @param {String/HTMLElement} el A DOM element or its id
4889  * @returns {Roo.Template} The created template
4890  * @static
4891  */
4892 Roo.Template.from = function(el){
4893     el = Roo.getDom(el);
4894     return new Roo.Template(el.value || el.innerHTML);
4895 };/*
4896  * Based on:
4897  * Ext JS Library 1.1.1
4898  * Copyright(c) 2006-2007, Ext JS, LLC.
4899  *
4900  * Originally Released Under LGPL - original licence link has changed is not relivant.
4901  *
4902  * Fork - LGPL
4903  * <script type="text/javascript">
4904  */
4905  
4906
4907 /*
4908  * This is code is also distributed under MIT license for use
4909  * with jQuery and prototype JavaScript libraries.
4910  */
4911 /**
4912  * @class Roo.DomQuery
4913 Provides high performance selector/xpath processing by compiling queries into reusable functions. New pseudo classes and matchers can be plugged. It works on HTML and XML documents (if a content node is passed in).
4914 <p>
4915 DomQuery supports most of the <a href="http://www.w3.org/TR/2005/WD-css3-selectors-20051215/">CSS3 selectors spec</a>, along with some custom selectors and basic XPath.</p>
4916
4917 <p>
4918 All selectors, attribute filters and pseudos below can be combined infinitely in any order. For example "div.foo:nth-child(odd)[@foo=bar].bar:first" would be a perfectly valid selector. Node filters are processed in the order in which they appear, which allows you to optimize your queries for your document structure.
4919 </p>
4920 <h4>Element Selectors:</h4>
4921 <ul class="list">
4922     <li> <b>*</b> any element</li>
4923     <li> <b>E</b> an element with the tag E</li>
4924     <li> <b>E F</b> All descendent elements of E that have the tag F</li>
4925     <li> <b>E > F</b> or <b>E/F</b> all direct children elements of E that have the tag F</li>
4926     <li> <b>E + F</b> all elements with the tag F that are immediately preceded by an element with the tag E</li>
4927     <li> <b>E ~ F</b> all elements with the tag F that are preceded by a sibling element with the tag E</li>
4928 </ul>
4929 <h4>Attribute Selectors:</h4>
4930 <p>The use of @ and quotes are optional. For example, div[@foo='bar'] is also a valid attribute selector.</p>
4931 <ul class="list">
4932     <li> <b>E[foo]</b> has an attribute "foo"</li>
4933     <li> <b>E[foo=bar]</b> has an attribute "foo" that equals "bar"</li>
4934     <li> <b>E[foo^=bar]</b> has an attribute "foo" that starts with "bar"</li>
4935     <li> <b>E[foo$=bar]</b> has an attribute "foo" that ends with "bar"</li>
4936     <li> <b>E[foo*=bar]</b> has an attribute "foo" that contains the substring "bar"</li>
4937     <li> <b>E[foo%=2]</b> has an attribute "foo" that is evenly divisible by 2</li>
4938     <li> <b>E[foo!=bar]</b> has an attribute "foo" that does not equal "bar"</li>
4939 </ul>
4940 <h4>Pseudo Classes:</h4>
4941 <ul class="list">
4942     <li> <b>E:first-child</b> E is the first child of its parent</li>
4943     <li> <b>E:last-child</b> E is the last child of its parent</li>
4944     <li> <b>E:nth-child(<i>n</i>)</b> E is the <i>n</i>th child of its parent (1 based as per the spec)</li>
4945     <li> <b>E:nth-child(odd)</b> E is an odd child of its parent</li>
4946     <li> <b>E:nth-child(even)</b> E is an even child of its parent</li>
4947     <li> <b>E:only-child</b> E is the only child of its parent</li>
4948     <li> <b>E:checked</b> E is an element that is has a checked attribute that is true (e.g. a radio or checkbox) </li>
4949     <li> <b>E:first</b> the first E in the resultset</li>
4950     <li> <b>E:last</b> the last E in the resultset</li>
4951     <li> <b>E:nth(<i>n</i>)</b> the <i>n</i>th E in the resultset (1 based)</li>
4952     <li> <b>E:odd</b> shortcut for :nth-child(odd)</li>
4953     <li> <b>E:even</b> shortcut for :nth-child(even)</li>
4954     <li> <b>E:contains(foo)</b> E's innerHTML contains the substring "foo"</li>
4955     <li> <b>E:nodeValue(foo)</b> E contains a textNode with a nodeValue that equals "foo"</li>
4956     <li> <b>E:not(S)</b> an E element that does not match simple selector S</li>
4957     <li> <b>E:has(S)</b> an E element that has a descendent that matches simple selector S</li>
4958     <li> <b>E:next(S)</b> an E element whose next sibling matches simple selector S</li>
4959     <li> <b>E:prev(S)</b> an E element whose previous sibling matches simple selector S</li>
4960 </ul>
4961 <h4>CSS Value Selectors:</h4>
4962 <ul class="list">
4963     <li> <b>E{display=none}</b> css value "display" that equals "none"</li>
4964     <li> <b>E{display^=none}</b> css value "display" that starts with "none"</li>
4965     <li> <b>E{display$=none}</b> css value "display" that ends with "none"</li>
4966     <li> <b>E{display*=none}</b> css value "display" that contains the substring "none"</li>
4967     <li> <b>E{display%=2}</b> css value "display" that is evenly divisible by 2</li>
4968     <li> <b>E{display!=none}</b> css value "display" that does not equal "none"</li>
4969 </ul>
4970  * @singleton
4971  */
4972 Roo.DomQuery = function(){
4973     var cache = {}, simpleCache = {}, valueCache = {};
4974     var nonSpace = /\S/;
4975     var trimRe = /^\s+|\s+$/g;
4976     var tplRe = /\{(\d+)\}/g;
4977     var modeRe = /^(\s?[\/>+~]\s?|\s|$)/;
4978     var tagTokenRe = /^(#)?([\w-\*]+)/;
4979     var nthRe = /(\d*)n\+?(\d*)/, nthRe2 = /\D/;
4980
4981     function child(p, index){
4982         var i = 0;
4983         var n = p.firstChild;
4984         while(n){
4985             if(n.nodeType == 1){
4986                if(++i == index){
4987                    return n;
4988                }
4989             }
4990             n = n.nextSibling;
4991         }
4992         return null;
4993     };
4994
4995     function next(n){
4996         while((n = n.nextSibling) && n.nodeType != 1);
4997         return n;
4998     };
4999
5000     function prev(n){
5001         while((n = n.previousSibling) && n.nodeType != 1);
5002         return n;
5003     };
5004
5005     function children(d){
5006         var n = d.firstChild, ni = -1;
5007             while(n){
5008                 var nx = n.nextSibling;
5009                 if(n.nodeType == 3 && !nonSpace.test(n.nodeValue)){
5010                     d.removeChild(n);
5011                 }else{
5012                     n.nodeIndex = ++ni;
5013                 }
5014                 n = nx;
5015             }
5016             return this;
5017         };
5018
5019     function byClassName(c, a, v){
5020         if(!v){
5021             return c;
5022         }
5023         var r = [], ri = -1, cn;
5024         for(var i = 0, ci; ci = c[i]; i++){
5025             if((' '+ci.className+' ').indexOf(v) != -1){
5026                 r[++ri] = ci;
5027             }
5028         }
5029         return r;
5030     };
5031
5032     function attrValue(n, attr){
5033         if(!n.tagName && typeof n.length != "undefined"){
5034             n = n[0];
5035         }
5036         if(!n){
5037             return null;
5038         }
5039         if(attr == "for"){
5040             return n.htmlFor;
5041         }
5042         if(attr == "class" || attr == "className"){
5043             return n.className;
5044         }
5045         return n.getAttribute(attr) || n[attr];
5046
5047     };
5048
5049     function getNodes(ns, mode, tagName){
5050         var result = [], ri = -1, cs;
5051         if(!ns){
5052             return result;
5053         }
5054         tagName = tagName || "*";
5055         if(typeof ns.getElementsByTagName != "undefined"){
5056             ns = [ns];
5057         }
5058         if(!mode){
5059             for(var i = 0, ni; ni = ns[i]; i++){
5060                 cs = ni.getElementsByTagName(tagName);
5061                 for(var j = 0, ci; ci = cs[j]; j++){
5062                     result[++ri] = ci;
5063                 }
5064             }
5065         }else if(mode == "/" || mode == ">"){
5066             var utag = tagName.toUpperCase();
5067             for(var i = 0, ni, cn; ni = ns[i]; i++){
5068                 cn = ni.children || ni.childNodes;
5069                 for(var j = 0, cj; cj = cn[j]; j++){
5070                     if(cj.nodeName == utag || cj.nodeName == tagName  || tagName == '*'){
5071                         result[++ri] = cj;
5072                     }
5073                 }
5074             }
5075         }else if(mode == "+"){
5076             var utag = tagName.toUpperCase();
5077             for(var i = 0, n; n = ns[i]; i++){
5078                 while((n = n.nextSibling) && n.nodeType != 1);
5079                 if(n && (n.nodeName == utag || n.nodeName == tagName || tagName == '*')){
5080                     result[++ri] = n;
5081                 }
5082             }
5083         }else if(mode == "~"){
5084             for(var i = 0, n; n = ns[i]; i++){
5085                 while((n = n.nextSibling) && (n.nodeType != 1 || (tagName == '*' || n.tagName.toLowerCase()!=tagName)));
5086                 if(n){
5087                     result[++ri] = n;
5088                 }
5089             }
5090         }
5091         return result;
5092     };
5093
5094     function concat(a, b){
5095         if(b.slice){
5096             return a.concat(b);
5097         }
5098         for(var i = 0, l = b.length; i < l; i++){
5099             a[a.length] = b[i];
5100         }
5101         return a;
5102     }
5103
5104     function byTag(cs, tagName){
5105         if(cs.tagName || cs == document){
5106             cs = [cs];
5107         }
5108         if(!tagName){
5109             return cs;
5110         }
5111         var r = [], ri = -1;
5112         tagName = tagName.toLowerCase();
5113         for(var i = 0, ci; ci = cs[i]; i++){
5114             if(ci.nodeType == 1 && ci.tagName.toLowerCase()==tagName){
5115                 r[++ri] = ci;
5116             }
5117         }
5118         return r;
5119     };
5120
5121     function byId(cs, attr, id){
5122         if(cs.tagName || cs == document){
5123             cs = [cs];
5124         }
5125         if(!id){
5126             return cs;
5127         }
5128         var r = [], ri = -1;
5129         for(var i = 0,ci; ci = cs[i]; i++){
5130             if(ci && ci.id == id){
5131                 r[++ri] = ci;
5132                 return r;
5133             }
5134         }
5135         return r;
5136     };
5137
5138     function byAttribute(cs, attr, value, op, custom){
5139         var r = [], ri = -1, st = custom=="{";
5140         var f = Roo.DomQuery.operators[op];
5141         for(var i = 0, ci; ci = cs[i]; i++){
5142             var a;
5143             if(st){
5144                 a = Roo.DomQuery.getStyle(ci, attr);
5145             }
5146             else if(attr == "class" || attr == "className"){
5147                 a = ci.className;
5148             }else if(attr == "for"){
5149                 a = ci.htmlFor;
5150             }else if(attr == "href"){
5151                 a = ci.getAttribute("href", 2);
5152             }else{
5153                 a = ci.getAttribute(attr);
5154             }
5155             if((f && f(a, value)) || (!f && a)){
5156                 r[++ri] = ci;
5157             }
5158         }
5159         return r;
5160     };
5161
5162     function byPseudo(cs, name, value){
5163         return Roo.DomQuery.pseudos[name](cs, value);
5164     };
5165
5166     // This is for IE MSXML which does not support expandos.
5167     // IE runs the same speed using setAttribute, however FF slows way down
5168     // and Safari completely fails so they need to continue to use expandos.
5169     var isIE = window.ActiveXObject ? true : false;
5170
5171     // this eval is stop the compressor from
5172     // renaming the variable to something shorter
5173     
5174     /** eval:var:batch */
5175     var batch = 30803; 
5176
5177     var key = 30803;
5178
5179     function nodupIEXml(cs){
5180         var d = ++key;
5181         cs[0].setAttribute("_nodup", d);
5182         var r = [cs[0]];
5183         for(var i = 1, len = cs.length; i < len; i++){
5184             var c = cs[i];
5185             if(!c.getAttribute("_nodup") != d){
5186                 c.setAttribute("_nodup", d);
5187                 r[r.length] = c;
5188             }
5189         }
5190         for(var i = 0, len = cs.length; i < len; i++){
5191             cs[i].removeAttribute("_nodup");
5192         }
5193         return r;
5194     }
5195
5196     function nodup(cs){
5197         if(!cs){
5198             return [];
5199         }
5200         var len = cs.length, c, i, r = cs, cj, ri = -1;
5201         if(!len || typeof cs.nodeType != "undefined" || len == 1){
5202             return cs;
5203         }
5204         if(isIE && typeof cs[0].selectSingleNode != "undefined"){
5205             return nodupIEXml(cs);
5206         }
5207         var d = ++key;
5208         cs[0]._nodup = d;
5209         for(i = 1; c = cs[i]; i++){
5210             if(c._nodup != d){
5211                 c._nodup = d;
5212             }else{
5213                 r = [];
5214                 for(var j = 0; j < i; j++){
5215                     r[++ri] = cs[j];
5216                 }
5217                 for(j = i+1; cj = cs[j]; j++){
5218                     if(cj._nodup != d){
5219                         cj._nodup = d;
5220                         r[++ri] = cj;
5221                     }
5222                 }
5223                 return r;
5224             }
5225         }
5226         return r;
5227     }
5228
5229     function quickDiffIEXml(c1, c2){
5230         var d = ++key;
5231         for(var i = 0, len = c1.length; i < len; i++){
5232             c1[i].setAttribute("_qdiff", d);
5233         }
5234         var r = [];
5235         for(var i = 0, len = c2.length; i < len; i++){
5236             if(c2[i].getAttribute("_qdiff") != d){
5237                 r[r.length] = c2[i];
5238             }
5239         }
5240         for(var i = 0, len = c1.length; i < len; i++){
5241            c1[i].removeAttribute("_qdiff");
5242         }
5243         return r;
5244     }
5245
5246     function quickDiff(c1, c2){
5247         var len1 = c1.length;
5248         if(!len1){
5249             return c2;
5250         }
5251         if(isIE && c1[0].selectSingleNode){
5252             return quickDiffIEXml(c1, c2);
5253         }
5254         var d = ++key;
5255         for(var i = 0; i < len1; i++){
5256             c1[i]._qdiff = d;
5257         }
5258         var r = [];
5259         for(var i = 0, len = c2.length; i < len; i++){
5260             if(c2[i]._qdiff != d){
5261                 r[r.length] = c2[i];
5262             }
5263         }
5264         return r;
5265     }
5266
5267     function quickId(ns, mode, root, id){
5268         if(ns == root){
5269            var d = root.ownerDocument || root;
5270            return d.getElementById(id);
5271         }
5272         ns = getNodes(ns, mode, "*");
5273         return byId(ns, null, id);
5274     }
5275
5276     return {
5277         getStyle : function(el, name){
5278             return Roo.fly(el).getStyle(name);
5279         },
5280         /**
5281          * Compiles a selector/xpath query into a reusable function. The returned function
5282          * takes one parameter "root" (optional), which is the context node from where the query should start.
5283          * @param {String} selector The selector/xpath query
5284          * @param {String} type (optional) Either "select" (the default) or "simple" for a simple selector match
5285          * @return {Function}
5286          */
5287         compile : function(path, type){
5288             type = type || "select";
5289             
5290             var fn = ["var f = function(root){\n var mode; ++batch; var n = root || document;\n"];
5291             var q = path, mode, lq;
5292             var tk = Roo.DomQuery.matchers;
5293             var tklen = tk.length;
5294             var mm;
5295
5296             // accept leading mode switch
5297             var lmode = q.match(modeRe);
5298             if(lmode && lmode[1]){
5299                 fn[fn.length] = 'mode="'+lmode[1].replace(trimRe, "")+'";';
5300                 q = q.replace(lmode[1], "");
5301             }
5302             // strip leading slashes
5303             while(path.substr(0, 1)=="/"){
5304                 path = path.substr(1);
5305             }
5306
5307             while(q && lq != q){
5308                 lq = q;
5309                 var tm = q.match(tagTokenRe);
5310                 if(type == "select"){
5311                     if(tm){
5312                         if(tm[1] == "#"){
5313                             fn[fn.length] = 'n = quickId(n, mode, root, "'+tm[2]+'");';
5314                         }else{
5315                             fn[fn.length] = 'n = getNodes(n, mode, "'+tm[2]+'");';
5316                         }
5317                         q = q.replace(tm[0], "");
5318                     }else if(q.substr(0, 1) != '@'){
5319                         fn[fn.length] = 'n = getNodes(n, mode, "*");';
5320                     }
5321                 }else{
5322                     if(tm){
5323                         if(tm[1] == "#"){
5324                             fn[fn.length] = 'n = byId(n, null, "'+tm[2]+'");';
5325                         }else{
5326                             fn[fn.length] = 'n = byTag(n, "'+tm[2]+'");';
5327                         }
5328                         q = q.replace(tm[0], "");
5329                     }
5330                 }
5331                 while(!(mm = q.match(modeRe))){
5332                     var matched = false;
5333                     for(var j = 0; j < tklen; j++){
5334                         var t = tk[j];
5335                         var m = q.match(t.re);
5336                         if(m){
5337                             fn[fn.length] = t.select.replace(tplRe, function(x, i){
5338                                                     return m[i];
5339                                                 });
5340                             q = q.replace(m[0], "");
5341                             matched = true;
5342                             break;
5343                         }
5344                     }
5345                     // prevent infinite loop on bad selector
5346                     if(!matched){
5347                         throw 'Error parsing selector, parsing failed at "' + q + '"';
5348                     }
5349                 }
5350                 if(mm[1]){
5351                     fn[fn.length] = 'mode="'+mm[1].replace(trimRe, "")+'";';
5352                     q = q.replace(mm[1], "");
5353                 }
5354             }
5355             fn[fn.length] = "return nodup(n);\n}";
5356             
5357              /** 
5358               * list of variables that need from compression as they are used by eval.
5359              *  eval:var:batch 
5360              *  eval:var:nodup
5361              *  eval:var:byTag
5362              *  eval:var:ById
5363              *  eval:var:getNodes
5364              *  eval:var:quickId
5365              *  eval:var:mode
5366              *  eval:var:root
5367              *  eval:var:n
5368              *  eval:var:byClassName
5369              *  eval:var:byPseudo
5370              *  eval:var:byAttribute
5371              *  eval:var:attrValue
5372              * 
5373              **/ 
5374             eval(fn.join(""));
5375             return f;
5376         },
5377
5378         /**
5379          * Selects a group of elements.
5380          * @param {String} selector The selector/xpath query (can be a comma separated list of selectors)
5381          * @param {Node} root (optional) The start of the query (defaults to document).
5382          * @return {Array}
5383          */
5384         select : function(path, root, type){
5385             if(!root || root == document){
5386                 root = document;
5387             }
5388             if(typeof root == "string"){
5389                 root = document.getElementById(root);
5390             }
5391             var paths = path.split(",");
5392             var results = [];
5393             for(var i = 0, len = paths.length; i < len; i++){
5394                 var p = paths[i].replace(trimRe, "");
5395                 if(!cache[p]){
5396                     cache[p] = Roo.DomQuery.compile(p);
5397                     if(!cache[p]){
5398                         throw p + " is not a valid selector";
5399                     }
5400                 }
5401                 var result = cache[p](root);
5402                 if(result && result != document){
5403                     results = results.concat(result);
5404                 }
5405             }
5406             if(paths.length > 1){
5407                 return nodup(results);
5408             }
5409             return results;
5410         },
5411
5412         /**
5413          * Selects a single element.
5414          * @param {String} selector The selector/xpath query
5415          * @param {Node} root (optional) The start of the query (defaults to document).
5416          * @return {Element}
5417          */
5418         selectNode : function(path, root){
5419             return Roo.DomQuery.select(path, root)[0];
5420         },
5421
5422         /**
5423          * Selects the value of a node, optionally replacing null with the defaultValue.
5424          * @param {String} selector The selector/xpath query
5425          * @param {Node} root (optional) The start of the query (defaults to document).
5426          * @param {String} defaultValue
5427          */
5428         selectValue : function(path, root, defaultValue){
5429             path = path.replace(trimRe, "");
5430             if(!valueCache[path]){
5431                 valueCache[path] = Roo.DomQuery.compile(path, "select");
5432             }
5433             var n = valueCache[path](root);
5434             n = n[0] ? n[0] : n;
5435             var v = (n && n.firstChild ? n.firstChild.nodeValue : null);
5436             return ((v === null||v === undefined||v==='') ? defaultValue : v);
5437         },
5438
5439         /**
5440          * Selects the value of a node, parsing integers and floats.
5441          * @param {String} selector The selector/xpath query
5442          * @param {Node} root (optional) The start of the query (defaults to document).
5443          * @param {Number} defaultValue
5444          * @return {Number}
5445          */
5446         selectNumber : function(path, root, defaultValue){
5447             var v = Roo.DomQuery.selectValue(path, root, defaultValue || 0);
5448             return parseFloat(v);
5449         },
5450
5451         /**
5452          * Returns true if the passed element(s) match the passed simple selector (e.g. div.some-class or span:first-child)
5453          * @param {String/HTMLElement/Array} el An element id, element or array of elements
5454          * @param {String} selector The simple selector to test
5455          * @return {Boolean}
5456          */
5457         is : function(el, ss){
5458             if(typeof el == "string"){
5459                 el = document.getElementById(el);
5460             }
5461             var isArray = (el instanceof Array);
5462             var result = Roo.DomQuery.filter(isArray ? el : [el], ss);
5463             return isArray ? (result.length == el.length) : (result.length > 0);
5464         },
5465
5466         /**
5467          * Filters an array of elements to only include matches of a simple selector (e.g. div.some-class or span:first-child)
5468          * @param {Array} el An array of elements to filter
5469          * @param {String} selector The simple selector to test
5470          * @param {Boolean} nonMatches If true, it returns the elements that DON'T match
5471          * the selector instead of the ones that match
5472          * @return {Array}
5473          */
5474         filter : function(els, ss, nonMatches){
5475             ss = ss.replace(trimRe, "");
5476             if(!simpleCache[ss]){
5477                 simpleCache[ss] = Roo.DomQuery.compile(ss, "simple");
5478             }
5479             var result = simpleCache[ss](els);
5480             return nonMatches ? quickDiff(result, els) : result;
5481         },
5482
5483         /**
5484          * Collection of matching regular expressions and code snippets.
5485          */
5486         matchers : [{
5487                 re: /^\.([\w-]+)/,
5488                 select: 'n = byClassName(n, null, " {1} ");'
5489             }, {
5490                 re: /^\:([\w-]+)(?:\(((?:[^\s>\/]*|.*?))\))?/,
5491                 select: 'n = byPseudo(n, "{1}", "{2}");'
5492             },{
5493                 re: /^(?:([\[\{])(?:@)?([\w-]+)\s?(?:(=|.=)\s?['"]?(.*?)["']?)?[\]\}])/,
5494                 select: 'n = byAttribute(n, "{2}", "{4}", "{3}", "{1}");'
5495             }, {
5496                 re: /^#([\w-]+)/,
5497                 select: 'n = byId(n, null, "{1}");'
5498             },{
5499                 re: /^@([\w-]+)/,
5500                 select: 'return {firstChild:{nodeValue:attrValue(n, "{1}")}};'
5501             }
5502         ],
5503
5504         /**
5505          * Collection of operator comparison functions. The default operators are =, !=, ^=, $=, *=, %=, |= and ~=.
5506          * New operators can be added as long as the match the format <i>c</i>= where <i>c</i> is any character other than space, &gt; &lt;.
5507          */
5508         operators : {
5509             "=" : function(a, v){
5510                 return a == v;
5511             },
5512             "!=" : function(a, v){
5513                 return a != v;
5514             },
5515             "^=" : function(a, v){
5516                 return a && a.substr(0, v.length) == v;
5517             },
5518             "$=" : function(a, v){
5519                 return a && a.substr(a.length-v.length) == v;
5520             },
5521             "*=" : function(a, v){
5522                 return a && a.indexOf(v) !== -1;
5523             },
5524             "%=" : function(a, v){
5525                 return (a % v) == 0;
5526             },
5527             "|=" : function(a, v){
5528                 return a && (a == v || a.substr(0, v.length+1) == v+'-');
5529             },
5530             "~=" : function(a, v){
5531                 return a && (' '+a+' ').indexOf(' '+v+' ') != -1;
5532             }
5533         },
5534
5535         /**
5536          * Collection of "pseudo class" processors. Each processor is passed the current nodeset (array)
5537          * and the argument (if any) supplied in the selector.
5538          */
5539         pseudos : {
5540             "first-child" : function(c){
5541                 var r = [], ri = -1, n;
5542                 for(var i = 0, ci; ci = n = c[i]; i++){
5543                     while((n = n.previousSibling) && n.nodeType != 1);
5544                     if(!n){
5545                         r[++ri] = ci;
5546                     }
5547                 }
5548                 return r;
5549             },
5550
5551             "last-child" : function(c){
5552                 var r = [], ri = -1, n;
5553                 for(var i = 0, ci; ci = n = c[i]; i++){
5554                     while((n = n.nextSibling) && n.nodeType != 1);
5555                     if(!n){
5556                         r[++ri] = ci;
5557                     }
5558                 }
5559                 return r;
5560             },
5561
5562             "nth-child" : function(c, a) {
5563                 var r = [], ri = -1;
5564                 var m = nthRe.exec(a == "even" && "2n" || a == "odd" && "2n+1" || !nthRe2.test(a) && "n+" + a || a);
5565                 var f = (m[1] || 1) - 0, l = m[2] - 0;
5566                 for(var i = 0, n; n = c[i]; i++){
5567                     var pn = n.parentNode;
5568                     if (batch != pn._batch) {
5569                         var j = 0;
5570                         for(var cn = pn.firstChild; cn; cn = cn.nextSibling){
5571                             if(cn.nodeType == 1){
5572                                cn.nodeIndex = ++j;
5573                             }
5574                         }
5575                         pn._batch = batch;
5576                     }
5577                     if (f == 1) {
5578                         if (l == 0 || n.nodeIndex == l){
5579                             r[++ri] = n;
5580                         }
5581                     } else if ((n.nodeIndex + l) % f == 0){
5582                         r[++ri] = n;
5583                     }
5584                 }
5585
5586                 return r;
5587             },
5588
5589             "only-child" : function(c){
5590                 var r = [], ri = -1;;
5591                 for(var i = 0, ci; ci = c[i]; i++){
5592                     if(!prev(ci) && !next(ci)){
5593                         r[++ri] = ci;
5594                     }
5595                 }
5596                 return r;
5597             },
5598
5599             "empty" : function(c){
5600                 var r = [], ri = -1;
5601                 for(var i = 0, ci; ci = c[i]; i++){
5602                     var cns = ci.childNodes, j = 0, cn, empty = true;
5603                     while(cn = cns[j]){
5604                         ++j;
5605                         if(cn.nodeType == 1 || cn.nodeType == 3){
5606                             empty = false;
5607                             break;
5608                         }
5609                     }
5610                     if(empty){
5611                         r[++ri] = ci;
5612                     }
5613                 }
5614                 return r;
5615             },
5616
5617             "contains" : function(c, v){
5618                 var r = [], ri = -1;
5619                 for(var i = 0, ci; ci = c[i]; i++){
5620                     if((ci.textContent||ci.innerText||'').indexOf(v) != -1){
5621                         r[++ri] = ci;
5622                     }
5623                 }
5624                 return r;
5625             },
5626
5627             "nodeValue" : function(c, v){
5628                 var r = [], ri = -1;
5629                 for(var i = 0, ci; ci = c[i]; i++){
5630                     if(ci.firstChild && ci.firstChild.nodeValue == v){
5631                         r[++ri] = ci;
5632                     }
5633                 }
5634                 return r;
5635             },
5636
5637             "checked" : function(c){
5638                 var r = [], ri = -1;
5639                 for(var i = 0, ci; ci = c[i]; i++){
5640                     if(ci.checked == true){
5641                         r[++ri] = ci;
5642                     }
5643                 }
5644                 return r;
5645             },
5646
5647             "not" : function(c, ss){
5648                 return Roo.DomQuery.filter(c, ss, true);
5649             },
5650
5651             "odd" : function(c){
5652                 return this["nth-child"](c, "odd");
5653             },
5654
5655             "even" : function(c){
5656                 return this["nth-child"](c, "even");
5657             },
5658
5659             "nth" : function(c, a){
5660                 return c[a-1] || [];
5661             },
5662
5663             "first" : function(c){
5664                 return c[0] || [];
5665             },
5666
5667             "last" : function(c){
5668                 return c[c.length-1] || [];
5669             },
5670
5671             "has" : function(c, ss){
5672                 var s = Roo.DomQuery.select;
5673                 var r = [], ri = -1;
5674                 for(var i = 0, ci; ci = c[i]; i++){
5675                     if(s(ss, ci).length > 0){
5676                         r[++ri] = ci;
5677                     }
5678                 }
5679                 return r;
5680             },
5681
5682             "next" : function(c, ss){
5683                 var is = Roo.DomQuery.is;
5684                 var r = [], ri = -1;
5685                 for(var i = 0, ci; ci = c[i]; i++){
5686                     var n = next(ci);
5687                     if(n && is(n, ss)){
5688                         r[++ri] = ci;
5689                     }
5690                 }
5691                 return r;
5692             },
5693
5694             "prev" : function(c, ss){
5695                 var is = Roo.DomQuery.is;
5696                 var r = [], ri = -1;
5697                 for(var i = 0, ci; ci = c[i]; i++){
5698                     var n = prev(ci);
5699                     if(n && is(n, ss)){
5700                         r[++ri] = ci;
5701                     }
5702                 }
5703                 return r;
5704             }
5705         }
5706     };
5707 }();
5708
5709 /**
5710  * Selects an array of DOM nodes by CSS/XPath selector. Shorthand of {@link Roo.DomQuery#select}
5711  * @param {String} path The selector/xpath query
5712  * @param {Node} root (optional) The start of the query (defaults to document).
5713  * @return {Array}
5714  * @member Roo
5715  * @method query
5716  */
5717 Roo.query = Roo.DomQuery.select;
5718 /*
5719  * Based on:
5720  * Ext JS Library 1.1.1
5721  * Copyright(c) 2006-2007, Ext JS, LLC.
5722  *
5723  * Originally Released Under LGPL - original licence link has changed is not relivant.
5724  *
5725  * Fork - LGPL
5726  * <script type="text/javascript">
5727  */
5728
5729 /**
5730  * @class Roo.util.Observable
5731  * Base class that provides a common interface for publishing events. Subclasses are expected to
5732  * to have a property "events" with all the events defined.<br>
5733  * For example:
5734  * <pre><code>
5735  Employee = function(name){
5736     this.name = name;
5737     this.addEvents({
5738         "fired" : true,
5739         "quit" : true
5740     });
5741  }
5742  Roo.extend(Employee, Roo.util.Observable);
5743 </code></pre>
5744  * @param {Object} config properties to use (incuding events / listeners)
5745  */
5746
5747 Roo.util.Observable = function(cfg){
5748     
5749     cfg = cfg|| {};
5750     this.addEvents(cfg.events || {});
5751     if (cfg.events) {
5752         delete cfg.events; // make sure
5753     }
5754      
5755     Roo.apply(this, cfg);
5756     
5757     if(this.listeners){
5758         this.on(this.listeners);
5759         delete this.listeners;
5760     }
5761 };
5762 Roo.util.Observable.prototype = {
5763     /** 
5764  * @cfg {Object} listeners  list of events and functions to call for this object, 
5765  * For example :
5766  * <pre><code>
5767     listeners :  { 
5768        'click' : function(e) {
5769            ..... 
5770         } ,
5771         .... 
5772     } 
5773   </code></pre>
5774  */
5775     
5776     
5777     /**
5778      * Fires the specified event with the passed parameters (minus the event name).
5779      * @param {String} eventName
5780      * @param {Object...} args Variable number of parameters are passed to handlers
5781      * @return {Boolean} returns false if any of the handlers return false otherwise it returns true
5782      */
5783     fireEvent : function(){
5784         var ce = this.events[arguments[0].toLowerCase()];
5785         if(typeof ce == "object"){
5786             return ce.fire.apply(ce, Array.prototype.slice.call(arguments, 1));
5787         }else{
5788             return true;
5789         }
5790     },
5791
5792     // private
5793     filterOptRe : /^(?:scope|delay|buffer|single)$/,
5794
5795     /**
5796      * Appends an event handler to this component
5797      * @param {String}   eventName The type of event to listen for
5798      * @param {Function} handler The method the event invokes
5799      * @param {Object}   scope (optional) The scope in which to execute the handler
5800      * function. The handler function's "this" context.
5801      * @param {Object}   options (optional) An object containing handler configuration
5802      * properties. This may contain any of the following properties:<ul>
5803      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
5804      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
5805      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
5806      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
5807      * by the specified number of milliseconds. If the event fires again within that time, the original
5808      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
5809      * </ul><br>
5810      * <p>
5811      * <b>Combining Options</b><br>
5812      * Using the options argument, it is possible to combine different types of listeners:<br>
5813      * <br>
5814      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)
5815                 <pre><code>
5816                 el.on('click', this.onClick, this, {
5817                         single: true,
5818                 delay: 100,
5819                 forumId: 4
5820                 });
5821                 </code></pre>
5822      * <p>
5823      * <b>Attaching multiple handlers in 1 call</b><br>
5824      * The method also allows for a single argument to be passed which is a config object containing properties
5825      * which specify multiple handlers.
5826      * <pre><code>
5827                 el.on({
5828                         'click': {
5829                         fn: this.onClick,
5830                         scope: this,
5831                         delay: 100
5832                 }, 
5833                 'mouseover': {
5834                         fn: this.onMouseOver,
5835                         scope: this
5836                 },
5837                 'mouseout': {
5838                         fn: this.onMouseOut,
5839                         scope: this
5840                 }
5841                 });
5842                 </code></pre>
5843      * <p>
5844      * Or a shorthand syntax which passes the same scope object to all handlers:
5845         <pre><code>
5846                 el.on({
5847                         'click': this.onClick,
5848                 'mouseover': this.onMouseOver,
5849                 'mouseout': this.onMouseOut,
5850                 scope: this
5851                 });
5852                 </code></pre>
5853      */
5854     addListener : function(eventName, fn, scope, o){
5855         if(typeof eventName == "object"){
5856             o = eventName;
5857             for(var e in o){
5858                 if(this.filterOptRe.test(e)){
5859                     continue;
5860                 }
5861                 if(typeof o[e] == "function"){
5862                     // shared options
5863                     this.addListener(e, o[e], o.scope,  o);
5864                 }else{
5865                     // individual options
5866                     this.addListener(e, o[e].fn, o[e].scope, o[e]);
5867                 }
5868             }
5869             return;
5870         }
5871         o = (!o || typeof o == "boolean") ? {} : o;
5872         eventName = eventName.toLowerCase();
5873         var ce = this.events[eventName] || true;
5874         if(typeof ce == "boolean"){
5875             ce = new Roo.util.Event(this, eventName);
5876             this.events[eventName] = ce;
5877         }
5878         ce.addListener(fn, scope, o);
5879     },
5880
5881     /**
5882      * Removes a listener
5883      * @param {String}   eventName     The type of event to listen for
5884      * @param {Function} handler        The handler to remove
5885      * @param {Object}   scope  (optional) The scope (this object) for the handler
5886      */
5887     removeListener : function(eventName, fn, scope){
5888         var ce = this.events[eventName.toLowerCase()];
5889         if(typeof ce == "object"){
5890             ce.removeListener(fn, scope);
5891         }
5892     },
5893
5894     /**
5895      * Removes all listeners for this object
5896      */
5897     purgeListeners : function(){
5898         for(var evt in this.events){
5899             if(typeof this.events[evt] == "object"){
5900                  this.events[evt].clearListeners();
5901             }
5902         }
5903     },
5904
5905     relayEvents : function(o, events){
5906         var createHandler = function(ename){
5907             return function(){
5908                 return this.fireEvent.apply(this, Roo.combine(ename, Array.prototype.slice.call(arguments, 0)));
5909             };
5910         };
5911         for(var i = 0, len = events.length; i < len; i++){
5912             var ename = events[i];
5913             if(!this.events[ename]){ this.events[ename] = true; };
5914             o.on(ename, createHandler(ename), this);
5915         }
5916     },
5917
5918     /**
5919      * Used to define events on this Observable
5920      * @param {Object} object The object with the events defined
5921      */
5922     addEvents : function(o){
5923         if(!this.events){
5924             this.events = {};
5925         }
5926         Roo.applyIf(this.events, o);
5927     },
5928
5929     /**
5930      * Checks to see if this object has any listeners for a specified event
5931      * @param {String} eventName The name of the event to check for
5932      * @return {Boolean} True if the event is being listened for, else false
5933      */
5934     hasListener : function(eventName){
5935         var e = this.events[eventName];
5936         return typeof e == "object" && e.listeners.length > 0;
5937     }
5938 };
5939 /**
5940  * Appends an event handler to this element (shorthand for addListener)
5941  * @param {String}   eventName     The type of event to listen for
5942  * @param {Function} handler        The method the event invokes
5943  * @param {Object}   scope (optional) The scope in which to execute the handler
5944  * function. The handler function's "this" context.
5945  * @param {Object}   options  (optional)
5946  * @method
5947  */
5948 Roo.util.Observable.prototype.on = Roo.util.Observable.prototype.addListener;
5949 /**
5950  * Removes a listener (shorthand for removeListener)
5951  * @param {String}   eventName     The type of event to listen for
5952  * @param {Function} handler        The handler to remove
5953  * @param {Object}   scope  (optional) The scope (this object) for the handler
5954  * @method
5955  */
5956 Roo.util.Observable.prototype.un = Roo.util.Observable.prototype.removeListener;
5957
5958 /**
5959  * Starts capture on the specified Observable. All events will be passed
5960  * to the supplied function with the event name + standard signature of the event
5961  * <b>before</b> the event is fired. If the supplied function returns false,
5962  * the event will not fire.
5963  * @param {Observable} o The Observable to capture
5964  * @param {Function} fn The function to call
5965  * @param {Object} scope (optional) The scope (this object) for the fn
5966  * @static
5967  */
5968 Roo.util.Observable.capture = function(o, fn, scope){
5969     o.fireEvent = o.fireEvent.createInterceptor(fn, scope);
5970 };
5971
5972 /**
5973  * Removes <b>all</b> added captures from the Observable.
5974  * @param {Observable} o The Observable to release
5975  * @static
5976  */
5977 Roo.util.Observable.releaseCapture = function(o){
5978     o.fireEvent = Roo.util.Observable.prototype.fireEvent;
5979 };
5980
5981 (function(){
5982
5983     var createBuffered = function(h, o, scope){
5984         var task = new Roo.util.DelayedTask();
5985         return function(){
5986             task.delay(o.buffer, h, scope, Array.prototype.slice.call(arguments, 0));
5987         };
5988     };
5989
5990     var createSingle = function(h, e, fn, scope){
5991         return function(){
5992             e.removeListener(fn, scope);
5993             return h.apply(scope, arguments);
5994         };
5995     };
5996
5997     var createDelayed = function(h, o, scope){
5998         return function(){
5999             var args = Array.prototype.slice.call(arguments, 0);
6000             setTimeout(function(){
6001                 h.apply(scope, args);
6002             }, o.delay || 10);
6003         };
6004     };
6005
6006     Roo.util.Event = function(obj, name){
6007         this.name = name;
6008         this.obj = obj;
6009         this.listeners = [];
6010     };
6011
6012     Roo.util.Event.prototype = {
6013         addListener : function(fn, scope, options){
6014             var o = options || {};
6015             scope = scope || this.obj;
6016             if(!this.isListening(fn, scope)){
6017                 var l = {fn: fn, scope: scope, options: o};
6018                 var h = fn;
6019                 if(o.delay){
6020                     h = createDelayed(h, o, scope);
6021                 }
6022                 if(o.single){
6023                     h = createSingle(h, this, fn, scope);
6024                 }
6025                 if(o.buffer){
6026                     h = createBuffered(h, o, scope);
6027                 }
6028                 l.fireFn = h;
6029                 if(!this.firing){ // if we are currently firing this event, don't disturb the listener loop
6030                     this.listeners.push(l);
6031                 }else{
6032                     this.listeners = this.listeners.slice(0);
6033                     this.listeners.push(l);
6034                 }
6035             }
6036         },
6037
6038         findListener : function(fn, scope){
6039             scope = scope || this.obj;
6040             var ls = this.listeners;
6041             for(var i = 0, len = ls.length; i < len; i++){
6042                 var l = ls[i];
6043                 if(l.fn == fn && l.scope == scope){
6044                     return i;
6045                 }
6046             }
6047             return -1;
6048         },
6049
6050         isListening : function(fn, scope){
6051             return this.findListener(fn, scope) != -1;
6052         },
6053
6054         removeListener : function(fn, scope){
6055             var index;
6056             if((index = this.findListener(fn, scope)) != -1){
6057                 if(!this.firing){
6058                     this.listeners.splice(index, 1);
6059                 }else{
6060                     this.listeners = this.listeners.slice(0);
6061                     this.listeners.splice(index, 1);
6062                 }
6063                 return true;
6064             }
6065             return false;
6066         },
6067
6068         clearListeners : function(){
6069             this.listeners = [];
6070         },
6071
6072         fire : function(){
6073             var ls = this.listeners, scope, len = ls.length;
6074             if(len > 0){
6075                 this.firing = true;
6076                 var args = Array.prototype.slice.call(arguments, 0);
6077                 for(var i = 0; i < len; i++){
6078                     var l = ls[i];
6079                     if(l.fireFn.apply(l.scope||this.obj||window, arguments) === false){
6080                         this.firing = false;
6081                         return false;
6082                     }
6083                 }
6084                 this.firing = false;
6085             }
6086             return true;
6087         }
6088     };
6089 })();/*
6090  * RooJS Library 
6091  * Copyright(c) 2007-2017, Roo J Solutions Ltd
6092  *
6093  * Licence LGPL 
6094  *
6095  */
6096  
6097 /**
6098  * @class Roo.Document
6099  * @extends Roo.util.Observable
6100  * This is a convience class to wrap up the main document loading code.. , rather than adding Roo.onReady(......)
6101  * 
6102  * @param {Object} config the methods and properties of the 'base' class for the application.
6103  * 
6104  *  Generic Page handler - implement this to start your app..
6105  * 
6106  * eg.
6107  *  MyProject = new Roo.Document({
6108         events : {
6109             'load' : true // your events..
6110         },
6111         listeners : {
6112             'ready' : function() {
6113                 // fired on Roo.onReady()
6114             }
6115         }
6116  * 
6117  */
6118 Roo.Document = function(cfg) {
6119      
6120     this.addEvents({ 
6121         'ready' : true
6122     });
6123     Roo.util.Observable.call(this,cfg);
6124     
6125     var _this = this;
6126     
6127     Roo.onReady(function() {
6128         _this.fireEvent('ready');
6129     },null,false);
6130     
6131     
6132 }
6133
6134 Roo.extend(Roo.Document, Roo.util.Observable, {});/*
6135  * Based on:
6136  * Ext JS Library 1.1.1
6137  * Copyright(c) 2006-2007, Ext JS, LLC.
6138  *
6139  * Originally Released Under LGPL - original licence link has changed is not relivant.
6140  *
6141  * Fork - LGPL
6142  * <script type="text/javascript">
6143  */
6144
6145 /**
6146  * @class Roo.EventManager
6147  * Registers event handlers that want to receive a normalized EventObject instead of the standard browser event and provides 
6148  * several useful events directly.
6149  * See {@link Roo.EventObject} for more details on normalized event objects.
6150  * @singleton
6151  */
6152 Roo.EventManager = function(){
6153     var docReadyEvent, docReadyProcId, docReadyState = false;
6154     var resizeEvent, resizeTask, textEvent, textSize;
6155     var E = Roo.lib.Event;
6156     var D = Roo.lib.Dom;
6157
6158     
6159     
6160
6161     var fireDocReady = function(){
6162         if(!docReadyState){
6163             docReadyState = true;
6164             Roo.isReady = true;
6165             if(docReadyProcId){
6166                 clearInterval(docReadyProcId);
6167             }
6168             if(Roo.isGecko || Roo.isOpera) {
6169                 document.removeEventListener("DOMContentLoaded", fireDocReady, false);
6170             }
6171             if(Roo.isIE){
6172                 var defer = document.getElementById("ie-deferred-loader");
6173                 if(defer){
6174                     defer.onreadystatechange = null;
6175                     defer.parentNode.removeChild(defer);
6176                 }
6177             }
6178             if(docReadyEvent){
6179                 docReadyEvent.fire();
6180                 docReadyEvent.clearListeners();
6181             }
6182         }
6183     };
6184     
6185     var initDocReady = function(){
6186         docReadyEvent = new Roo.util.Event();
6187         if(Roo.isGecko || Roo.isOpera) {
6188             document.addEventListener("DOMContentLoaded", fireDocReady, false);
6189         }else if(Roo.isIE){
6190             document.write("<s"+'cript id="ie-deferred-loader" defer="defer" src="/'+'/:"></s'+"cript>");
6191             var defer = document.getElementById("ie-deferred-loader");
6192             defer.onreadystatechange = function(){
6193                 if(this.readyState == "complete"){
6194                     fireDocReady();
6195                 }
6196             };
6197         }else if(Roo.isSafari){ 
6198             docReadyProcId = setInterval(function(){
6199                 var rs = document.readyState;
6200                 if(rs == "complete") {
6201                     fireDocReady();     
6202                  }
6203             }, 10);
6204         }
6205         // no matter what, make sure it fires on load
6206         E.on(window, "load", fireDocReady);
6207     };
6208
6209     var createBuffered = function(h, o){
6210         var task = new Roo.util.DelayedTask(h);
6211         return function(e){
6212             // create new event object impl so new events don't wipe out properties
6213             e = new Roo.EventObjectImpl(e);
6214             task.delay(o.buffer, h, null, [e]);
6215         };
6216     };
6217
6218     var createSingle = function(h, el, ename, fn){
6219         return function(e){
6220             Roo.EventManager.removeListener(el, ename, fn);
6221             h(e);
6222         };
6223     };
6224
6225     var createDelayed = function(h, o){
6226         return function(e){
6227             // create new event object impl so new events don't wipe out properties
6228             e = new Roo.EventObjectImpl(e);
6229             setTimeout(function(){
6230                 h(e);
6231             }, o.delay || 10);
6232         };
6233     };
6234     var transitionEndVal = false;
6235     
6236     var transitionEnd = function()
6237     {
6238         if (transitionEndVal) {
6239             return transitionEndVal;
6240         }
6241         var el = document.createElement('div');
6242
6243         var transEndEventNames = {
6244             WebkitTransition : 'webkitTransitionEnd',
6245             MozTransition    : 'transitionend',
6246             OTransition      : 'oTransitionEnd otransitionend',
6247             transition       : 'transitionend'
6248         };
6249     
6250         for (var name in transEndEventNames) {
6251             if (el.style[name] !== undefined) {
6252                 transitionEndVal = transEndEventNames[name];
6253                 return  transitionEndVal ;
6254             }
6255         }
6256     }
6257     
6258
6259     var listen = function(element, ename, opt, fn, scope){
6260         var o = (!opt || typeof opt == "boolean") ? {} : opt;
6261         fn = fn || o.fn; scope = scope || o.scope;
6262         var el = Roo.getDom(element);
6263         
6264         
6265         if(!el){
6266             throw "Error listening for \"" + ename + '\". Element "' + element + '" doesn\'t exist.';
6267         }
6268         
6269         if (ename == 'transitionend') {
6270             ename = transitionEnd();
6271         }
6272         var h = function(e){
6273             e = Roo.EventObject.setEvent(e);
6274             var t;
6275             if(o.delegate){
6276                 t = e.getTarget(o.delegate, el);
6277                 if(!t){
6278                     return;
6279                 }
6280             }else{
6281                 t = e.target;
6282             }
6283             if(o.stopEvent === true){
6284                 e.stopEvent();
6285             }
6286             if(o.preventDefault === true){
6287                e.preventDefault();
6288             }
6289             if(o.stopPropagation === true){
6290                 e.stopPropagation();
6291             }
6292
6293             if(o.normalized === false){
6294                 e = e.browserEvent;
6295             }
6296
6297             fn.call(scope || el, e, t, o);
6298         };
6299         if(o.delay){
6300             h = createDelayed(h, o);
6301         }
6302         if(o.single){
6303             h = createSingle(h, el, ename, fn);
6304         }
6305         if(o.buffer){
6306             h = createBuffered(h, o);
6307         }
6308         
6309         fn._handlers = fn._handlers || [];
6310         
6311         
6312         fn._handlers.push([Roo.id(el), ename, h]);
6313         
6314         
6315          
6316         E.on(el, ename, h);
6317         if(ename == "mousewheel" && el.addEventListener){ // workaround for jQuery
6318             el.addEventListener("DOMMouseScroll", h, false);
6319             E.on(window, 'unload', function(){
6320                 el.removeEventListener("DOMMouseScroll", h, false);
6321             });
6322         }
6323         if(ename == "mousedown" && el == document){ // fix stopped mousedowns on the document
6324             Roo.EventManager.stoppedMouseDownEvent.addListener(h);
6325         }
6326         return h;
6327     };
6328
6329     var stopListening = function(el, ename, fn){
6330         var id = Roo.id(el), hds = fn._handlers, hd = fn;
6331         if(hds){
6332             for(var i = 0, len = hds.length; i < len; i++){
6333                 var h = hds[i];
6334                 if(h[0] == id && h[1] == ename){
6335                     hd = h[2];
6336                     hds.splice(i, 1);
6337                     break;
6338                 }
6339             }
6340         }
6341         E.un(el, ename, hd);
6342         el = Roo.getDom(el);
6343         if(ename == "mousewheel" && el.addEventListener){
6344             el.removeEventListener("DOMMouseScroll", hd, false);
6345         }
6346         if(ename == "mousedown" && el == document){ // fix stopped mousedowns on the document
6347             Roo.EventManager.stoppedMouseDownEvent.removeListener(hd);
6348         }
6349     };
6350
6351     var propRe = /^(?:scope|delay|buffer|single|stopEvent|preventDefault|stopPropagation|normalized|args|delegate)$/;
6352     
6353     var pub = {
6354         
6355         
6356         /** 
6357          * Fix for doc tools
6358          * @scope Roo.EventManager
6359          */
6360         
6361         
6362         /** 
6363          * This is no longer needed and is deprecated. Places a simple wrapper around an event handler to override the browser event
6364          * object with a Roo.EventObject
6365          * @param {Function} fn        The method the event invokes
6366          * @param {Object}   scope    An object that becomes the scope of the handler
6367          * @param {boolean}  override If true, the obj passed in becomes
6368          *                             the execution scope of the listener
6369          * @return {Function} The wrapped function
6370          * @deprecated
6371          */
6372         wrap : function(fn, scope, override){
6373             return function(e){
6374                 Roo.EventObject.setEvent(e);
6375                 fn.call(override ? scope || window : window, Roo.EventObject, scope);
6376             };
6377         },
6378         
6379         /**
6380      * Appends an event handler to an element (shorthand for addListener)
6381      * @param {String/HTMLElement}   element        The html element or id to assign the
6382      * @param {String}   eventName The type of event to listen for
6383      * @param {Function} handler The method the event invokes
6384      * @param {Object}   scope (optional) The scope in which to execute the handler
6385      * function. The handler function's "this" context.
6386      * @param {Object}   options (optional) An object containing handler configuration
6387      * properties. This may contain any of the following properties:<ul>
6388      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
6389      * <li>delegate {String} A simple selector to filter the target or look for a descendant of the target</li>
6390      * <li>stopEvent {Boolean} True to stop the event. That is stop propagation, and prevent the default action.</li>
6391      * <li>preventDefault {Boolean} True to prevent the default action</li>
6392      * <li>stopPropagation {Boolean} True to prevent event propagation</li>
6393      * <li>normalized {Boolean} False to pass a browser event to the handler function instead of an Roo.EventObject</li>
6394      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
6395      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
6396      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
6397      * by the specified number of milliseconds. If the event fires again within that time, the original
6398      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
6399      * </ul><br>
6400      * <p>
6401      * <b>Combining Options</b><br>
6402      * Using the options argument, it is possible to combine different types of listeners:<br>
6403      * <br>
6404      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)<div style="margin: 5px 20px 20px;">
6405      * Code:<pre><code>
6406 el.on('click', this.onClick, this, {
6407     single: true,
6408     delay: 100,
6409     stopEvent : true,
6410     forumId: 4
6411 });</code></pre>
6412      * <p>
6413      * <b>Attaching multiple handlers in 1 call</b><br>
6414       * The method also allows for a single argument to be passed which is a config object containing properties
6415      * which specify multiple handlers.
6416      * <p>
6417      * Code:<pre><code>
6418 el.on({
6419     'click' : {
6420         fn: this.onClick
6421         scope: this,
6422         delay: 100
6423     },
6424     'mouseover' : {
6425         fn: this.onMouseOver
6426         scope: this
6427     },
6428     'mouseout' : {
6429         fn: this.onMouseOut
6430         scope: this
6431     }
6432 });</code></pre>
6433      * <p>
6434      * Or a shorthand syntax:<br>
6435      * Code:<pre><code>
6436 el.on({
6437     'click' : this.onClick,
6438     'mouseover' : this.onMouseOver,
6439     'mouseout' : this.onMouseOut
6440     scope: this
6441 });</code></pre>
6442      */
6443         addListener : function(element, eventName, fn, scope, options){
6444             if(typeof eventName == "object"){
6445                 var o = eventName;
6446                 for(var e in o){
6447                     if(propRe.test(e)){
6448                         continue;
6449                     }
6450                     if(typeof o[e] == "function"){
6451                         // shared options
6452                         listen(element, e, o, o[e], o.scope);
6453                     }else{
6454                         // individual options
6455                         listen(element, e, o[e]);
6456                     }
6457                 }
6458                 return;
6459             }
6460             return listen(element, eventName, options, fn, scope);
6461         },
6462         
6463         /**
6464          * Removes an event handler
6465          *
6466          * @param {String/HTMLElement}   element        The id or html element to remove the 
6467          *                             event from
6468          * @param {String}   eventName     The type of event
6469          * @param {Function} fn
6470          * @return {Boolean} True if a listener was actually removed
6471          */
6472         removeListener : function(element, eventName, fn){
6473             return stopListening(element, eventName, fn);
6474         },
6475         
6476         /**
6477          * Fires when the document is ready (before onload and before images are loaded). Can be 
6478          * accessed shorthanded Roo.onReady().
6479          * @param {Function} fn        The method the event invokes
6480          * @param {Object}   scope    An  object that becomes the scope of the handler
6481          * @param {boolean}  options
6482          */
6483         onDocumentReady : function(fn, scope, options){
6484             if(docReadyState){ // if it already fired
6485                 docReadyEvent.addListener(fn, scope, options);
6486                 docReadyEvent.fire();
6487                 docReadyEvent.clearListeners();
6488                 return;
6489             }
6490             if(!docReadyEvent){
6491                 initDocReady();
6492             }
6493             docReadyEvent.addListener(fn, scope, options);
6494         },
6495         
6496         /**
6497          * Fires when the window is resized and provides resize event buffering (50 milliseconds), passes new viewport width and height to handlers.
6498          * @param {Function} fn        The method the event invokes
6499          * @param {Object}   scope    An object that becomes the scope of the handler
6500          * @param {boolean}  options
6501          */
6502         onWindowResize : function(fn, scope, options){
6503             if(!resizeEvent){
6504                 resizeEvent = new Roo.util.Event();
6505                 resizeTask = new Roo.util.DelayedTask(function(){
6506                     resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6507                 });
6508                 E.on(window, "resize", function(){
6509                     if(Roo.isIE){
6510                         resizeTask.delay(50);
6511                     }else{
6512                         resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6513                     }
6514                 });
6515             }
6516             resizeEvent.addListener(fn, scope, options);
6517         },
6518
6519         /**
6520          * Fires when the user changes the active text size. Handler gets called with 2 params, the old size and the new size.
6521          * @param {Function} fn        The method the event invokes
6522          * @param {Object}   scope    An object that becomes the scope of the handler
6523          * @param {boolean}  options
6524          */
6525         onTextResize : function(fn, scope, options){
6526             if(!textEvent){
6527                 textEvent = new Roo.util.Event();
6528                 var textEl = new Roo.Element(document.createElement('div'));
6529                 textEl.dom.className = 'x-text-resize';
6530                 textEl.dom.innerHTML = 'X';
6531                 textEl.appendTo(document.body);
6532                 textSize = textEl.dom.offsetHeight;
6533                 setInterval(function(){
6534                     if(textEl.dom.offsetHeight != textSize){
6535                         textEvent.fire(textSize, textSize = textEl.dom.offsetHeight);
6536                     }
6537                 }, this.textResizeInterval);
6538             }
6539             textEvent.addListener(fn, scope, options);
6540         },
6541
6542         /**
6543          * Removes the passed window resize listener.
6544          * @param {Function} fn        The method the event invokes
6545          * @param {Object}   scope    The scope of handler
6546          */
6547         removeResizeListener : function(fn, scope){
6548             if(resizeEvent){
6549                 resizeEvent.removeListener(fn, scope);
6550             }
6551         },
6552
6553         // private
6554         fireResize : function(){
6555             if(resizeEvent){
6556                 resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6557             }   
6558         },
6559         /**
6560          * Url used for onDocumentReady with using SSL (defaults to Roo.SSL_SECURE_URL)
6561          */
6562         ieDeferSrc : false,
6563         /**
6564          * The frequency, in milliseconds, to check for text resize events (defaults to 50)
6565          */
6566         textResizeInterval : 50
6567     };
6568     
6569     /**
6570      * Fix for doc tools
6571      * @scopeAlias pub=Roo.EventManager
6572      */
6573     
6574      /**
6575      * Appends an event handler to an element (shorthand for addListener)
6576      * @param {String/HTMLElement}   element        The html element or id to assign the
6577      * @param {String}   eventName The type of event to listen for
6578      * @param {Function} handler The method the event invokes
6579      * @param {Object}   scope (optional) The scope in which to execute the handler
6580      * function. The handler function's "this" context.
6581      * @param {Object}   options (optional) An object containing handler configuration
6582      * properties. This may contain any of the following properties:<ul>
6583      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
6584      * <li>delegate {String} A simple selector to filter the target or look for a descendant of the target</li>
6585      * <li>stopEvent {Boolean} True to stop the event. That is stop propagation, and prevent the default action.</li>
6586      * <li>preventDefault {Boolean} True to prevent the default action</li>
6587      * <li>stopPropagation {Boolean} True to prevent event propagation</li>
6588      * <li>normalized {Boolean} False to pass a browser event to the handler function instead of an Roo.EventObject</li>
6589      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
6590      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
6591      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
6592      * by the specified number of milliseconds. If the event fires again within that time, the original
6593      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
6594      * </ul><br>
6595      * <p>
6596      * <b>Combining Options</b><br>
6597      * Using the options argument, it is possible to combine different types of listeners:<br>
6598      * <br>
6599      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)<div style="margin: 5px 20px 20px;">
6600      * Code:<pre><code>
6601 el.on('click', this.onClick, this, {
6602     single: true,
6603     delay: 100,
6604     stopEvent : true,
6605     forumId: 4
6606 });</code></pre>
6607      * <p>
6608      * <b>Attaching multiple handlers in 1 call</b><br>
6609       * The method also allows for a single argument to be passed which is a config object containing properties
6610      * which specify multiple handlers.
6611      * <p>
6612      * Code:<pre><code>
6613 el.on({
6614     'click' : {
6615         fn: this.onClick
6616         scope: this,
6617         delay: 100
6618     },
6619     'mouseover' : {
6620         fn: this.onMouseOver
6621         scope: this
6622     },
6623     'mouseout' : {
6624         fn: this.onMouseOut
6625         scope: this
6626     }
6627 });</code></pre>
6628      * <p>
6629      * Or a shorthand syntax:<br>
6630      * Code:<pre><code>
6631 el.on({
6632     'click' : this.onClick,
6633     'mouseover' : this.onMouseOver,
6634     'mouseout' : this.onMouseOut
6635     scope: this
6636 });</code></pre>
6637      */
6638     pub.on = pub.addListener;
6639     pub.un = pub.removeListener;
6640
6641     pub.stoppedMouseDownEvent = new Roo.util.Event();
6642     return pub;
6643 }();
6644 /**
6645   * Fires when the document is ready (before onload and before images are loaded).  Shorthand of {@link Roo.EventManager#onDocumentReady}.
6646   * @param {Function} fn        The method the event invokes
6647   * @param {Object}   scope    An  object that becomes the scope of the handler
6648   * @param {boolean}  override If true, the obj passed in becomes
6649   *                             the execution scope of the listener
6650   * @member Roo
6651   * @method onReady
6652  */
6653 Roo.onReady = Roo.EventManager.onDocumentReady;
6654
6655 Roo.onReady(function(){
6656     var bd = Roo.get(document.body);
6657     if(!bd){ return; }
6658
6659     var cls = [
6660             Roo.isIE ? "roo-ie"
6661             : Roo.isIE11 ? "roo-ie11"
6662             : Roo.isEdge ? "roo-edge"
6663             : Roo.isGecko ? "roo-gecko"
6664             : Roo.isOpera ? "roo-opera"
6665             : Roo.isSafari ? "roo-safari" : ""];
6666
6667     if(Roo.isMac){
6668         cls.push("roo-mac");
6669     }
6670     if(Roo.isLinux){
6671         cls.push("roo-linux");
6672     }
6673     if(Roo.isIOS){
6674         cls.push("roo-ios");
6675     }
6676     if(Roo.isTouch){
6677         cls.push("roo-touch");
6678     }
6679     if(Roo.isBorderBox){
6680         cls.push('roo-border-box');
6681     }
6682     if(Roo.isStrict){ // add to the parent to allow for selectors like ".ext-strict .ext-ie"
6683         var p = bd.dom.parentNode;
6684         if(p){
6685             p.className += ' roo-strict';
6686         }
6687     }
6688     bd.addClass(cls.join(' '));
6689 });
6690
6691 /**
6692  * @class Roo.EventObject
6693  * EventObject exposes the Yahoo! UI Event functionality directly on the object
6694  * passed to your event handler. It exists mostly for convenience. It also fixes the annoying null checks automatically to cleanup your code 
6695  * Example:
6696  * <pre><code>
6697  function handleClick(e){ // e is not a standard event object, it is a Roo.EventObject
6698     e.preventDefault();
6699     var target = e.getTarget();
6700     ...
6701  }
6702  var myDiv = Roo.get("myDiv");
6703  myDiv.on("click", handleClick);
6704  //or
6705  Roo.EventManager.on("myDiv", 'click', handleClick);
6706  Roo.EventManager.addListener("myDiv", 'click', handleClick);
6707  </code></pre>
6708  * @singleton
6709  */
6710 Roo.EventObject = function(){
6711     
6712     var E = Roo.lib.Event;
6713     
6714     // safari keypress events for special keys return bad keycodes
6715     var safariKeys = {
6716         63234 : 37, // left
6717         63235 : 39, // right
6718         63232 : 38, // up
6719         63233 : 40, // down
6720         63276 : 33, // page up
6721         63277 : 34, // page down
6722         63272 : 46, // delete
6723         63273 : 36, // home
6724         63275 : 35  // end
6725     };
6726
6727     // normalize button clicks
6728     var btnMap = Roo.isIE ? {1:0,4:1,2:2} :
6729                 (Roo.isSafari ? {1:0,2:1,3:2} : {0:0,1:1,2:2});
6730
6731     Roo.EventObjectImpl = function(e){
6732         if(e){
6733             this.setEvent(e.browserEvent || e);
6734         }
6735     };
6736     Roo.EventObjectImpl.prototype = {
6737         /**
6738          * Used to fix doc tools.
6739          * @scope Roo.EventObject.prototype
6740          */
6741             
6742
6743         
6744         
6745         /** The normal browser event */
6746         browserEvent : null,
6747         /** The button pressed in a mouse event */
6748         button : -1,
6749         /** True if the shift key was down during the event */
6750         shiftKey : false,
6751         /** True if the control key was down during the event */
6752         ctrlKey : false,
6753         /** True if the alt key was down during the event */
6754         altKey : false,
6755
6756         /** Key constant 
6757         * @type Number */
6758         BACKSPACE : 8,
6759         /** Key constant 
6760         * @type Number */
6761         TAB : 9,
6762         /** Key constant 
6763         * @type Number */
6764         RETURN : 13,
6765         /** Key constant 
6766         * @type Number */
6767         ENTER : 13,
6768         /** Key constant 
6769         * @type Number */
6770         SHIFT : 16,
6771         /** Key constant 
6772         * @type Number */
6773         CONTROL : 17,
6774         /** Key constant 
6775         * @type Number */
6776         ESC : 27,
6777         /** Key constant 
6778         * @type Number */
6779         SPACE : 32,
6780         /** Key constant 
6781         * @type Number */
6782         PAGEUP : 33,
6783         /** Key constant 
6784         * @type Number */
6785         PAGEDOWN : 34,
6786         /** Key constant 
6787         * @type Number */
6788         END : 35,
6789         /** Key constant 
6790         * @type Number */
6791         HOME : 36,
6792         /** Key constant 
6793         * @type Number */
6794         LEFT : 37,
6795         /** Key constant 
6796         * @type Number */
6797         UP : 38,
6798         /** Key constant 
6799         * @type Number */
6800         RIGHT : 39,
6801         /** Key constant 
6802         * @type Number */
6803         DOWN : 40,
6804         /** Key constant 
6805         * @type Number */
6806         DELETE : 46,
6807         /** Key constant 
6808         * @type Number */
6809         F5 : 116,
6810
6811            /** @private */
6812         setEvent : function(e){
6813             if(e == this || (e && e.browserEvent)){ // already wrapped
6814                 return e;
6815             }
6816             this.browserEvent = e;
6817             if(e){
6818                 // normalize buttons
6819                 this.button = e.button ? btnMap[e.button] : (e.which ? e.which-1 : -1);
6820                 if(e.type == 'click' && this.button == -1){
6821                     this.button = 0;
6822                 }
6823                 this.type = e.type;
6824                 this.shiftKey = e.shiftKey;
6825                 // mac metaKey behaves like ctrlKey
6826                 this.ctrlKey = e.ctrlKey || e.metaKey;
6827                 this.altKey = e.altKey;
6828                 // in getKey these will be normalized for the mac
6829                 this.keyCode = e.keyCode;
6830                 // keyup warnings on firefox.
6831                 this.charCode = (e.type == 'keyup' || e.type == 'keydown') ? 0 : e.charCode;
6832                 // cache the target for the delayed and or buffered events
6833                 this.target = E.getTarget(e);
6834                 // same for XY
6835                 this.xy = E.getXY(e);
6836             }else{
6837                 this.button = -1;
6838                 this.shiftKey = false;
6839                 this.ctrlKey = false;
6840                 this.altKey = false;
6841                 this.keyCode = 0;
6842                 this.charCode =0;
6843                 this.target = null;
6844                 this.xy = [0, 0];
6845             }
6846             return this;
6847         },
6848
6849         /**
6850          * Stop the event (preventDefault and stopPropagation)
6851          */
6852         stopEvent : function(){
6853             if(this.browserEvent){
6854                 if(this.browserEvent.type == 'mousedown'){
6855                     Roo.EventManager.stoppedMouseDownEvent.fire(this);
6856                 }
6857                 E.stopEvent(this.browserEvent);
6858             }
6859         },
6860
6861         /**
6862          * Prevents the browsers default handling of the event.
6863          */
6864         preventDefault : function(){
6865             if(this.browserEvent){
6866                 E.preventDefault(this.browserEvent);
6867             }
6868         },
6869
6870         /** @private */
6871         isNavKeyPress : function(){
6872             var k = this.keyCode;
6873             k = Roo.isSafari ? (safariKeys[k] || k) : k;
6874             return (k >= 33 && k <= 40) || k == this.RETURN || k == this.TAB || k == this.ESC;
6875         },
6876
6877         isSpecialKey : function(){
6878             var k = this.keyCode;
6879             return (this.type == 'keypress' && this.ctrlKey) || k == 9 || k == 13  || k == 40 || k == 27 ||
6880             (k == 16) || (k == 17) ||
6881             (k >= 18 && k <= 20) ||
6882             (k >= 33 && k <= 35) ||
6883             (k >= 36 && k <= 39) ||
6884             (k >= 44 && k <= 45);
6885         },
6886         /**
6887          * Cancels bubbling of the event.
6888          */
6889         stopPropagation : function(){
6890             if(this.browserEvent){
6891                 if(this.type == 'mousedown'){
6892                     Roo.EventManager.stoppedMouseDownEvent.fire(this);
6893                 }
6894                 E.stopPropagation(this.browserEvent);
6895             }
6896         },
6897
6898         /**
6899          * Gets the key code for the event.
6900          * @return {Number}
6901          */
6902         getCharCode : function(){
6903             return this.charCode || this.keyCode;
6904         },
6905
6906         /**
6907          * Returns a normalized keyCode for the event.
6908          * @return {Number} The key code
6909          */
6910         getKey : function(){
6911             var k = this.keyCode || this.charCode;
6912             return Roo.isSafari ? (safariKeys[k] || k) : k;
6913         },
6914
6915         /**
6916          * Gets the x coordinate of the event.
6917          * @return {Number}
6918          */
6919         getPageX : function(){
6920             return this.xy[0];
6921         },
6922
6923         /**
6924          * Gets the y coordinate of the event.
6925          * @return {Number}
6926          */
6927         getPageY : function(){
6928             return this.xy[1];
6929         },
6930
6931         /**
6932          * Gets the time of the event.
6933          * @return {Number}
6934          */
6935         getTime : function(){
6936             if(this.browserEvent){
6937                 return E.getTime(this.browserEvent);
6938             }
6939             return null;
6940         },
6941
6942         /**
6943          * Gets the page coordinates of the event.
6944          * @return {Array} The xy values like [x, y]
6945          */
6946         getXY : function(){
6947             return this.xy;
6948         },
6949
6950         /**
6951          * Gets the target for the event.
6952          * @param {String} selector (optional) A simple selector to filter the target or look for an ancestor of the target
6953          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
6954                 search as a number or element (defaults to 10 || document.body)
6955          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
6956          * @return {HTMLelement}
6957          */
6958         getTarget : function(selector, maxDepth, returnEl){
6959             return selector ? Roo.fly(this.target).findParent(selector, maxDepth, returnEl) : this.target;
6960         },
6961         /**
6962          * Gets the related target.
6963          * @return {HTMLElement}
6964          */
6965         getRelatedTarget : function(){
6966             if(this.browserEvent){
6967                 return E.getRelatedTarget(this.browserEvent);
6968             }
6969             return null;
6970         },
6971
6972         /**
6973          * Normalizes mouse wheel delta across browsers
6974          * @return {Number} The delta
6975          */
6976         getWheelDelta : function(){
6977             var e = this.browserEvent;
6978             var delta = 0;
6979             if(e.wheelDelta){ /* IE/Opera. */
6980                 delta = e.wheelDelta/120;
6981             }else if(e.detail){ /* Mozilla case. */
6982                 delta = -e.detail/3;
6983             }
6984             return delta;
6985         },
6986
6987         /**
6988          * Returns true if the control, meta, shift or alt key was pressed during this event.
6989          * @return {Boolean}
6990          */
6991         hasModifier : function(){
6992             return !!((this.ctrlKey || this.altKey) || this.shiftKey);
6993         },
6994
6995         /**
6996          * Returns true if the target of this event equals el or is a child of el
6997          * @param {String/HTMLElement/Element} el
6998          * @param {Boolean} related (optional) true to test if the related target is within el instead of the target
6999          * @return {Boolean}
7000          */
7001         within : function(el, related){
7002             var t = this[related ? "getRelatedTarget" : "getTarget"]();
7003             return t && Roo.fly(el).contains(t);
7004         },
7005
7006         getPoint : function(){
7007             return new Roo.lib.Point(this.xy[0], this.xy[1]);
7008         }
7009     };
7010
7011     return new Roo.EventObjectImpl();
7012 }();
7013             
7014     /*
7015  * Based on:
7016  * Ext JS Library 1.1.1
7017  * Copyright(c) 2006-2007, Ext JS, LLC.
7018  *
7019  * Originally Released Under LGPL - original licence link has changed is not relivant.
7020  *
7021  * Fork - LGPL
7022  * <script type="text/javascript">
7023  */
7024
7025  
7026 // was in Composite Element!??!?!
7027  
7028 (function(){
7029     var D = Roo.lib.Dom;
7030     var E = Roo.lib.Event;
7031     var A = Roo.lib.Anim;
7032
7033     // local style camelizing for speed
7034     var propCache = {};
7035     var camelRe = /(-[a-z])/gi;
7036     var camelFn = function(m, a){ return a.charAt(1).toUpperCase(); };
7037     var view = document.defaultView;
7038
7039 /**
7040  * @class Roo.Element
7041  * Represents an Element in the DOM.<br><br>
7042  * Usage:<br>
7043 <pre><code>
7044 var el = Roo.get("my-div");
7045
7046 // or with getEl
7047 var el = getEl("my-div");
7048
7049 // or with a DOM element
7050 var el = Roo.get(myDivElement);
7051 </code></pre>
7052  * Using Roo.get() or getEl() instead of calling the constructor directly ensures you get the same object
7053  * each call instead of constructing a new one.<br><br>
7054  * <b>Animations</b><br />
7055  * Many of the functions for manipulating an element have an optional "animate" parameter. The animate parameter
7056  * should either be a boolean (true) or an object literal with animation options. The animation options are:
7057 <pre>
7058 Option    Default   Description
7059 --------- --------  ---------------------------------------------
7060 duration  .35       The duration of the animation in seconds
7061 easing    easeOut   The YUI easing method
7062 callback  none      A function to execute when the anim completes
7063 scope     this      The scope (this) of the callback function
7064 </pre>
7065 * Also, the Anim object being used for the animation will be set on your options object as "anim", which allows you to stop or
7066 * manipulate the animation. Here's an example:
7067 <pre><code>
7068 var el = Roo.get("my-div");
7069
7070 // no animation
7071 el.setWidth(100);
7072
7073 // default animation
7074 el.setWidth(100, true);
7075
7076 // animation with some options set
7077 el.setWidth(100, {
7078     duration: 1,
7079     callback: this.foo,
7080     scope: this
7081 });
7082
7083 // using the "anim" property to get the Anim object
7084 var opt = {
7085     duration: 1,
7086     callback: this.foo,
7087     scope: this
7088 };
7089 el.setWidth(100, opt);
7090 ...
7091 if(opt.anim.isAnimated()){
7092     opt.anim.stop();
7093 }
7094 </code></pre>
7095 * <b> Composite (Collections of) Elements</b><br />
7096  * For working with collections of Elements, see <a href="Roo.CompositeElement.html">Roo.CompositeElement</a>
7097  * @constructor Create a new Element directly.
7098  * @param {String/HTMLElement} element
7099  * @param {Boolean} forceNew (optional) By default the constructor checks to see if there is already an instance of this element in the cache and if there is it returns the same instance. This will skip that check (useful for extending this class).
7100  */
7101     Roo.Element = function(element, forceNew){
7102         var dom = typeof element == "string" ?
7103                 document.getElementById(element) : element;
7104         if(!dom){ // invalid id/element
7105             return null;
7106         }
7107         var id = dom.id;
7108         if(forceNew !== true && id && Roo.Element.cache[id]){ // element object already exists
7109             return Roo.Element.cache[id];
7110         }
7111
7112         /**
7113          * The DOM element
7114          * @type HTMLElement
7115          */
7116         this.dom = dom;
7117
7118         /**
7119          * The DOM element ID
7120          * @type String
7121          */
7122         this.id = id || Roo.id(dom);
7123     };
7124
7125     var El = Roo.Element;
7126
7127     El.prototype = {
7128         /**
7129          * The element's default display mode  (defaults to "")
7130          * @type String
7131          */
7132         originalDisplay : "",
7133
7134         visibilityMode : 1,
7135         /**
7136          * The default unit to append to CSS values where a unit isn't provided (defaults to px).
7137          * @type String
7138          */
7139         defaultUnit : "px",
7140         
7141         /**
7142          * Sets the element's visibility mode. When setVisible() is called it
7143          * will use this to determine whether to set the visibility or the display property.
7144          * @param visMode Element.VISIBILITY or Element.DISPLAY
7145          * @return {Roo.Element} this
7146          */
7147         setVisibilityMode : function(visMode){
7148             this.visibilityMode = visMode;
7149             return this;
7150         },
7151         /**
7152          * Convenience method for setVisibilityMode(Element.DISPLAY)
7153          * @param {String} display (optional) What to set display to when visible
7154          * @return {Roo.Element} this
7155          */
7156         enableDisplayMode : function(display){
7157             this.setVisibilityMode(El.DISPLAY);
7158             if(typeof display != "undefined") { this.originalDisplay = display; }
7159             return this;
7160         },
7161
7162         /**
7163          * Looks at this node and then at parent nodes for a match of the passed simple selector (e.g. div.some-class or span:first-child)
7164          * @param {String} selector The simple selector to test
7165          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7166                 search as a number or element (defaults to 10 || document.body)
7167          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
7168          * @return {HTMLElement} The matching DOM node (or null if no match was found)
7169          */
7170         findParent : function(simpleSelector, maxDepth, returnEl){
7171             var p = this.dom, b = document.body, depth = 0, dq = Roo.DomQuery, stopEl;
7172             maxDepth = maxDepth || 50;
7173             if(typeof maxDepth != "number"){
7174                 stopEl = Roo.getDom(maxDepth);
7175                 maxDepth = 10;
7176             }
7177             while(p && p.nodeType == 1 && depth < maxDepth && p != b && p != stopEl){
7178                 if(dq.is(p, simpleSelector)){
7179                     return returnEl ? Roo.get(p) : p;
7180                 }
7181                 depth++;
7182                 p = p.parentNode;
7183             }
7184             return null;
7185         },
7186
7187
7188         /**
7189          * Looks at parent nodes for a match of the passed simple selector (e.g. div.some-class or span:first-child)
7190          * @param {String} selector The simple selector to test
7191          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7192                 search as a number or element (defaults to 10 || document.body)
7193          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
7194          * @return {HTMLElement} The matching DOM node (or null if no match was found)
7195          */
7196         findParentNode : function(simpleSelector, maxDepth, returnEl){
7197             var p = Roo.fly(this.dom.parentNode, '_internal');
7198             return p ? p.findParent(simpleSelector, maxDepth, returnEl) : null;
7199         },
7200         
7201         /**
7202          * Looks at  the scrollable parent element
7203          */
7204         findScrollableParent : function()
7205         {
7206             var overflowRegex = /(auto|scroll)/;
7207             
7208             if(this.getStyle('position') === 'fixed'){
7209                 return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7210             }
7211             
7212             var excludeStaticParent = this.getStyle('position') === "absolute";
7213             
7214             for (var parent = this; (parent = Roo.get(parent.dom.parentNode));){
7215                 
7216                 if (excludeStaticParent && parent.getStyle('position') === "static") {
7217                     continue;
7218                 }
7219                 
7220                 if (overflowRegex.test(parent.getStyle('overflow') + parent.getStyle('overflow-x') + parent.getStyle('overflow-y'))){
7221                     return parent;
7222                 }
7223                 
7224                 if(parent.dom.nodeName.toLowerCase() == 'body'){
7225                     return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7226                 }
7227             }
7228             
7229             return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7230         },
7231
7232         /**
7233          * Walks up the dom looking for a parent node that matches the passed simple selector (e.g. div.some-class or span:first-child).
7234          * This is a shortcut for findParentNode() that always returns an Roo.Element.
7235          * @param {String} selector The simple selector to test
7236          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7237                 search as a number or element (defaults to 10 || document.body)
7238          * @return {Roo.Element} The matching DOM node (or null if no match was found)
7239          */
7240         up : function(simpleSelector, maxDepth){
7241             return this.findParentNode(simpleSelector, maxDepth, true);
7242         },
7243
7244
7245
7246         /**
7247          * Returns true if this element matches the passed simple selector (e.g. div.some-class or span:first-child)
7248          * @param {String} selector The simple selector to test
7249          * @return {Boolean} True if this element matches the selector, else false
7250          */
7251         is : function(simpleSelector){
7252             return Roo.DomQuery.is(this.dom, simpleSelector);
7253         },
7254
7255         /**
7256          * Perform animation on this element.
7257          * @param {Object} args The YUI animation control args
7258          * @param {Float} duration (optional) How long the animation lasts in seconds (defaults to .35)
7259          * @param {Function} onComplete (optional) Function to call when animation completes
7260          * @param {String} easing (optional) Easing method to use (defaults to 'easeOut')
7261          * @param {String} animType (optional) 'run' is the default. Can also be 'color', 'motion', or 'scroll'
7262          * @return {Roo.Element} this
7263          */
7264         animate : function(args, duration, onComplete, easing, animType){
7265             this.anim(args, {duration: duration, callback: onComplete, easing: easing}, animType);
7266             return this;
7267         },
7268
7269         /*
7270          * @private Internal animation call
7271          */
7272         anim : function(args, opt, animType, defaultDur, defaultEase, cb){
7273             animType = animType || 'run';
7274             opt = opt || {};
7275             var anim = Roo.lib.Anim[animType](
7276                 this.dom, args,
7277                 (opt.duration || defaultDur) || .35,
7278                 (opt.easing || defaultEase) || 'easeOut',
7279                 function(){
7280                     Roo.callback(cb, this);
7281                     Roo.callback(opt.callback, opt.scope || this, [this, opt]);
7282                 },
7283                 this
7284             );
7285             opt.anim = anim;
7286             return anim;
7287         },
7288
7289         // private legacy anim prep
7290         preanim : function(a, i){
7291             return !a[i] ? false : (typeof a[i] == "object" ? a[i]: {duration: a[i+1], callback: a[i+2], easing: a[i+3]});
7292         },
7293
7294         /**
7295          * Removes worthless text nodes
7296          * @param {Boolean} forceReclean (optional) By default the element
7297          * keeps track if it has been cleaned already so
7298          * you can call this over and over. However, if you update the element and
7299          * need to force a reclean, you can pass true.
7300          */
7301         clean : function(forceReclean){
7302             if(this.isCleaned && forceReclean !== true){
7303                 return this;
7304             }
7305             var ns = /\S/;
7306             var d = this.dom, n = d.firstChild, ni = -1;
7307             while(n){
7308                 var nx = n.nextSibling;
7309                 if(n.nodeType == 3 && !ns.test(n.nodeValue)){
7310                     d.removeChild(n);
7311                 }else{
7312                     n.nodeIndex = ++ni;
7313                 }
7314                 n = nx;
7315             }
7316             this.isCleaned = true;
7317             return this;
7318         },
7319
7320         // private
7321         calcOffsetsTo : function(el){
7322             el = Roo.get(el);
7323             var d = el.dom;
7324             var restorePos = false;
7325             if(el.getStyle('position') == 'static'){
7326                 el.position('relative');
7327                 restorePos = true;
7328             }
7329             var x = 0, y =0;
7330             var op = this.dom;
7331             while(op && op != d && op.tagName != 'HTML'){
7332                 x+= op.offsetLeft;
7333                 y+= op.offsetTop;
7334                 op = op.offsetParent;
7335             }
7336             if(restorePos){
7337                 el.position('static');
7338             }
7339             return [x, y];
7340         },
7341
7342         /**
7343          * Scrolls this element into view within the passed container.
7344          * @param {String/HTMLElement/Element} container (optional) The container element to scroll (defaults to document.body)
7345          * @param {Boolean} hscroll (optional) False to disable horizontal scroll (defaults to true)
7346          * @return {Roo.Element} this
7347          */
7348         scrollIntoView : function(container, hscroll){
7349             var c = Roo.getDom(container) || document.body;
7350             var el = this.dom;
7351
7352             var o = this.calcOffsetsTo(c),
7353                 l = o[0],
7354                 t = o[1],
7355                 b = t+el.offsetHeight,
7356                 r = l+el.offsetWidth;
7357
7358             var ch = c.clientHeight;
7359             var ct = parseInt(c.scrollTop, 10);
7360             var cl = parseInt(c.scrollLeft, 10);
7361             var cb = ct + ch;
7362             var cr = cl + c.clientWidth;
7363
7364             if(t < ct){
7365                 c.scrollTop = t;
7366             }else if(b > cb){
7367                 c.scrollTop = b-ch;
7368             }
7369
7370             if(hscroll !== false){
7371                 if(l < cl){
7372                     c.scrollLeft = l;
7373                 }else if(r > cr){
7374                     c.scrollLeft = r-c.clientWidth;
7375                 }
7376             }
7377             return this;
7378         },
7379
7380         // private
7381         scrollChildIntoView : function(child, hscroll){
7382             Roo.fly(child, '_scrollChildIntoView').scrollIntoView(this, hscroll);
7383         },
7384
7385         /**
7386          * Measures the element's content height and updates height to match. Note: this function uses setTimeout so
7387          * the new height may not be available immediately.
7388          * @param {Boolean} animate (optional) Animate the transition (defaults to false)
7389          * @param {Float} duration (optional) Length of the animation in seconds (defaults to .35)
7390          * @param {Function} onComplete (optional) Function to call when animation completes
7391          * @param {String} easing (optional) Easing method to use (defaults to easeOut)
7392          * @return {Roo.Element} this
7393          */
7394         autoHeight : function(animate, duration, onComplete, easing){
7395             var oldHeight = this.getHeight();
7396             this.clip();
7397             this.setHeight(1); // force clipping
7398             setTimeout(function(){
7399                 var height = parseInt(this.dom.scrollHeight, 10); // parseInt for Safari
7400                 if(!animate){
7401                     this.setHeight(height);
7402                     this.unclip();
7403                     if(typeof onComplete == "function"){
7404                         onComplete();
7405                     }
7406                 }else{
7407                     this.setHeight(oldHeight); // restore original height
7408                     this.setHeight(height, animate, duration, function(){
7409                         this.unclip();
7410                         if(typeof onComplete == "function") { onComplete(); }
7411                     }.createDelegate(this), easing);
7412                 }
7413             }.createDelegate(this), 0);
7414             return this;
7415         },
7416
7417         /**
7418          * Returns true if this element is an ancestor of the passed element
7419          * @param {HTMLElement/String} el The element to check
7420          * @return {Boolean} True if this element is an ancestor of el, else false
7421          */
7422         contains : function(el){
7423             if(!el){return false;}
7424             return D.isAncestor(this.dom, el.dom ? el.dom : el);
7425         },
7426
7427         /**
7428          * Checks whether the element is currently visible using both visibility and display properties.
7429          * @param {Boolean} deep (optional) True to walk the dom and see if parent elements are hidden (defaults to false)
7430          * @return {Boolean} True if the element is currently visible, else false
7431          */
7432         isVisible : function(deep) {
7433             var vis = !(this.getStyle("visibility") == "hidden" || this.getStyle("display") == "none");
7434             if(deep !== true || !vis){
7435                 return vis;
7436             }
7437             var p = this.dom.parentNode;
7438             while(p && p.tagName.toLowerCase() != "body"){
7439                 if(!Roo.fly(p, '_isVisible').isVisible()){
7440                     return false;
7441                 }
7442                 p = p.parentNode;
7443             }
7444             return true;
7445         },
7446
7447         /**
7448          * Creates a {@link Roo.CompositeElement} for child nodes based on the passed CSS selector (the selector should not contain an id).
7449          * @param {String} selector The CSS selector
7450          * @param {Boolean} unique (optional) True to create a unique Roo.Element for each child (defaults to false, which creates a single shared flyweight object)
7451          * @return {CompositeElement/CompositeElementLite} The composite element
7452          */
7453         select : function(selector, unique){
7454             return El.select(selector, unique, this.dom);
7455         },
7456
7457         /**
7458          * Selects child nodes based on the passed CSS selector (the selector should not contain an id).
7459          * @param {String} selector The CSS selector
7460          * @return {Array} An array of the matched nodes
7461          */
7462         query : function(selector, unique){
7463             return Roo.DomQuery.select(selector, this.dom);
7464         },
7465
7466         /**
7467          * Selects a single child at any depth below this element based on the passed CSS selector (the selector should not contain an id).
7468          * @param {String} selector The CSS selector
7469          * @param {Boolean} returnDom (optional) True to return the DOM node instead of Roo.Element (defaults to false)
7470          * @return {HTMLElement/Roo.Element} The child Roo.Element (or DOM node if returnDom = true)
7471          */
7472         child : function(selector, returnDom){
7473             var n = Roo.DomQuery.selectNode(selector, this.dom);
7474             return returnDom ? n : Roo.get(n);
7475         },
7476
7477         /**
7478          * Selects a single *direct* child based on the passed CSS selector (the selector should not contain an id).
7479          * @param {String} selector The CSS selector
7480          * @param {Boolean} returnDom (optional) True to return the DOM node instead of Roo.Element (defaults to false)
7481          * @return {HTMLElement/Roo.Element} The child Roo.Element (or DOM node if returnDom = true)
7482          */
7483         down : function(selector, returnDom){
7484             var n = Roo.DomQuery.selectNode(" > " + selector, this.dom);
7485             return returnDom ? n : Roo.get(n);
7486         },
7487
7488         /**
7489          * Initializes a {@link Roo.dd.DD} drag drop object for this element.
7490          * @param {String} group The group the DD object is member of
7491          * @param {Object} config The DD config object
7492          * @param {Object} overrides An object containing methods to override/implement on the DD object
7493          * @return {Roo.dd.DD} The DD object
7494          */
7495         initDD : function(group, config, overrides){
7496             var dd = new Roo.dd.DD(Roo.id(this.dom), group, config);
7497             return Roo.apply(dd, overrides);
7498         },
7499
7500         /**
7501          * Initializes a {@link Roo.dd.DDProxy} object for this element.
7502          * @param {String} group The group the DDProxy object is member of
7503          * @param {Object} config The DDProxy config object
7504          * @param {Object} overrides An object containing methods to override/implement on the DDProxy object
7505          * @return {Roo.dd.DDProxy} The DDProxy object
7506          */
7507         initDDProxy : function(group, config, overrides){
7508             var dd = new Roo.dd.DDProxy(Roo.id(this.dom), group, config);
7509             return Roo.apply(dd, overrides);
7510         },
7511
7512         /**
7513          * Initializes a {@link Roo.dd.DDTarget} object for this element.
7514          * @param {String} group The group the DDTarget object is member of
7515          * @param {Object} config The DDTarget config object
7516          * @param {Object} overrides An object containing methods to override/implement on the DDTarget object
7517          * @return {Roo.dd.DDTarget} The DDTarget object
7518          */
7519         initDDTarget : function(group, config, overrides){
7520             var dd = new Roo.dd.DDTarget(Roo.id(this.dom), group, config);
7521             return Roo.apply(dd, overrides);
7522         },
7523
7524         /**
7525          * Sets the visibility of the element (see details). If the visibilityMode is set to Element.DISPLAY, it will use
7526          * the display property to hide the element, otherwise it uses visibility. The default is to hide and show using the visibility property.
7527          * @param {Boolean} visible Whether the element is visible
7528          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7529          * @return {Roo.Element} this
7530          */
7531          setVisible : function(visible, animate){
7532             if(!animate || !A){
7533                 if(this.visibilityMode == El.DISPLAY){
7534                     this.setDisplayed(visible);
7535                 }else{
7536                     this.fixDisplay();
7537                     this.dom.style.visibility = visible ? "visible" : "hidden";
7538                 }
7539             }else{
7540                 // closure for composites
7541                 var dom = this.dom;
7542                 var visMode = this.visibilityMode;
7543                 if(visible){
7544                     this.setOpacity(.01);
7545                     this.setVisible(true);
7546                 }
7547                 this.anim({opacity: { to: (visible?1:0) }},
7548                       this.preanim(arguments, 1),
7549                       null, .35, 'easeIn', function(){
7550                          if(!visible){
7551                              if(visMode == El.DISPLAY){
7552                                  dom.style.display = "none";
7553                              }else{
7554                                  dom.style.visibility = "hidden";
7555                              }
7556                              Roo.get(dom).setOpacity(1);
7557                          }
7558                      });
7559             }
7560             return this;
7561         },
7562
7563         /**
7564          * Returns true if display is not "none"
7565          * @return {Boolean}
7566          */
7567         isDisplayed : function() {
7568             return this.getStyle("display") != "none";
7569         },
7570
7571         /**
7572          * Toggles the element's visibility or display, depending on visibility mode.
7573          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7574          * @return {Roo.Element} this
7575          */
7576         toggle : function(animate){
7577             this.setVisible(!this.isVisible(), this.preanim(arguments, 0));
7578             return this;
7579         },
7580
7581         /**
7582          * Sets the CSS display property. Uses originalDisplay if the specified value is a boolean true.
7583          * @param {Boolean} value Boolean value to display the element using its default display, or a string to set the display directly
7584          * @return {Roo.Element} this
7585          */
7586         setDisplayed : function(value) {
7587             if(typeof value == "boolean"){
7588                value = value ? this.originalDisplay : "none";
7589             }
7590             this.setStyle("display", value);
7591             return this;
7592         },
7593
7594         /**
7595          * Tries to focus the element. Any exceptions are caught and ignored.
7596          * @return {Roo.Element} this
7597          */
7598         focus : function() {
7599             try{
7600                 this.dom.focus();
7601             }catch(e){}
7602             return this;
7603         },
7604
7605         /**
7606          * Tries to blur the element. Any exceptions are caught and ignored.
7607          * @return {Roo.Element} this
7608          */
7609         blur : function() {
7610             try{
7611                 this.dom.blur();
7612             }catch(e){}
7613             return this;
7614         },
7615
7616         /**
7617          * Adds one or more CSS classes to the element. Duplicate classes are automatically filtered out.
7618          * @param {String/Array} className The CSS class to add, or an array of classes
7619          * @return {Roo.Element} this
7620          */
7621         addClass : function(className){
7622             if(className instanceof Array){
7623                 for(var i = 0, len = className.length; i < len; i++) {
7624                     this.addClass(className[i]);
7625                 }
7626             }else{
7627                 if(className && !this.hasClass(className)){
7628                     this.dom.className = this.dom.className + " " + className;
7629                 }
7630             }
7631             return this;
7632         },
7633
7634         /**
7635          * Adds one or more CSS classes to this element and removes the same class(es) from all siblings.
7636          * @param {String/Array} className The CSS class to add, or an array of classes
7637          * @return {Roo.Element} this
7638          */
7639         radioClass : function(className){
7640             var siblings = this.dom.parentNode.childNodes;
7641             for(var i = 0; i < siblings.length; i++) {
7642                 var s = siblings[i];
7643                 if(s.nodeType == 1){
7644                     Roo.get(s).removeClass(className);
7645                 }
7646             }
7647             this.addClass(className);
7648             return this;
7649         },
7650
7651         /**
7652          * Removes one or more CSS classes from the element.
7653          * @param {String/Array} className The CSS class to remove, or an array of classes
7654          * @return {Roo.Element} this
7655          */
7656         removeClass : function(className){
7657             if(!className || !this.dom.className){
7658                 return this;
7659             }
7660             if(className instanceof Array){
7661                 for(var i = 0, len = className.length; i < len; i++) {
7662                     this.removeClass(className[i]);
7663                 }
7664             }else{
7665                 if(this.hasClass(className)){
7666                     var re = this.classReCache[className];
7667                     if (!re) {
7668                        re = new RegExp('(?:^|\\s+)' + className + '(?:\\s+|$)', "g");
7669                        this.classReCache[className] = re;
7670                     }
7671                     this.dom.className =
7672                         this.dom.className.replace(re, " ");
7673                 }
7674             }
7675             return this;
7676         },
7677
7678         // private
7679         classReCache: {},
7680
7681         /**
7682          * Toggles the specified CSS class on this element (removes it if it already exists, otherwise adds it).
7683          * @param {String} className The CSS class to toggle
7684          * @return {Roo.Element} this
7685          */
7686         toggleClass : function(className){
7687             if(this.hasClass(className)){
7688                 this.removeClass(className);
7689             }else{
7690                 this.addClass(className);
7691             }
7692             return this;
7693         },
7694
7695         /**
7696          * Checks if the specified CSS class exists on this element's DOM node.
7697          * @param {String} className The CSS class to check for
7698          * @return {Boolean} True if the class exists, else false
7699          */
7700         hasClass : function(className){
7701             return className && (' '+this.dom.className+' ').indexOf(' '+className+' ') != -1;
7702         },
7703
7704         /**
7705          * Replaces a CSS class on the element with another.  If the old name does not exist, the new name will simply be added.
7706          * @param {String} oldClassName The CSS class to replace
7707          * @param {String} newClassName The replacement CSS class
7708          * @return {Roo.Element} this
7709          */
7710         replaceClass : function(oldClassName, newClassName){
7711             this.removeClass(oldClassName);
7712             this.addClass(newClassName);
7713             return this;
7714         },
7715
7716         /**
7717          * Returns an object with properties matching the styles requested.
7718          * For example, el.getStyles('color', 'font-size', 'width') might return
7719          * {'color': '#FFFFFF', 'font-size': '13px', 'width': '100px'}.
7720          * @param {String} style1 A style name
7721          * @param {String} style2 A style name
7722          * @param {String} etc.
7723          * @return {Object} The style object
7724          */
7725         getStyles : function(){
7726             var a = arguments, len = a.length, r = {};
7727             for(var i = 0; i < len; i++){
7728                 r[a[i]] = this.getStyle(a[i]);
7729             }
7730             return r;
7731         },
7732
7733         /**
7734          * Normalizes currentStyle and computedStyle. This is not YUI getStyle, it is an optimised version.
7735          * @param {String} property The style property whose value is returned.
7736          * @return {String} The current value of the style property for this element.
7737          */
7738         getStyle : function(){
7739             return view && view.getComputedStyle ?
7740                 function(prop){
7741                     var el = this.dom, v, cs, camel;
7742                     if(prop == 'float'){
7743                         prop = "cssFloat";
7744                     }
7745                     if(el.style && (v = el.style[prop])){
7746                         return v;
7747                     }
7748                     if(cs = view.getComputedStyle(el, "")){
7749                         if(!(camel = propCache[prop])){
7750                             camel = propCache[prop] = prop.replace(camelRe, camelFn);
7751                         }
7752                         return cs[camel];
7753                     }
7754                     return null;
7755                 } :
7756                 function(prop){
7757                     var el = this.dom, v, cs, camel;
7758                     if(prop == 'opacity'){
7759                         if(typeof el.style.filter == 'string'){
7760                             var m = el.style.filter.match(/alpha\(opacity=(.*)\)/i);
7761                             if(m){
7762                                 var fv = parseFloat(m[1]);
7763                                 if(!isNaN(fv)){
7764                                     return fv ? fv / 100 : 0;
7765                                 }
7766                             }
7767                         }
7768                         return 1;
7769                     }else if(prop == 'float'){
7770                         prop = "styleFloat";
7771                     }
7772                     if(!(camel = propCache[prop])){
7773                         camel = propCache[prop] = prop.replace(camelRe, camelFn);
7774                     }
7775                     if(v = el.style[camel]){
7776                         return v;
7777                     }
7778                     if(cs = el.currentStyle){
7779                         return cs[camel];
7780                     }
7781                     return null;
7782                 };
7783         }(),
7784
7785         /**
7786          * Wrapper for setting style properties, also takes single object parameter of multiple styles.
7787          * @param {String/Object} property The style property to be set, or an object of multiple styles.
7788          * @param {String} value (optional) The value to apply to the given property, or null if an object was passed.
7789          * @return {Roo.Element} this
7790          */
7791         setStyle : function(prop, value){
7792             if(typeof prop == "string"){
7793                 
7794                 if (prop == 'float') {
7795                     this.setStyle(Roo.isIE ? 'styleFloat'  : 'cssFloat', value);
7796                     return this;
7797                 }
7798                 
7799                 var camel;
7800                 if(!(camel = propCache[prop])){
7801                     camel = propCache[prop] = prop.replace(camelRe, camelFn);
7802                 }
7803                 
7804                 if(camel == 'opacity') {
7805                     this.setOpacity(value);
7806                 }else{
7807                     this.dom.style[camel] = value;
7808                 }
7809             }else{
7810                 for(var style in prop){
7811                     if(typeof prop[style] != "function"){
7812                        this.setStyle(style, prop[style]);
7813                     }
7814                 }
7815             }
7816             return this;
7817         },
7818
7819         /**
7820          * More flexible version of {@link #setStyle} for setting style properties.
7821          * @param {String/Object/Function} styles A style specification string, e.g. "width:100px", or object in the form {width:"100px"}, or
7822          * a function which returns such a specification.
7823          * @return {Roo.Element} this
7824          */
7825         applyStyles : function(style){
7826             Roo.DomHelper.applyStyles(this.dom, style);
7827             return this;
7828         },
7829
7830         /**
7831           * Gets the current X position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7832           * @return {Number} The X position of the element
7833           */
7834         getX : function(){
7835             return D.getX(this.dom);
7836         },
7837
7838         /**
7839           * Gets the current Y position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7840           * @return {Number} The Y position of the element
7841           */
7842         getY : function(){
7843             return D.getY(this.dom);
7844         },
7845
7846         /**
7847           * Gets the current position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7848           * @return {Array} The XY position of the element
7849           */
7850         getXY : function(){
7851             return D.getXY(this.dom);
7852         },
7853
7854         /**
7855          * Sets the X position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7856          * @param {Number} The X position of the element
7857          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7858          * @return {Roo.Element} this
7859          */
7860         setX : function(x, animate){
7861             if(!animate || !A){
7862                 D.setX(this.dom, x);
7863             }else{
7864                 this.setXY([x, this.getY()], this.preanim(arguments, 1));
7865             }
7866             return this;
7867         },
7868
7869         /**
7870          * Sets the Y position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7871          * @param {Number} The Y position of the element
7872          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7873          * @return {Roo.Element} this
7874          */
7875         setY : function(y, animate){
7876             if(!animate || !A){
7877                 D.setY(this.dom, y);
7878             }else{
7879                 this.setXY([this.getX(), y], this.preanim(arguments, 1));
7880             }
7881             return this;
7882         },
7883
7884         /**
7885          * Sets the element's left position directly using CSS style (instead of {@link #setX}).
7886          * @param {String} left The left CSS property value
7887          * @return {Roo.Element} this
7888          */
7889         setLeft : function(left){
7890             this.setStyle("left", this.addUnits(left));
7891             return this;
7892         },
7893
7894         /**
7895          * Sets the element's top position directly using CSS style (instead of {@link #setY}).
7896          * @param {String} top The top CSS property value
7897          * @return {Roo.Element} this
7898          */
7899         setTop : function(top){
7900             this.setStyle("top", this.addUnits(top));
7901             return this;
7902         },
7903
7904         /**
7905          * Sets the element's CSS right style.
7906          * @param {String} right The right CSS property value
7907          * @return {Roo.Element} this
7908          */
7909         setRight : function(right){
7910             this.setStyle("right", this.addUnits(right));
7911             return this;
7912         },
7913
7914         /**
7915          * Sets the element's CSS bottom style.
7916          * @param {String} bottom The bottom CSS property value
7917          * @return {Roo.Element} this
7918          */
7919         setBottom : function(bottom){
7920             this.setStyle("bottom", this.addUnits(bottom));
7921             return this;
7922         },
7923
7924         /**
7925          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7926          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7927          * @param {Array} pos Contains X & Y [x, y] values for new position (coordinates are page-based)
7928          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7929          * @return {Roo.Element} this
7930          */
7931         setXY : function(pos, animate){
7932             if(!animate || !A){
7933                 D.setXY(this.dom, pos);
7934             }else{
7935                 this.anim({points: {to: pos}}, this.preanim(arguments, 1), 'motion');
7936             }
7937             return this;
7938         },
7939
7940         /**
7941          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7942          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7943          * @param {Number} x X value for new position (coordinates are page-based)
7944          * @param {Number} y Y value for new position (coordinates are page-based)
7945          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7946          * @return {Roo.Element} this
7947          */
7948         setLocation : function(x, y, animate){
7949             this.setXY([x, y], this.preanim(arguments, 2));
7950             return this;
7951         },
7952
7953         /**
7954          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7955          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7956          * @param {Number} x X value for new position (coordinates are page-based)
7957          * @param {Number} y Y value for new position (coordinates are page-based)
7958          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7959          * @return {Roo.Element} this
7960          */
7961         moveTo : function(x, y, animate){
7962             this.setXY([x, y], this.preanim(arguments, 2));
7963             return this;
7964         },
7965
7966         /**
7967          * Returns the region of the given element.
7968          * The element must be part of the DOM tree to have a region (display:none or elements not appended return false).
7969          * @return {Region} A Roo.lib.Region containing "top, left, bottom, right" member data.
7970          */
7971         getRegion : function(){
7972             return D.getRegion(this.dom);
7973         },
7974
7975         /**
7976          * Returns the offset height of the element
7977          * @param {Boolean} contentHeight (optional) true to get the height minus borders and padding
7978          * @return {Number} The element's height
7979          */
7980         getHeight : function(contentHeight){
7981             var h = this.dom.offsetHeight || 0;
7982             return contentHeight !== true ? h : h-this.getBorderWidth("tb")-this.getPadding("tb");
7983         },
7984
7985         /**
7986          * Returns the offset width of the element
7987          * @param {Boolean} contentWidth (optional) true to get the width minus borders and padding
7988          * @return {Number} The element's width
7989          */
7990         getWidth : function(contentWidth){
7991             var w = this.dom.offsetWidth || 0;
7992             return contentWidth !== true ? w : w-this.getBorderWidth("lr")-this.getPadding("lr");
7993         },
7994
7995         /**
7996          * Returns either the offsetHeight or the height of this element based on CSS height adjusted by padding or borders
7997          * when needed to simulate offsetHeight when offsets aren't available. This may not work on display:none elements
7998          * if a height has not been set using CSS.
7999          * @return {Number}
8000          */
8001         getComputedHeight : function(){
8002             var h = Math.max(this.dom.offsetHeight, this.dom.clientHeight);
8003             if(!h){
8004                 h = parseInt(this.getStyle('height'), 10) || 0;
8005                 if(!this.isBorderBox()){
8006                     h += this.getFrameWidth('tb');
8007                 }
8008             }
8009             return h;
8010         },
8011
8012         /**
8013          * Returns either the offsetWidth or the width of this element based on CSS width adjusted by padding or borders
8014          * when needed to simulate offsetWidth when offsets aren't available. This may not work on display:none elements
8015          * if a width has not been set using CSS.
8016          * @return {Number}
8017          */
8018         getComputedWidth : function(){
8019             var w = Math.max(this.dom.offsetWidth, this.dom.clientWidth);
8020             if(!w){
8021                 w = parseInt(this.getStyle('width'), 10) || 0;
8022                 if(!this.isBorderBox()){
8023                     w += this.getFrameWidth('lr');
8024                 }
8025             }
8026             return w;
8027         },
8028
8029         /**
8030          * Returns the size of the element.
8031          * @param {Boolean} contentSize (optional) true to get the width/size minus borders and padding
8032          * @return {Object} An object containing the element's size {width: (element width), height: (element height)}
8033          */
8034         getSize : function(contentSize){
8035             return {width: this.getWidth(contentSize), height: this.getHeight(contentSize)};
8036         },
8037
8038         /**
8039          * Returns the width and height of the viewport.
8040          * @return {Object} An object containing the viewport's size {width: (viewport width), height: (viewport height)}
8041          */
8042         getViewSize : function(){
8043             var d = this.dom, doc = document, aw = 0, ah = 0;
8044             if(d == doc || d == doc.body){
8045                 return {width : D.getViewWidth(), height: D.getViewHeight()};
8046             }else{
8047                 return {
8048                     width : d.clientWidth,
8049                     height: d.clientHeight
8050                 };
8051             }
8052         },
8053
8054         /**
8055          * Returns the value of the "value" attribute
8056          * @param {Boolean} asNumber true to parse the value as a number
8057          * @return {String/Number}
8058          */
8059         getValue : function(asNumber){
8060             return asNumber ? parseInt(this.dom.value, 10) : this.dom.value;
8061         },
8062
8063         // private
8064         adjustWidth : function(width){
8065             if(typeof width == "number"){
8066                 if(this.autoBoxAdjust && !this.isBorderBox()){
8067                    width -= (this.getBorderWidth("lr") + this.getPadding("lr"));
8068                 }
8069                 if(width < 0){
8070                     width = 0;
8071                 }
8072             }
8073             return width;
8074         },
8075
8076         // private
8077         adjustHeight : function(height){
8078             if(typeof height == "number"){
8079                if(this.autoBoxAdjust && !this.isBorderBox()){
8080                    height -= (this.getBorderWidth("tb") + this.getPadding("tb"));
8081                }
8082                if(height < 0){
8083                    height = 0;
8084                }
8085             }
8086             return height;
8087         },
8088
8089         /**
8090          * Set the width of the element
8091          * @param {Number} width The new width
8092          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8093          * @return {Roo.Element} this
8094          */
8095         setWidth : function(width, animate){
8096             width = this.adjustWidth(width);
8097             if(!animate || !A){
8098                 this.dom.style.width = this.addUnits(width);
8099             }else{
8100                 this.anim({width: {to: width}}, this.preanim(arguments, 1));
8101             }
8102             return this;
8103         },
8104
8105         /**
8106          * Set the height of the element
8107          * @param {Number} height The new height
8108          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8109          * @return {Roo.Element} this
8110          */
8111          setHeight : function(height, animate){
8112             height = this.adjustHeight(height);
8113             if(!animate || !A){
8114                 this.dom.style.height = this.addUnits(height);
8115             }else{
8116                 this.anim({height: {to: height}}, this.preanim(arguments, 1));
8117             }
8118             return this;
8119         },
8120
8121         /**
8122          * Set the size of the element. If animation is true, both width an height will be animated concurrently.
8123          * @param {Number} width The new width
8124          * @param {Number} height The new height
8125          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8126          * @return {Roo.Element} this
8127          */
8128          setSize : function(width, height, animate){
8129             if(typeof width == "object"){ // in case of object from getSize()
8130                 height = width.height; width = width.width;
8131             }
8132             width = this.adjustWidth(width); height = this.adjustHeight(height);
8133             if(!animate || !A){
8134                 this.dom.style.width = this.addUnits(width);
8135                 this.dom.style.height = this.addUnits(height);
8136             }else{
8137                 this.anim({width: {to: width}, height: {to: height}}, this.preanim(arguments, 2));
8138             }
8139             return this;
8140         },
8141
8142         /**
8143          * Sets the element's position and size in one shot. If animation is true then width, height, x and y will be animated concurrently.
8144          * @param {Number} x X value for new position (coordinates are page-based)
8145          * @param {Number} y Y value for new position (coordinates are page-based)
8146          * @param {Number} width The new width
8147          * @param {Number} height The new height
8148          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8149          * @return {Roo.Element} this
8150          */
8151         setBounds : function(x, y, width, height, animate){
8152             if(!animate || !A){
8153                 this.setSize(width, height);
8154                 this.setLocation(x, y);
8155             }else{
8156                 width = this.adjustWidth(width); height = this.adjustHeight(height);
8157                 this.anim({points: {to: [x, y]}, width: {to: width}, height: {to: height}},
8158                               this.preanim(arguments, 4), 'motion');
8159             }
8160             return this;
8161         },
8162
8163         /**
8164          * Sets the element's position and size the the specified region. If animation is true then width, height, x and y will be animated concurrently.
8165          * @param {Roo.lib.Region} region The region to fill
8166          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8167          * @return {Roo.Element} this
8168          */
8169         setRegion : function(region, animate){
8170             this.setBounds(region.left, region.top, region.right-region.left, region.bottom-region.top, this.preanim(arguments, 1));
8171             return this;
8172         },
8173
8174         /**
8175          * Appends an event handler
8176          *
8177          * @param {String}   eventName     The type of event to append
8178          * @param {Function} fn        The method the event invokes
8179          * @param {Object} scope       (optional) The scope (this object) of the fn
8180          * @param {Object}   options   (optional)An object with standard {@link Roo.EventManager#addListener} options
8181          */
8182         addListener : function(eventName, fn, scope, options){
8183             if (this.dom) {
8184                 Roo.EventManager.on(this.dom,  eventName, fn, scope || this, options);
8185             }
8186         },
8187
8188         /**
8189          * Removes an event handler from this element
8190          * @param {String} eventName the type of event to remove
8191          * @param {Function} fn the method the event invokes
8192          * @return {Roo.Element} this
8193          */
8194         removeListener : function(eventName, fn){
8195             Roo.EventManager.removeListener(this.dom,  eventName, fn);
8196             return this;
8197         },
8198
8199         /**
8200          * Removes all previous added listeners from this element
8201          * @return {Roo.Element} this
8202          */
8203         removeAllListeners : function(){
8204             E.purgeElement(this.dom);
8205             return this;
8206         },
8207
8208         relayEvent : function(eventName, observable){
8209             this.on(eventName, function(e){
8210                 observable.fireEvent(eventName, e);
8211             });
8212         },
8213
8214         /**
8215          * Set the opacity of the element
8216          * @param {Float} opacity The new opacity. 0 = transparent, .5 = 50% visibile, 1 = fully visible, etc
8217          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8218          * @return {Roo.Element} this
8219          */
8220          setOpacity : function(opacity, animate){
8221             if(!animate || !A){
8222                 var s = this.dom.style;
8223                 if(Roo.isIE){
8224                     s.zoom = 1;
8225                     s.filter = (s.filter || '').replace(/alpha\([^\)]*\)/gi,"") +
8226                                (opacity == 1 ? "" : "alpha(opacity=" + opacity * 100 + ")");
8227                 }else{
8228                     s.opacity = opacity;
8229                 }
8230             }else{
8231                 this.anim({opacity: {to: opacity}}, this.preanim(arguments, 1), null, .35, 'easeIn');
8232             }
8233             return this;
8234         },
8235
8236         /**
8237          * Gets the left X coordinate
8238          * @param {Boolean} local True to get the local css position instead of page coordinate
8239          * @return {Number}
8240          */
8241         getLeft : function(local){
8242             if(!local){
8243                 return this.getX();
8244             }else{
8245                 return parseInt(this.getStyle("left"), 10) || 0;
8246             }
8247         },
8248
8249         /**
8250          * Gets the right X coordinate of the element (element X position + element width)
8251          * @param {Boolean} local True to get the local css position instead of page coordinate
8252          * @return {Number}
8253          */
8254         getRight : function(local){
8255             if(!local){
8256                 return this.getX() + this.getWidth();
8257             }else{
8258                 return (this.getLeft(true) + this.getWidth()) || 0;
8259             }
8260         },
8261
8262         /**
8263          * Gets the top Y coordinate
8264          * @param {Boolean} local True to get the local css position instead of page coordinate
8265          * @return {Number}
8266          */
8267         getTop : function(local) {
8268             if(!local){
8269                 return this.getY();
8270             }else{
8271                 return parseInt(this.getStyle("top"), 10) || 0;
8272             }
8273         },
8274
8275         /**
8276          * Gets the bottom Y coordinate of the element (element Y position + element height)
8277          * @param {Boolean} local True to get the local css position instead of page coordinate
8278          * @return {Number}
8279          */
8280         getBottom : function(local){
8281             if(!local){
8282                 return this.getY() + this.getHeight();
8283             }else{
8284                 return (this.getTop(true) + this.getHeight()) || 0;
8285             }
8286         },
8287
8288         /**
8289         * Initializes positioning on this element. If a desired position is not passed, it will make the
8290         * the element positioned relative IF it is not already positioned.
8291         * @param {String} pos (optional) Positioning to use "relative", "absolute" or "fixed"
8292         * @param {Number} zIndex (optional) The zIndex to apply
8293         * @param {Number} x (optional) Set the page X position
8294         * @param {Number} y (optional) Set the page Y position
8295         */
8296         position : function(pos, zIndex, x, y){
8297             if(!pos){
8298                if(this.getStyle('position') == 'static'){
8299                    this.setStyle('position', 'relative');
8300                }
8301             }else{
8302                 this.setStyle("position", pos);
8303             }
8304             if(zIndex){
8305                 this.setStyle("z-index", zIndex);
8306             }
8307             if(x !== undefined && y !== undefined){
8308                 this.setXY([x, y]);
8309             }else if(x !== undefined){
8310                 this.setX(x);
8311             }else if(y !== undefined){
8312                 this.setY(y);
8313             }
8314         },
8315
8316         /**
8317         * Clear positioning back to the default when the document was loaded
8318         * @param {String} value (optional) The value to use for the left,right,top,bottom, defaults to '' (empty string). You could use 'auto'.
8319         * @return {Roo.Element} this
8320          */
8321         clearPositioning : function(value){
8322             value = value ||'';
8323             this.setStyle({
8324                 "left": value,
8325                 "right": value,
8326                 "top": value,
8327                 "bottom": value,
8328                 "z-index": "",
8329                 "position" : "static"
8330             });
8331             return this;
8332         },
8333
8334         /**
8335         * Gets an object with all CSS positioning properties. Useful along with setPostioning to get
8336         * snapshot before performing an update and then restoring the element.
8337         * @return {Object}
8338         */
8339         getPositioning : function(){
8340             var l = this.getStyle("left");
8341             var t = this.getStyle("top");
8342             return {
8343                 "position" : this.getStyle("position"),
8344                 "left" : l,
8345                 "right" : l ? "" : this.getStyle("right"),
8346                 "top" : t,
8347                 "bottom" : t ? "" : this.getStyle("bottom"),
8348                 "z-index" : this.getStyle("z-index")
8349             };
8350         },
8351
8352         /**
8353          * Gets the width of the border(s) for the specified side(s)
8354          * @param {String} side Can be t, l, r, b or any combination of those to add multiple values. For example,
8355          * passing lr would get the border (l)eft width + the border (r)ight width.
8356          * @return {Number} The width of the sides passed added together
8357          */
8358         getBorderWidth : function(side){
8359             return this.addStyles(side, El.borders);
8360         },
8361
8362         /**
8363          * Gets the width of the padding(s) for the specified side(s)
8364          * @param {String} side Can be t, l, r, b or any combination of those to add multiple values. For example,
8365          * passing lr would get the padding (l)eft + the padding (r)ight.
8366          * @return {Number} The padding of the sides passed added together
8367          */
8368         getPadding : function(side){
8369             return this.addStyles(side, El.paddings);
8370         },
8371
8372         /**
8373         * Set positioning with an object returned by getPositioning().
8374         * @param {Object} posCfg
8375         * @return {Roo.Element} this
8376          */
8377         setPositioning : function(pc){
8378             this.applyStyles(pc);
8379             if(pc.right == "auto"){
8380                 this.dom.style.right = "";
8381             }
8382             if(pc.bottom == "auto"){
8383                 this.dom.style.bottom = "";
8384             }
8385             return this;
8386         },
8387
8388         // private
8389         fixDisplay : function(){
8390             if(this.getStyle("display") == "none"){
8391                 this.setStyle("visibility", "hidden");
8392                 this.setStyle("display", this.originalDisplay); // first try reverting to default
8393                 if(this.getStyle("display") == "none"){ // if that fails, default to block
8394                     this.setStyle("display", "block");
8395                 }
8396             }
8397         },
8398
8399         /**
8400          * Quick set left and top adding default units
8401          * @param {String} left The left CSS property value
8402          * @param {String} top The top CSS property value
8403          * @return {Roo.Element} this
8404          */
8405          setLeftTop : function(left, top){
8406             this.dom.style.left = this.addUnits(left);
8407             this.dom.style.top = this.addUnits(top);
8408             return this;
8409         },
8410
8411         /**
8412          * Move this element relative to its current position.
8413          * @param {String} direction Possible values are: "l","left" - "r","right" - "t","top","up" - "b","bottom","down".
8414          * @param {Number} distance How far to move the element in pixels
8415          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8416          * @return {Roo.Element} this
8417          */
8418          move : function(direction, distance, animate){
8419             var xy = this.getXY();
8420             direction = direction.toLowerCase();
8421             switch(direction){
8422                 case "l":
8423                 case "left":
8424                     this.moveTo(xy[0]-distance, xy[1], this.preanim(arguments, 2));
8425                     break;
8426                case "r":
8427                case "right":
8428                     this.moveTo(xy[0]+distance, xy[1], this.preanim(arguments, 2));
8429                     break;
8430                case "t":
8431                case "top":
8432                case "up":
8433                     this.moveTo(xy[0], xy[1]-distance, this.preanim(arguments, 2));
8434                     break;
8435                case "b":
8436                case "bottom":
8437                case "down":
8438                     this.moveTo(xy[0], xy[1]+distance, this.preanim(arguments, 2));
8439                     break;
8440             }
8441             return this;
8442         },
8443
8444         /**
8445          *  Store the current overflow setting and clip overflow on the element - use {@link #unclip} to remove
8446          * @return {Roo.Element} this
8447          */
8448         clip : function(){
8449             if(!this.isClipped){
8450                this.isClipped = true;
8451                this.originalClip = {
8452                    "o": this.getStyle("overflow"),
8453                    "x": this.getStyle("overflow-x"),
8454                    "y": this.getStyle("overflow-y")
8455                };
8456                this.setStyle("overflow", "hidden");
8457                this.setStyle("overflow-x", "hidden");
8458                this.setStyle("overflow-y", "hidden");
8459             }
8460             return this;
8461         },
8462
8463         /**
8464          *  Return clipping (overflow) to original clipping before clip() was called
8465          * @return {Roo.Element} this
8466          */
8467         unclip : function(){
8468             if(this.isClipped){
8469                 this.isClipped = false;
8470                 var o = this.originalClip;
8471                 if(o.o){this.setStyle("overflow", o.o);}
8472                 if(o.x){this.setStyle("overflow-x", o.x);}
8473                 if(o.y){this.setStyle("overflow-y", o.y);}
8474             }
8475             return this;
8476         },
8477
8478
8479         /**
8480          * Gets the x,y coordinates specified by the anchor position on the element.
8481          * @param {String} anchor (optional) The specified anchor position (defaults to "c").  See {@link #alignTo} for details on supported anchor positions.
8482          * @param {Object} size (optional) An object containing the size to use for calculating anchor position
8483          *                       {width: (target width), height: (target height)} (defaults to the element's current size)
8484          * @param {Boolean} local (optional) True to get the local (element top/left-relative) anchor position instead of page coordinates
8485          * @return {Array} [x, y] An array containing the element's x and y coordinates
8486          */
8487         getAnchorXY : function(anchor, local, s){
8488             //Passing a different size is useful for pre-calculating anchors,
8489             //especially for anchored animations that change the el size.
8490
8491             var w, h, vp = false;
8492             if(!s){
8493                 var d = this.dom;
8494                 if(d == document.body || d == document){
8495                     vp = true;
8496                     w = D.getViewWidth(); h = D.getViewHeight();
8497                 }else{
8498                     w = this.getWidth(); h = this.getHeight();
8499                 }
8500             }else{
8501                 w = s.width;  h = s.height;
8502             }
8503             var x = 0, y = 0, r = Math.round;
8504             switch((anchor || "tl").toLowerCase()){
8505                 case "c":
8506                     x = r(w*.5);
8507                     y = r(h*.5);
8508                 break;
8509                 case "t":
8510                     x = r(w*.5);
8511                     y = 0;
8512                 break;
8513                 case "l":
8514                     x = 0;
8515                     y = r(h*.5);
8516                 break;
8517                 case "r":
8518                     x = w;
8519                     y = r(h*.5);
8520                 break;
8521                 case "b":
8522                     x = r(w*.5);
8523                     y = h;
8524                 break;
8525                 case "tl":
8526                     x = 0;
8527                     y = 0;
8528                 break;
8529                 case "bl":
8530                     x = 0;
8531                     y = h;
8532                 break;
8533                 case "br":
8534                     x = w;
8535                     y = h;
8536                 break;
8537                 case "tr":
8538                     x = w;
8539                     y = 0;
8540                 break;
8541             }
8542             if(local === true){
8543                 return [x, y];
8544             }
8545             if(vp){
8546                 var sc = this.getScroll();
8547                 return [x + sc.left, y + sc.top];
8548             }
8549             //Add the element's offset xy
8550             var o = this.getXY();
8551             return [x+o[0], y+o[1]];
8552         },
8553
8554         /**
8555          * Gets the x,y coordinates to align this element with another element. See {@link #alignTo} for more info on the
8556          * supported position values.
8557          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8558          * @param {String} position The position to align to.
8559          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8560          * @return {Array} [x, y]
8561          */
8562         getAlignToXY : function(el, p, o){
8563             el = Roo.get(el);
8564             var d = this.dom;
8565             if(!el.dom){
8566                 throw "Element.alignTo with an element that doesn't exist";
8567             }
8568             var c = false; //constrain to viewport
8569             var p1 = "", p2 = "";
8570             o = o || [0,0];
8571
8572             if(!p){
8573                 p = "tl-bl";
8574             }else if(p == "?"){
8575                 p = "tl-bl?";
8576             }else if(p.indexOf("-") == -1){
8577                 p = "tl-" + p;
8578             }
8579             p = p.toLowerCase();
8580             var m = p.match(/^([a-z]+)-([a-z]+)(\?)?$/);
8581             if(!m){
8582                throw "Element.alignTo with an invalid alignment " + p;
8583             }
8584             p1 = m[1]; p2 = m[2]; c = !!m[3];
8585
8586             //Subtract the aligned el's internal xy from the target's offset xy
8587             //plus custom offset to get the aligned el's new offset xy
8588             var a1 = this.getAnchorXY(p1, true);
8589             var a2 = el.getAnchorXY(p2, false);
8590             var x = a2[0] - a1[0] + o[0];
8591             var y = a2[1] - a1[1] + o[1];
8592             if(c){
8593                 //constrain the aligned el to viewport if necessary
8594                 var w = this.getWidth(), h = this.getHeight(), r = el.getRegion();
8595                 // 5px of margin for ie
8596                 var dw = D.getViewWidth()-5, dh = D.getViewHeight()-5;
8597
8598                 //If we are at a viewport boundary and the aligned el is anchored on a target border that is
8599                 //perpendicular to the vp border, allow the aligned el to slide on that border,
8600                 //otherwise swap the aligned el to the opposite border of the target.
8601                 var p1y = p1.charAt(0), p1x = p1.charAt(p1.length-1);
8602                var p2y = p2.charAt(0), p2x = p2.charAt(p2.length-1);
8603                var swapY = ((p1y=="t" && p2y=="b") || (p1y=="b" && p2y=="t"));
8604                var swapX = ((p1x=="r" && p2x=="l") || (p1x=="l" && p2x=="r"));
8605
8606                var doc = document;
8607                var scrollX = (doc.documentElement.scrollLeft || doc.body.scrollLeft || 0)+5;
8608                var scrollY = (doc.documentElement.scrollTop || doc.body.scrollTop || 0)+5;
8609
8610                if((x+w) > dw + scrollX){
8611                     x = swapX ? r.left-w : dw+scrollX-w;
8612                 }
8613                if(x < scrollX){
8614                    x = swapX ? r.right : scrollX;
8615                }
8616                if((y+h) > dh + scrollY){
8617                     y = swapY ? r.top-h : dh+scrollY-h;
8618                 }
8619                if (y < scrollY){
8620                    y = swapY ? r.bottom : scrollY;
8621                }
8622             }
8623             return [x,y];
8624         },
8625
8626         // private
8627         getConstrainToXY : function(){
8628             var os = {top:0, left:0, bottom:0, right: 0};
8629
8630             return function(el, local, offsets, proposedXY){
8631                 el = Roo.get(el);
8632                 offsets = offsets ? Roo.applyIf(offsets, os) : os;
8633
8634                 var vw, vh, vx = 0, vy = 0;
8635                 if(el.dom == document.body || el.dom == document){
8636                     vw = Roo.lib.Dom.getViewWidth();
8637                     vh = Roo.lib.Dom.getViewHeight();
8638                 }else{
8639                     vw = el.dom.clientWidth;
8640                     vh = el.dom.clientHeight;
8641                     if(!local){
8642                         var vxy = el.getXY();
8643                         vx = vxy[0];
8644                         vy = vxy[1];
8645                     }
8646                 }
8647
8648                 var s = el.getScroll();
8649
8650                 vx += offsets.left + s.left;
8651                 vy += offsets.top + s.top;
8652
8653                 vw -= offsets.right;
8654                 vh -= offsets.bottom;
8655
8656                 var vr = vx+vw;
8657                 var vb = vy+vh;
8658
8659                 var xy = proposedXY || (!local ? this.getXY() : [this.getLeft(true), this.getTop(true)]);
8660                 var x = xy[0], y = xy[1];
8661                 var w = this.dom.offsetWidth, h = this.dom.offsetHeight;
8662
8663                 // only move it if it needs it
8664                 var moved = false;
8665
8666                 // first validate right/bottom
8667                 if((x + w) > vr){
8668                     x = vr - w;
8669                     moved = true;
8670                 }
8671                 if((y + h) > vb){
8672                     y = vb - h;
8673                     moved = true;
8674                 }
8675                 // then make sure top/left isn't negative
8676                 if(x < vx){
8677                     x = vx;
8678                     moved = true;
8679                 }
8680                 if(y < vy){
8681                     y = vy;
8682                     moved = true;
8683                 }
8684                 return moved ? [x, y] : false;
8685             };
8686         }(),
8687
8688         // private
8689         adjustForConstraints : function(xy, parent, offsets){
8690             return this.getConstrainToXY(parent || document, false, offsets, xy) ||  xy;
8691         },
8692
8693         /**
8694          * Aligns this element with another element relative to the specified anchor points. If the other element is the
8695          * document it aligns it to the viewport.
8696          * The position parameter is optional, and can be specified in any one of the following formats:
8697          * <ul>
8698          *   <li><b>Blank</b>: Defaults to aligning the element's top-left corner to the target's bottom-left corner ("tl-bl").</li>
8699          *   <li><b>One anchor (deprecated)</b>: The passed anchor position is used as the target element's anchor point.
8700          *       The element being aligned will position its top-left corner (tl) to that point.  <i>This method has been
8701          *       deprecated in favor of the newer two anchor syntax below</i>.</li>
8702          *   <li><b>Two anchors</b>: If two values from the table below are passed separated by a dash, the first value is used as the
8703          *       element's anchor point, and the second value is used as the target's anchor point.</li>
8704          * </ul>
8705          * In addition to the anchor points, the position parameter also supports the "?" character.  If "?" is passed at the end of
8706          * the position string, the element will attempt to align as specified, but the position will be adjusted to constrain to
8707          * the viewport if necessary.  Note that the element being aligned might be swapped to align to a different position than
8708          * that specified in order to enforce the viewport constraints.
8709          * Following are all of the supported anchor positions:
8710     <pre>
8711     Value  Description
8712     -----  -----------------------------
8713     tl     The top left corner (default)
8714     t      The center of the top edge
8715     tr     The top right corner
8716     l      The center of the left edge
8717     c      In the center of the element
8718     r      The center of the right edge
8719     bl     The bottom left corner
8720     b      The center of the bottom edge
8721     br     The bottom right corner
8722     </pre>
8723     Example Usage:
8724     <pre><code>
8725     // align el to other-el using the default positioning ("tl-bl", non-constrained)
8726     el.alignTo("other-el");
8727
8728     // align the top left corner of el with the top right corner of other-el (constrained to viewport)
8729     el.alignTo("other-el", "tr?");
8730
8731     // align the bottom right corner of el with the center left edge of other-el
8732     el.alignTo("other-el", "br-l?");
8733
8734     // align the center of el with the bottom left corner of other-el and
8735     // adjust the x position by -6 pixels (and the y position by 0)
8736     el.alignTo("other-el", "c-bl", [-6, 0]);
8737     </code></pre>
8738          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8739          * @param {String} position The position to align to.
8740          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8741          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8742          * @return {Roo.Element} this
8743          */
8744         alignTo : function(element, position, offsets, animate){
8745             var xy = this.getAlignToXY(element, position, offsets);
8746             this.setXY(xy, this.preanim(arguments, 3));
8747             return this;
8748         },
8749
8750         /**
8751          * Anchors an element to another element and realigns it when the window is resized.
8752          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8753          * @param {String} position The position to align to.
8754          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8755          * @param {Boolean/Object} animate (optional) True for the default animation or a standard Element animation config object
8756          * @param {Boolean/Number} monitorScroll (optional) True to monitor body scroll and reposition. If this parameter
8757          * is a number, it is used as the buffer delay (defaults to 50ms).
8758          * @param {Function} callback The function to call after the animation finishes
8759          * @return {Roo.Element} this
8760          */
8761         anchorTo : function(el, alignment, offsets, animate, monitorScroll, callback){
8762             var action = function(){
8763                 this.alignTo(el, alignment, offsets, animate);
8764                 Roo.callback(callback, this);
8765             };
8766             Roo.EventManager.onWindowResize(action, this);
8767             var tm = typeof monitorScroll;
8768             if(tm != 'undefined'){
8769                 Roo.EventManager.on(window, 'scroll', action, this,
8770                     {buffer: tm == 'number' ? monitorScroll : 50});
8771             }
8772             action.call(this); // align immediately
8773             return this;
8774         },
8775         /**
8776          * Clears any opacity settings from this element. Required in some cases for IE.
8777          * @return {Roo.Element} this
8778          */
8779         clearOpacity : function(){
8780             if (window.ActiveXObject) {
8781                 if(typeof this.dom.style.filter == 'string' && (/alpha/i).test(this.dom.style.filter)){
8782                     this.dom.style.filter = "";
8783                 }
8784             } else {
8785                 this.dom.style.opacity = "";
8786                 this.dom.style["-moz-opacity"] = "";
8787                 this.dom.style["-khtml-opacity"] = "";
8788             }
8789             return this;
8790         },
8791
8792         /**
8793          * Hide this element - Uses display mode to determine whether to use "display" or "visibility". See {@link #setVisible}.
8794          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8795          * @return {Roo.Element} this
8796          */
8797         hide : function(animate){
8798             this.setVisible(false, this.preanim(arguments, 0));
8799             return this;
8800         },
8801
8802         /**
8803         * Show this element - Uses display mode to determine whether to use "display" or "visibility". See {@link #setVisible}.
8804         * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8805          * @return {Roo.Element} this
8806          */
8807         show : function(animate){
8808             this.setVisible(true, this.preanim(arguments, 0));
8809             return this;
8810         },
8811
8812         /**
8813          * @private Test if size has a unit, otherwise appends the default
8814          */
8815         addUnits : function(size){
8816             return Roo.Element.addUnits(size, this.defaultUnit);
8817         },
8818
8819         /**
8820          * Temporarily enables offsets (width,height,x,y) for an element with display:none, use endMeasure() when done.
8821          * @return {Roo.Element} this
8822          */
8823         beginMeasure : function(){
8824             var el = this.dom;
8825             if(el.offsetWidth || el.offsetHeight){
8826                 return this; // offsets work already
8827             }
8828             var changed = [];
8829             var p = this.dom, b = document.body; // start with this element
8830             while((!el.offsetWidth && !el.offsetHeight) && p && p.tagName && p != b){
8831                 var pe = Roo.get(p);
8832                 if(pe.getStyle('display') == 'none'){
8833                     changed.push({el: p, visibility: pe.getStyle("visibility")});
8834                     p.style.visibility = "hidden";
8835                     p.style.display = "block";
8836                 }
8837                 p = p.parentNode;
8838             }
8839             this._measureChanged = changed;
8840             return this;
8841
8842         },
8843
8844         /**
8845          * Restores displays to before beginMeasure was called
8846          * @return {Roo.Element} this
8847          */
8848         endMeasure : function(){
8849             var changed = this._measureChanged;
8850             if(changed){
8851                 for(var i = 0, len = changed.length; i < len; i++) {
8852                     var r = changed[i];
8853                     r.el.style.visibility = r.visibility;
8854                     r.el.style.display = "none";
8855                 }
8856                 this._measureChanged = null;
8857             }
8858             return this;
8859         },
8860
8861         /**
8862         * Update the innerHTML of this element, optionally searching for and processing scripts
8863         * @param {String} html The new HTML
8864         * @param {Boolean} loadScripts (optional) true to look for and process scripts
8865         * @param {Function} callback For async script loading you can be noticed when the update completes
8866         * @return {Roo.Element} this
8867          */
8868         update : function(html, loadScripts, callback){
8869             if(typeof html == "undefined"){
8870                 html = "";
8871             }
8872             if(loadScripts !== true){
8873                 this.dom.innerHTML = html;
8874                 if(typeof callback == "function"){
8875                     callback();
8876                 }
8877                 return this;
8878             }
8879             var id = Roo.id();
8880             var dom = this.dom;
8881
8882             html += '<span id="' + id + '"></span>';
8883
8884             E.onAvailable(id, function(){
8885                 var hd = document.getElementsByTagName("head")[0];
8886                 var re = /(?:<script([^>]*)?>)((\n|\r|.)*?)(?:<\/script>)/ig;
8887                 var srcRe = /\ssrc=([\'\"])(.*?)\1/i;
8888                 var typeRe = /\stype=([\'\"])(.*?)\1/i;
8889
8890                 var match;
8891                 while(match = re.exec(html)){
8892                     var attrs = match[1];
8893                     var srcMatch = attrs ? attrs.match(srcRe) : false;
8894                     if(srcMatch && srcMatch[2]){
8895                        var s = document.createElement("script");
8896                        s.src = srcMatch[2];
8897                        var typeMatch = attrs.match(typeRe);
8898                        if(typeMatch && typeMatch[2]){
8899                            s.type = typeMatch[2];
8900                        }
8901                        hd.appendChild(s);
8902                     }else if(match[2] && match[2].length > 0){
8903                         if(window.execScript) {
8904                            window.execScript(match[2]);
8905                         } else {
8906                             /**
8907                              * eval:var:id
8908                              * eval:var:dom
8909                              * eval:var:html
8910                              * 
8911                              */
8912                            window.eval(match[2]);
8913                         }
8914                     }
8915                 }
8916                 var el = document.getElementById(id);
8917                 if(el){el.parentNode.removeChild(el);}
8918                 if(typeof callback == "function"){
8919                     callback();
8920                 }
8921             });
8922             dom.innerHTML = html.replace(/(?:<script.*?>)((\n|\r|.)*?)(?:<\/script>)/ig, "");
8923             return this;
8924         },
8925
8926         /**
8927          * Direct access to the UpdateManager update() method (takes the same parameters).
8928          * @param {String/Function} url The url for this request or a function to call to get the url
8929          * @param {String/Object} params (optional) The parameters to pass as either a url encoded string "param1=1&amp;param2=2" or an object {param1: 1, param2: 2}
8930          * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
8931          * @param {Boolean} discardUrl (optional) By default when you execute an update the defaultUrl is changed to the last used url. If true, it will not store the url.
8932          * @return {Roo.Element} this
8933          */
8934         load : function(){
8935             var um = this.getUpdateManager();
8936             um.update.apply(um, arguments);
8937             return this;
8938         },
8939
8940         /**
8941         * Gets this element's UpdateManager
8942         * @return {Roo.UpdateManager} The UpdateManager
8943         */
8944         getUpdateManager : function(){
8945             if(!this.updateManager){
8946                 this.updateManager = new Roo.UpdateManager(this);
8947             }
8948             return this.updateManager;
8949         },
8950
8951         /**
8952          * Disables text selection for this element (normalized across browsers)
8953          * @return {Roo.Element} this
8954          */
8955         unselectable : function(){
8956             this.dom.unselectable = "on";
8957             this.swallowEvent("selectstart", true);
8958             this.applyStyles("-moz-user-select:none;-khtml-user-select:none;");
8959             this.addClass("x-unselectable");
8960             return this;
8961         },
8962
8963         /**
8964         * Calculates the x, y to center this element on the screen
8965         * @return {Array} The x, y values [x, y]
8966         */
8967         getCenterXY : function(){
8968             return this.getAlignToXY(document, 'c-c');
8969         },
8970
8971         /**
8972         * Centers the Element in either the viewport, or another Element.
8973         * @param {String/HTMLElement/Roo.Element} centerIn (optional) The element in which to center the element.
8974         */
8975         center : function(centerIn){
8976             this.alignTo(centerIn || document, 'c-c');
8977             return this;
8978         },
8979
8980         /**
8981          * Tests various css rules/browsers to determine if this element uses a border box
8982          * @return {Boolean}
8983          */
8984         isBorderBox : function(){
8985             return noBoxAdjust[this.dom.tagName.toLowerCase()] || Roo.isBorderBox;
8986         },
8987
8988         /**
8989          * Return a box {x, y, width, height} that can be used to set another elements
8990          * size/location to match this element.
8991          * @param {Boolean} contentBox (optional) If true a box for the content of the element is returned.
8992          * @param {Boolean} local (optional) If true the element's left and top are returned instead of page x/y.
8993          * @return {Object} box An object in the format {x, y, width, height}
8994          */
8995         getBox : function(contentBox, local){
8996             var xy;
8997             if(!local){
8998                 xy = this.getXY();
8999             }else{
9000                 var left = parseInt(this.getStyle("left"), 10) || 0;
9001                 var top = parseInt(this.getStyle("top"), 10) || 0;
9002                 xy = [left, top];
9003             }
9004             var el = this.dom, w = el.offsetWidth, h = el.offsetHeight, bx;
9005             if(!contentBox){
9006                 bx = {x: xy[0], y: xy[1], 0: xy[0], 1: xy[1], width: w, height: h};
9007             }else{
9008                 var l = this.getBorderWidth("l")+this.getPadding("l");
9009                 var r = this.getBorderWidth("r")+this.getPadding("r");
9010                 var t = this.getBorderWidth("t")+this.getPadding("t");
9011                 var b = this.getBorderWidth("b")+this.getPadding("b");
9012                 bx = {x: xy[0]+l, y: xy[1]+t, 0: xy[0]+l, 1: xy[1]+t, width: w-(l+r), height: h-(t+b)};
9013             }
9014             bx.right = bx.x + bx.width;
9015             bx.bottom = bx.y + bx.height;
9016             return bx;
9017         },
9018
9019         /**
9020          * Returns the sum width of the padding and borders for the passed "sides". See getBorderWidth()
9021          for more information about the sides.
9022          * @param {String} sides
9023          * @return {Number}
9024          */
9025         getFrameWidth : function(sides, onlyContentBox){
9026             return onlyContentBox && Roo.isBorderBox ? 0 : (this.getPadding(sides) + this.getBorderWidth(sides));
9027         },
9028
9029         /**
9030          * Sets the element's box. Use getBox() on another element to get a box obj. If animate is true then width, height, x and y will be animated concurrently.
9031          * @param {Object} box The box to fill {x, y, width, height}
9032          * @param {Boolean} adjust (optional) Whether to adjust for box-model issues automatically
9033          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9034          * @return {Roo.Element} this
9035          */
9036         setBox : function(box, adjust, animate){
9037             var w = box.width, h = box.height;
9038             if((adjust && !this.autoBoxAdjust) && !this.isBorderBox()){
9039                w -= (this.getBorderWidth("lr") + this.getPadding("lr"));
9040                h -= (this.getBorderWidth("tb") + this.getPadding("tb"));
9041             }
9042             this.setBounds(box.x, box.y, w, h, this.preanim(arguments, 2));
9043             return this;
9044         },
9045
9046         /**
9047          * Forces the browser to repaint this element
9048          * @return {Roo.Element} this
9049          */
9050          repaint : function(){
9051             var dom = this.dom;
9052             this.addClass("x-repaint");
9053             setTimeout(function(){
9054                 Roo.get(dom).removeClass("x-repaint");
9055             }, 1);
9056             return this;
9057         },
9058
9059         /**
9060          * Returns an object with properties top, left, right and bottom representing the margins of this element unless sides is passed,
9061          * then it returns the calculated width of the sides (see getPadding)
9062          * @param {String} sides (optional) Any combination of l, r, t, b to get the sum of those sides
9063          * @return {Object/Number}
9064          */
9065         getMargins : function(side){
9066             if(!side){
9067                 return {
9068                     top: parseInt(this.getStyle("margin-top"), 10) || 0,
9069                     left: parseInt(this.getStyle("margin-left"), 10) || 0,
9070                     bottom: parseInt(this.getStyle("margin-bottom"), 10) || 0,
9071                     right: parseInt(this.getStyle("margin-right"), 10) || 0
9072                 };
9073             }else{
9074                 return this.addStyles(side, El.margins);
9075              }
9076         },
9077
9078         // private
9079         addStyles : function(sides, styles){
9080             var val = 0, v, w;
9081             for(var i = 0, len = sides.length; i < len; i++){
9082                 v = this.getStyle(styles[sides.charAt(i)]);
9083                 if(v){
9084                      w = parseInt(v, 10);
9085                      if(w){ val += w; }
9086                 }
9087             }
9088             return val;
9089         },
9090
9091         /**
9092          * Creates a proxy element of this element
9093          * @param {String/Object} config The class name of the proxy element or a DomHelper config object
9094          * @param {String/HTMLElement} renderTo (optional) The element or element id to render the proxy to (defaults to document.body)
9095          * @param {Boolean} matchBox (optional) True to align and size the proxy to this element now (defaults to false)
9096          * @return {Roo.Element} The new proxy element
9097          */
9098         createProxy : function(config, renderTo, matchBox){
9099             if(renderTo){
9100                 renderTo = Roo.getDom(renderTo);
9101             }else{
9102                 renderTo = document.body;
9103             }
9104             config = typeof config == "object" ?
9105                 config : {tag : "div", cls: config};
9106             var proxy = Roo.DomHelper.append(renderTo, config, true);
9107             if(matchBox){
9108                proxy.setBox(this.getBox());
9109             }
9110             return proxy;
9111         },
9112
9113         /**
9114          * Puts a mask over this element to disable user interaction. Requires core.css.
9115          * This method can only be applied to elements which accept child nodes.
9116          * @param {String} msg (optional) A message to display in the mask
9117          * @param {String} msgCls (optional) A css class to apply to the msg element
9118          * @return {Element} The mask  element
9119          */
9120         mask : function(msg, msgCls)
9121         {
9122             if(this.getStyle("position") == "static" && this.dom.tagName !== 'BODY'){
9123                 this.setStyle("position", "relative");
9124             }
9125             if(!this._mask){
9126                 this._mask = Roo.DomHelper.append(this.dom, {cls:"roo-el-mask"}, true);
9127             }
9128             
9129             this.addClass("x-masked");
9130             this._mask.setDisplayed(true);
9131             
9132             // we wander
9133             var z = 0;
9134             var dom = this.dom;
9135             while (dom && dom.style) {
9136                 if (!isNaN(parseInt(dom.style.zIndex))) {
9137                     z = Math.max(z, parseInt(dom.style.zIndex));
9138                 }
9139                 dom = dom.parentNode;
9140             }
9141             // if we are masking the body - then it hides everything..
9142             if (this.dom == document.body) {
9143                 z = 1000000;
9144                 this._mask.setWidth(Roo.lib.Dom.getDocumentWidth());
9145                 this._mask.setHeight(Roo.lib.Dom.getDocumentHeight());
9146             }
9147            
9148             if(typeof msg == 'string'){
9149                 if(!this._maskMsg){
9150                     this._maskMsg = Roo.DomHelper.append(this.dom, {
9151                         cls: "roo-el-mask-msg", 
9152                         cn: [
9153                             {
9154                                 tag: 'i',
9155                                 cls: 'fa fa-spinner fa-spin'
9156                             },
9157                             {
9158                                 tag: 'div'
9159                             }   
9160                         ]
9161                     }, true);
9162                 }
9163                 var mm = this._maskMsg;
9164                 mm.dom.className = msgCls ? "roo-el-mask-msg " + msgCls : "roo-el-mask-msg";
9165                 if (mm.dom.lastChild) { // weird IE issue?
9166                     mm.dom.lastChild.innerHTML = msg;
9167                 }
9168                 mm.setDisplayed(true);
9169                 mm.center(this);
9170                 mm.setStyle('z-index', z + 102);
9171             }
9172             if(Roo.isIE && !(Roo.isIE7 && Roo.isStrict) && this.getStyle('height') == 'auto'){ // ie will not expand full height automatically
9173                 this._mask.setHeight(this.getHeight());
9174             }
9175             this._mask.setStyle('z-index', z + 100);
9176             
9177             return this._mask;
9178         },
9179
9180         /**
9181          * Removes a previously applied mask. If removeEl is true the mask overlay is destroyed, otherwise
9182          * it is cached for reuse.
9183          */
9184         unmask : function(removeEl){
9185             if(this._mask){
9186                 if(removeEl === true){
9187                     this._mask.remove();
9188                     delete this._mask;
9189                     if(this._maskMsg){
9190                         this._maskMsg.remove();
9191                         delete this._maskMsg;
9192                     }
9193                 }else{
9194                     this._mask.setDisplayed(false);
9195                     if(this._maskMsg){
9196                         this._maskMsg.setDisplayed(false);
9197                     }
9198                 }
9199             }
9200             this.removeClass("x-masked");
9201         },
9202
9203         /**
9204          * Returns true if this element is masked
9205          * @return {Boolean}
9206          */
9207         isMasked : function(){
9208             return this._mask && this._mask.isVisible();
9209         },
9210
9211         /**
9212          * Creates an iframe shim for this element to keep selects and other windowed objects from
9213          * showing through.
9214          * @return {Roo.Element} The new shim element
9215          */
9216         createShim : function(){
9217             var el = document.createElement('iframe');
9218             el.frameBorder = 'no';
9219             el.className = 'roo-shim';
9220             if(Roo.isIE && Roo.isSecure){
9221                 el.src = Roo.SSL_SECURE_URL;
9222             }
9223             var shim = Roo.get(this.dom.parentNode.insertBefore(el, this.dom));
9224             shim.autoBoxAdjust = false;
9225             return shim;
9226         },
9227
9228         /**
9229          * Removes this element from the DOM and deletes it from the cache
9230          */
9231         remove : function(){
9232             if(this.dom.parentNode){
9233                 this.dom.parentNode.removeChild(this.dom);
9234             }
9235             delete El.cache[this.dom.id];
9236         },
9237
9238         /**
9239          * Sets up event handlers to add and remove a css class when the mouse is over this element
9240          * @param {String} className
9241          * @param {Boolean} preventFlicker (optional) If set to true, it prevents flickering by filtering
9242          * mouseout events for children elements
9243          * @return {Roo.Element} this
9244          */
9245         addClassOnOver : function(className, preventFlicker){
9246             this.on("mouseover", function(){
9247                 Roo.fly(this, '_internal').addClass(className);
9248             }, this.dom);
9249             var removeFn = function(e){
9250                 if(preventFlicker !== true || !e.within(this, true)){
9251                     Roo.fly(this, '_internal').removeClass(className);
9252                 }
9253             };
9254             this.on("mouseout", removeFn, this.dom);
9255             return this;
9256         },
9257
9258         /**
9259          * Sets up event handlers to add and remove a css class when this element has the focus
9260          * @param {String} className
9261          * @return {Roo.Element} this
9262          */
9263         addClassOnFocus : function(className){
9264             this.on("focus", function(){
9265                 Roo.fly(this, '_internal').addClass(className);
9266             }, this.dom);
9267             this.on("blur", function(){
9268                 Roo.fly(this, '_internal').removeClass(className);
9269             }, this.dom);
9270             return this;
9271         },
9272         /**
9273          * Sets up event handlers to add and remove a css class when the mouse is down and then up on this element (a click effect)
9274          * @param {String} className
9275          * @return {Roo.Element} this
9276          */
9277         addClassOnClick : function(className){
9278             var dom = this.dom;
9279             this.on("mousedown", function(){
9280                 Roo.fly(dom, '_internal').addClass(className);
9281                 var d = Roo.get(document);
9282                 var fn = function(){
9283                     Roo.fly(dom, '_internal').removeClass(className);
9284                     d.removeListener("mouseup", fn);
9285                 };
9286                 d.on("mouseup", fn);
9287             });
9288             return this;
9289         },
9290
9291         /**
9292          * Stops the specified event from bubbling and optionally prevents the default action
9293          * @param {String} eventName
9294          * @param {Boolean} preventDefault (optional) true to prevent the default action too
9295          * @return {Roo.Element} this
9296          */
9297         swallowEvent : function(eventName, preventDefault){
9298             var fn = function(e){
9299                 e.stopPropagation();
9300                 if(preventDefault){
9301                     e.preventDefault();
9302                 }
9303             };
9304             if(eventName instanceof Array){
9305                 for(var i = 0, len = eventName.length; i < len; i++){
9306                      this.on(eventName[i], fn);
9307                 }
9308                 return this;
9309             }
9310             this.on(eventName, fn);
9311             return this;
9312         },
9313
9314         /**
9315          * @private
9316          */
9317       fitToParentDelegate : Roo.emptyFn, // keep a reference to the fitToParent delegate
9318
9319         /**
9320          * Sizes this element to its parent element's dimensions performing
9321          * neccessary box adjustments.
9322          * @param {Boolean} monitorResize (optional) If true maintains the fit when the browser window is resized.
9323          * @param {String/HTMLElment/Element} targetParent (optional) The target parent, default to the parentNode.
9324          * @return {Roo.Element} this
9325          */
9326         fitToParent : function(monitorResize, targetParent) {
9327           Roo.EventManager.removeResizeListener(this.fitToParentDelegate); // always remove previous fitToParent delegate from onWindowResize
9328           this.fitToParentDelegate = Roo.emptyFn; // remove reference to previous delegate
9329           if (monitorResize === true && !this.dom.parentNode) { // check if this Element still exists
9330             return;
9331           }
9332           var p = Roo.get(targetParent || this.dom.parentNode);
9333           this.setSize(p.getComputedWidth() - p.getFrameWidth('lr'), p.getComputedHeight() - p.getFrameWidth('tb'));
9334           if (monitorResize === true) {
9335             this.fitToParentDelegate = this.fitToParent.createDelegate(this, [true, targetParent]);
9336             Roo.EventManager.onWindowResize(this.fitToParentDelegate);
9337           }
9338           return this;
9339         },
9340
9341         /**
9342          * Gets the next sibling, skipping text nodes
9343          * @return {HTMLElement} The next sibling or null
9344          */
9345         getNextSibling : function(){
9346             var n = this.dom.nextSibling;
9347             while(n && n.nodeType != 1){
9348                 n = n.nextSibling;
9349             }
9350             return n;
9351         },
9352
9353         /**
9354          * Gets the previous sibling, skipping text nodes
9355          * @return {HTMLElement} The previous sibling or null
9356          */
9357         getPrevSibling : function(){
9358             var n = this.dom.previousSibling;
9359             while(n && n.nodeType != 1){
9360                 n = n.previousSibling;
9361             }
9362             return n;
9363         },
9364
9365
9366         /**
9367          * Appends the passed element(s) to this element
9368          * @param {String/HTMLElement/Array/Element/CompositeElement} el
9369          * @return {Roo.Element} this
9370          */
9371         appendChild: function(el){
9372             el = Roo.get(el);
9373             el.appendTo(this);
9374             return this;
9375         },
9376
9377         /**
9378          * Creates the passed DomHelper config and appends it to this element or optionally inserts it before the passed child element.
9379          * @param {Object} config DomHelper element config object.  If no tag is specified (e.g., {tag:'input'}) then a div will be
9380          * automatically generated with the specified attributes.
9381          * @param {HTMLElement} insertBefore (optional) a child element of this element
9382          * @param {Boolean} returnDom (optional) true to return the dom node instead of creating an Element
9383          * @return {Roo.Element} The new child element
9384          */
9385         createChild: function(config, insertBefore, returnDom){
9386             config = config || {tag:'div'};
9387             if(insertBefore){
9388                 return Roo.DomHelper.insertBefore(insertBefore, config, returnDom !== true);
9389             }
9390             return Roo.DomHelper[!this.dom.firstChild ? 'overwrite' : 'append'](this.dom, config,  returnDom !== true);
9391         },
9392
9393         /**
9394          * Appends this element to the passed element
9395          * @param {String/HTMLElement/Element} el The new parent element
9396          * @return {Roo.Element} this
9397          */
9398         appendTo: function(el){
9399             el = Roo.getDom(el);
9400             el.appendChild(this.dom);
9401             return this;
9402         },
9403
9404         /**
9405          * Inserts this element before the passed element in the DOM
9406          * @param {String/HTMLElement/Element} el The element to insert before
9407          * @return {Roo.Element} this
9408          */
9409         insertBefore: function(el){
9410             el = Roo.getDom(el);
9411             el.parentNode.insertBefore(this.dom, el);
9412             return this;
9413         },
9414
9415         /**
9416          * Inserts this element after the passed element in the DOM
9417          * @param {String/HTMLElement/Element} el The element to insert after
9418          * @return {Roo.Element} this
9419          */
9420         insertAfter: function(el){
9421             el = Roo.getDom(el);
9422             el.parentNode.insertBefore(this.dom, el.nextSibling);
9423             return this;
9424         },
9425
9426         /**
9427          * Inserts (or creates) an element (or DomHelper config) as the first child of the this element
9428          * @param {String/HTMLElement/Element/Object} el The id or element to insert or a DomHelper config to create and insert
9429          * @return {Roo.Element} The new child
9430          */
9431         insertFirst: function(el, returnDom){
9432             el = el || {};
9433             if(typeof el == 'object' && !el.nodeType){ // dh config
9434                 return this.createChild(el, this.dom.firstChild, returnDom);
9435             }else{
9436                 el = Roo.getDom(el);
9437                 this.dom.insertBefore(el, this.dom.firstChild);
9438                 return !returnDom ? Roo.get(el) : el;
9439             }
9440         },
9441
9442         /**
9443          * Inserts (or creates) the passed element (or DomHelper config) as a sibling of this element
9444          * @param {String/HTMLElement/Element/Object} el The id or element to insert or a DomHelper config to create and insert
9445          * @param {String} where (optional) 'before' or 'after' defaults to before
9446          * @param {Boolean} returnDom (optional) True to return the raw DOM element instead of Roo.Element
9447          * @return {Roo.Element} the inserted Element
9448          */
9449         insertSibling: function(el, where, returnDom){
9450             where = where ? where.toLowerCase() : 'before';
9451             el = el || {};
9452             var rt, refNode = where == 'before' ? this.dom : this.dom.nextSibling;
9453
9454             if(typeof el == 'object' && !el.nodeType){ // dh config
9455                 if(where == 'after' && !this.dom.nextSibling){
9456                     rt = Roo.DomHelper.append(this.dom.parentNode, el, !returnDom);
9457                 }else{
9458                     rt = Roo.DomHelper[where == 'after' ? 'insertAfter' : 'insertBefore'](this.dom, el, !returnDom);
9459                 }
9460
9461             }else{
9462                 rt = this.dom.parentNode.insertBefore(Roo.getDom(el),
9463                             where == 'before' ? this.dom : this.dom.nextSibling);
9464                 if(!returnDom){
9465                     rt = Roo.get(rt);
9466                 }
9467             }
9468             return rt;
9469         },
9470
9471         /**
9472          * Creates and wraps this element with another element
9473          * @param {Object} config (optional) DomHelper element config object for the wrapper element or null for an empty div
9474          * @param {Boolean} returnDom (optional) True to return the raw DOM element instead of Roo.Element
9475          * @return {HTMLElement/Element} The newly created wrapper element
9476          */
9477         wrap: function(config, returnDom){
9478             if(!config){
9479                 config = {tag: "div"};
9480             }
9481             var newEl = Roo.DomHelper.insertBefore(this.dom, config, !returnDom);
9482             newEl.dom ? newEl.dom.appendChild(this.dom) : newEl.appendChild(this.dom);
9483             return newEl;
9484         },
9485
9486         /**
9487          * Replaces the passed element with this element
9488          * @param {String/HTMLElement/Element} el The element to replace
9489          * @return {Roo.Element} this
9490          */
9491         replace: function(el){
9492             el = Roo.get(el);
9493             this.insertBefore(el);
9494             el.remove();
9495             return this;
9496         },
9497
9498         /**
9499          * Inserts an html fragment into this element
9500          * @param {String} where Where to insert the html in relation to the this element - beforeBegin, afterBegin, beforeEnd, afterEnd.
9501          * @param {String} html The HTML fragment
9502          * @param {Boolean} returnEl True to return an Roo.Element
9503          * @return {HTMLElement/Roo.Element} The inserted node (or nearest related if more than 1 inserted)
9504          */
9505         insertHtml : function(where, html, returnEl){
9506             var el = Roo.DomHelper.insertHtml(where, this.dom, html);
9507             return returnEl ? Roo.get(el) : el;
9508         },
9509
9510         /**
9511          * Sets the passed attributes as attributes of this element (a style attribute can be a string, object or function)
9512          * @param {Object} o The object with the attributes
9513          * @param {Boolean} useSet (optional) false to override the default setAttribute to use expandos.
9514          * @return {Roo.Element} this
9515          */
9516         set : function(o, useSet){
9517             var el = this.dom;
9518             useSet = typeof useSet == 'undefined' ? (el.setAttribute ? true : false) : useSet;
9519             for(var attr in o){
9520                 if(attr == "style" || typeof o[attr] == "function")  { continue; }
9521                 if(attr=="cls"){
9522                     el.className = o["cls"];
9523                 }else{
9524                     if(useSet) {
9525                         el.setAttribute(attr, o[attr]);
9526                     } else {
9527                         el[attr] = o[attr];
9528                     }
9529                 }
9530             }
9531             if(o.style){
9532                 Roo.DomHelper.applyStyles(el, o.style);
9533             }
9534             return this;
9535         },
9536
9537         /**
9538          * Convenience method for constructing a KeyMap
9539          * @param {Number/Array/Object/String} key Either a string with the keys to listen for, the numeric key code, array of key codes or an object with the following options:
9540          *                                  {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
9541          * @param {Function} fn The function to call
9542          * @param {Object} scope (optional) The scope of the function
9543          * @return {Roo.KeyMap} The KeyMap created
9544          */
9545         addKeyListener : function(key, fn, scope){
9546             var config;
9547             if(typeof key != "object" || key instanceof Array){
9548                 config = {
9549                     key: key,
9550                     fn: fn,
9551                     scope: scope
9552                 };
9553             }else{
9554                 config = {
9555                     key : key.key,
9556                     shift : key.shift,
9557                     ctrl : key.ctrl,
9558                     alt : key.alt,
9559                     fn: fn,
9560                     scope: scope
9561                 };
9562             }
9563             return new Roo.KeyMap(this, config);
9564         },
9565
9566         /**
9567          * Creates a KeyMap for this element
9568          * @param {Object} config The KeyMap config. See {@link Roo.KeyMap} for more details
9569          * @return {Roo.KeyMap} The KeyMap created
9570          */
9571         addKeyMap : function(config){
9572             return new Roo.KeyMap(this, config);
9573         },
9574
9575         /**
9576          * Returns true if this element is scrollable.
9577          * @return {Boolean}
9578          */
9579          isScrollable : function(){
9580             var dom = this.dom;
9581             return dom.scrollHeight > dom.clientHeight || dom.scrollWidth > dom.clientWidth;
9582         },
9583
9584         /**
9585          * Scrolls this element the specified scroll point. It does NOT do bounds checking so if you scroll to a weird value it will try to do it. For auto bounds checking, use scroll().
9586          * @param {String} side Either "left" for scrollLeft values or "top" for scrollTop values.
9587          * @param {Number} value The new scroll value
9588          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9589          * @return {Element} this
9590          */
9591
9592         scrollTo : function(side, value, animate){
9593             var prop = side.toLowerCase() == "left" ? "scrollLeft" : "scrollTop";
9594             if(!animate || !A){
9595                 this.dom[prop] = value;
9596             }else{
9597                 var to = prop == "scrollLeft" ? [value, this.dom.scrollTop] : [this.dom.scrollLeft, value];
9598                 this.anim({scroll: {"to": to}}, this.preanim(arguments, 2), 'scroll');
9599             }
9600             return this;
9601         },
9602
9603         /**
9604          * Scrolls this element the specified direction. Does bounds checking to make sure the scroll is
9605          * within this element's scrollable range.
9606          * @param {String} direction Possible values are: "l","left" - "r","right" - "t","top","up" - "b","bottom","down".
9607          * @param {Number} distance How far to scroll the element in pixels
9608          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9609          * @return {Boolean} Returns true if a scroll was triggered or false if the element
9610          * was scrolled as far as it could go.
9611          */
9612          scroll : function(direction, distance, animate){
9613              if(!this.isScrollable()){
9614                  return;
9615              }
9616              var el = this.dom;
9617              var l = el.scrollLeft, t = el.scrollTop;
9618              var w = el.scrollWidth, h = el.scrollHeight;
9619              var cw = el.clientWidth, ch = el.clientHeight;
9620              direction = direction.toLowerCase();
9621              var scrolled = false;
9622              var a = this.preanim(arguments, 2);
9623              switch(direction){
9624                  case "l":
9625                  case "left":
9626                      if(w - l > cw){
9627                          var v = Math.min(l + distance, w-cw);
9628                          this.scrollTo("left", v, a);
9629                          scrolled = true;
9630                      }
9631                      break;
9632                 case "r":
9633                 case "right":
9634                      if(l > 0){
9635                          var v = Math.max(l - distance, 0);
9636                          this.scrollTo("left", v, a);
9637                          scrolled = true;
9638                      }
9639                      break;
9640                 case "t":
9641                 case "top":
9642                 case "up":
9643                      if(t > 0){
9644                          var v = Math.max(t - distance, 0);
9645                          this.scrollTo("top", v, a);
9646                          scrolled = true;
9647                      }
9648                      break;
9649                 case "b":
9650                 case "bottom":
9651                 case "down":
9652                      if(h - t > ch){
9653                          var v = Math.min(t + distance, h-ch);
9654                          this.scrollTo("top", v, a);
9655                          scrolled = true;
9656                      }
9657                      break;
9658              }
9659              return scrolled;
9660         },
9661
9662         /**
9663          * Translates the passed page coordinates into left/top css values for this element
9664          * @param {Number/Array} x The page x or an array containing [x, y]
9665          * @param {Number} y The page y
9666          * @return {Object} An object with left and top properties. e.g. {left: (value), top: (value)}
9667          */
9668         translatePoints : function(x, y){
9669             if(typeof x == 'object' || x instanceof Array){
9670                 y = x[1]; x = x[0];
9671             }
9672             var p = this.getStyle('position');
9673             var o = this.getXY();
9674
9675             var l = parseInt(this.getStyle('left'), 10);
9676             var t = parseInt(this.getStyle('top'), 10);
9677
9678             if(isNaN(l)){
9679                 l = (p == "relative") ? 0 : this.dom.offsetLeft;
9680             }
9681             if(isNaN(t)){
9682                 t = (p == "relative") ? 0 : this.dom.offsetTop;
9683             }
9684
9685             return {left: (x - o[0] + l), top: (y - o[1] + t)};
9686         },
9687
9688         /**
9689          * Returns the current scroll position of the element.
9690          * @return {Object} An object containing the scroll position in the format {left: (scrollLeft), top: (scrollTop)}
9691          */
9692         getScroll : function(){
9693             var d = this.dom, doc = document;
9694             if(d == doc || d == doc.body){
9695                 var l = window.pageXOffset || doc.documentElement.scrollLeft || doc.body.scrollLeft || 0;
9696                 var t = window.pageYOffset || doc.documentElement.scrollTop || doc.body.scrollTop || 0;
9697                 return {left: l, top: t};
9698             }else{
9699                 return {left: d.scrollLeft, top: d.scrollTop};
9700             }
9701         },
9702
9703         /**
9704          * Return the CSS color for the specified CSS attribute. rgb, 3 digit (like #fff) and valid values
9705          * are convert to standard 6 digit hex color.
9706          * @param {String} attr The css attribute
9707          * @param {String} defaultValue The default value to use when a valid color isn't found
9708          * @param {String} prefix (optional) defaults to #. Use an empty string when working with
9709          * YUI color anims.
9710          */
9711         getColor : function(attr, defaultValue, prefix){
9712             var v = this.getStyle(attr);
9713             if(!v || v == "transparent" || v == "inherit") {
9714                 return defaultValue;
9715             }
9716             var color = typeof prefix == "undefined" ? "#" : prefix;
9717             if(v.substr(0, 4) == "rgb("){
9718                 var rvs = v.slice(4, v.length -1).split(",");
9719                 for(var i = 0; i < 3; i++){
9720                     var h = parseInt(rvs[i]).toString(16);
9721                     if(h < 16){
9722                         h = "0" + h;
9723                     }
9724                     color += h;
9725                 }
9726             } else {
9727                 if(v.substr(0, 1) == "#"){
9728                     if(v.length == 4) {
9729                         for(var i = 1; i < 4; i++){
9730                             var c = v.charAt(i);
9731                             color +=  c + c;
9732                         }
9733                     }else if(v.length == 7){
9734                         color += v.substr(1);
9735                     }
9736                 }
9737             }
9738             return(color.length > 5 ? color.toLowerCase() : defaultValue);
9739         },
9740
9741         /**
9742          * Wraps the specified element with a special markup/CSS block that renders by default as a gray container with a
9743          * gradient background, rounded corners and a 4-way shadow.
9744          * @param {String} class (optional) A base CSS class to apply to the containing wrapper element (defaults to 'x-box').
9745          * Note that there are a number of CSS rules that are dependent on this name to make the overall effect work,
9746          * so if you supply an alternate base class, make sure you also supply all of the necessary rules.
9747          * @return {Roo.Element} this
9748          */
9749         boxWrap : function(cls){
9750             cls = cls || 'x-box';
9751             var el = Roo.get(this.insertHtml('beforeBegin', String.format('<div class="{0}">'+El.boxMarkup+'</div>', cls)));
9752             el.child('.'+cls+'-mc').dom.appendChild(this.dom);
9753             return el;
9754         },
9755
9756         /**
9757          * Returns the value of a namespaced attribute from the element's underlying DOM node.
9758          * @param {String} namespace The namespace in which to look for the attribute
9759          * @param {String} name The attribute name
9760          * @return {String} The attribute value
9761          */
9762         getAttributeNS : Roo.isIE ? function(ns, name){
9763             var d = this.dom;
9764             var type = typeof d[ns+":"+name];
9765             if(type != 'undefined' && type != 'unknown'){
9766                 return d[ns+":"+name];
9767             }
9768             return d[name];
9769         } : function(ns, name){
9770             var d = this.dom;
9771             return d.getAttributeNS(ns, name) || d.getAttribute(ns+":"+name) || d.getAttribute(name) || d[name];
9772         },
9773         
9774         
9775         /**
9776          * Sets or Returns the value the dom attribute value
9777          * @param {String|Object} name The attribute name (or object to set multiple attributes)
9778          * @param {String} value (optional) The value to set the attribute to
9779          * @return {String} The attribute value
9780          */
9781         attr : function(name){
9782             if (arguments.length > 1) {
9783                 this.dom.setAttribute(name, arguments[1]);
9784                 return arguments[1];
9785             }
9786             if (typeof(name) == 'object') {
9787                 for(var i in name) {
9788                     this.attr(i, name[i]);
9789                 }
9790                 return name;
9791             }
9792             
9793             
9794             if (!this.dom.hasAttribute(name)) {
9795                 return undefined;
9796             }
9797             return this.dom.getAttribute(name);
9798         }
9799         
9800         
9801         
9802     };
9803
9804     var ep = El.prototype;
9805
9806     /**
9807      * Appends an event handler (Shorthand for addListener)
9808      * @param {String}   eventName     The type of event to append
9809      * @param {Function} fn        The method the event invokes
9810      * @param {Object} scope       (optional) The scope (this object) of the fn
9811      * @param {Object}   options   (optional)An object with standard {@link Roo.EventManager#addListener} options
9812      * @method
9813      */
9814     ep.on = ep.addListener;
9815         // backwards compat
9816     ep.mon = ep.addListener;
9817
9818     /**
9819      * Removes an event handler from this element (shorthand for removeListener)
9820      * @param {String} eventName the type of event to remove
9821      * @param {Function} fn the method the event invokes
9822      * @return {Roo.Element} this
9823      * @method
9824      */
9825     ep.un = ep.removeListener;
9826
9827     /**
9828      * true to automatically adjust width and height settings for box-model issues (default to true)
9829      */
9830     ep.autoBoxAdjust = true;
9831
9832     // private
9833     El.unitPattern = /\d+(px|em|%|en|ex|pt|in|cm|mm|pc)$/i;
9834
9835     // private
9836     El.addUnits = function(v, defaultUnit){
9837         if(v === "" || v == "auto"){
9838             return v;
9839         }
9840         if(v === undefined){
9841             return '';
9842         }
9843         if(typeof v == "number" || !El.unitPattern.test(v)){
9844             return v + (defaultUnit || 'px');
9845         }
9846         return v;
9847     };
9848
9849     // special markup used throughout Roo when box wrapping elements
9850     El.boxMarkup = '<div class="{0}-tl"><div class="{0}-tr"><div class="{0}-tc"></div></div></div><div class="{0}-ml"><div class="{0}-mr"><div class="{0}-mc"></div></div></div><div class="{0}-bl"><div class="{0}-br"><div class="{0}-bc"></div></div></div>';
9851     /**
9852      * Visibility mode constant - Use visibility to hide element
9853      * @static
9854      * @type Number
9855      */
9856     El.VISIBILITY = 1;
9857     /**
9858      * Visibility mode constant - Use display to hide element
9859      * @static
9860      * @type Number
9861      */
9862     El.DISPLAY = 2;
9863
9864     El.borders = {l: "border-left-width", r: "border-right-width", t: "border-top-width", b: "border-bottom-width"};
9865     El.paddings = {l: "padding-left", r: "padding-right", t: "padding-top", b: "padding-bottom"};
9866     El.margins = {l: "margin-left", r: "margin-right", t: "margin-top", b: "margin-bottom"};
9867
9868
9869
9870     /**
9871      * @private
9872      */
9873     El.cache = {};
9874
9875     var docEl;
9876
9877     /**
9878      * Static method to retrieve Element objects. Uses simple caching to consistently return the same object.
9879      * Automatically fixes if an object was recreated with the same id via AJAX or DOM.
9880      * @param {String/HTMLElement/Element} el The id of the node, a DOM Node or an existing Element.
9881      * @return {Element} The Element object
9882      * @static
9883      */
9884     El.get = function(el){
9885         var ex, elm, id;
9886         if(!el){ return null; }
9887         if(typeof el == "string"){ // element id
9888             if(!(elm = document.getElementById(el))){
9889                 return null;
9890             }
9891             if(ex = El.cache[el]){
9892                 ex.dom = elm;
9893             }else{
9894                 ex = El.cache[el] = new El(elm);
9895             }
9896             return ex;
9897         }else if(el.tagName){ // dom element
9898             if(!(id = el.id)){
9899                 id = Roo.id(el);
9900             }
9901             if(ex = El.cache[id]){
9902                 ex.dom = el;
9903             }else{
9904                 ex = El.cache[id] = new El(el);
9905             }
9906             return ex;
9907         }else if(el instanceof El){
9908             if(el != docEl){
9909                 el.dom = document.getElementById(el.id) || el.dom; // refresh dom element in case no longer valid,
9910                                                               // catch case where it hasn't been appended
9911                 El.cache[el.id] = el; // in case it was created directly with Element(), let's cache it
9912             }
9913             return el;
9914         }else if(el.isComposite){
9915             return el;
9916         }else if(el instanceof Array){
9917             return El.select(el);
9918         }else if(el == document){
9919             // create a bogus element object representing the document object
9920             if(!docEl){
9921                 var f = function(){};
9922                 f.prototype = El.prototype;
9923                 docEl = new f();
9924                 docEl.dom = document;
9925             }
9926             return docEl;
9927         }
9928         return null;
9929     };
9930
9931     // private
9932     El.uncache = function(el){
9933         for(var i = 0, a = arguments, len = a.length; i < len; i++) {
9934             if(a[i]){
9935                 delete El.cache[a[i].id || a[i]];
9936             }
9937         }
9938     };
9939
9940     // private
9941     // Garbage collection - uncache elements/purge listeners on orphaned elements
9942     // so we don't hold a reference and cause the browser to retain them
9943     El.garbageCollect = function(){
9944         if(!Roo.enableGarbageCollector){
9945             clearInterval(El.collectorThread);
9946             return;
9947         }
9948         for(var eid in El.cache){
9949             var el = El.cache[eid], d = el.dom;
9950             // -------------------------------------------------------
9951             // Determining what is garbage:
9952             // -------------------------------------------------------
9953             // !d
9954             // dom node is null, definitely garbage
9955             // -------------------------------------------------------
9956             // !d.parentNode
9957             // no parentNode == direct orphan, definitely garbage
9958             // -------------------------------------------------------
9959             // !d.offsetParent && !document.getElementById(eid)
9960             // display none elements have no offsetParent so we will
9961             // also try to look it up by it's id. However, check
9962             // offsetParent first so we don't do unneeded lookups.
9963             // This enables collection of elements that are not orphans
9964             // directly, but somewhere up the line they have an orphan
9965             // parent.
9966             // -------------------------------------------------------
9967             if(!d || !d.parentNode || (!d.offsetParent && !document.getElementById(eid))){
9968                 delete El.cache[eid];
9969                 if(d && Roo.enableListenerCollection){
9970                     E.purgeElement(d);
9971                 }
9972             }
9973         }
9974     }
9975     El.collectorThreadId = setInterval(El.garbageCollect, 30000);
9976
9977
9978     // dom is optional
9979     El.Flyweight = function(dom){
9980         this.dom = dom;
9981     };
9982     El.Flyweight.prototype = El.prototype;
9983
9984     El._flyweights = {};
9985     /**
9986      * Gets the globally shared flyweight Element, with the passed node as the active element. Do not store a reference to this element -
9987      * the dom node can be overwritten by other code.
9988      * @param {String/HTMLElement} el The dom node or id
9989      * @param {String} named (optional) Allows for creation of named reusable flyweights to
9990      *                                  prevent conflicts (e.g. internally Roo uses "_internal")
9991      * @static
9992      * @return {Element} The shared Element object
9993      */
9994     El.fly = function(el, named){
9995         named = named || '_global';
9996         el = Roo.getDom(el);
9997         if(!el){
9998             return null;
9999         }
10000         if(!El._flyweights[named]){
10001             El._flyweights[named] = new El.Flyweight();
10002         }
10003         El._flyweights[named].dom = el;
10004         return El._flyweights[named];
10005     };
10006
10007     /**
10008      * Static method to retrieve Element objects. Uses simple caching to consistently return the same object.
10009      * Automatically fixes if an object was recreated with the same id via AJAX or DOM.
10010      * Shorthand of {@link Roo.Element#get}
10011      * @param {String/HTMLElement/Element} el The id of the node, a DOM Node or an existing Element.
10012      * @return {Element} The Element object
10013      * @member Roo
10014      * @method get
10015      */
10016     Roo.get = El.get;
10017     /**
10018      * Gets the globally shared flyweight Element, with the passed node as the active element. Do not store a reference to this element -
10019      * the dom node can be overwritten by other code.
10020      * Shorthand of {@link Roo.Element#fly}
10021      * @param {String/HTMLElement} el The dom node or id
10022      * @param {String} named (optional) Allows for creation of named reusable flyweights to
10023      *                                  prevent conflicts (e.g. internally Roo uses "_internal")
10024      * @static
10025      * @return {Element} The shared Element object
10026      * @member Roo
10027      * @method fly
10028      */
10029     Roo.fly = El.fly;
10030
10031     // speedy lookup for elements never to box adjust
10032     var noBoxAdjust = Roo.isStrict ? {
10033         select:1
10034     } : {
10035         input:1, select:1, textarea:1
10036     };
10037     if(Roo.isIE || Roo.isGecko){
10038         noBoxAdjust['button'] = 1;
10039     }
10040
10041
10042     Roo.EventManager.on(window, 'unload', function(){
10043         delete El.cache;
10044         delete El._flyweights;
10045     });
10046 })();
10047
10048
10049
10050
10051 if(Roo.DomQuery){
10052     Roo.Element.selectorFunction = Roo.DomQuery.select;
10053 }
10054
10055 Roo.Element.select = function(selector, unique, root){
10056     var els;
10057     if(typeof selector == "string"){
10058         els = Roo.Element.selectorFunction(selector, root);
10059     }else if(selector.length !== undefined){
10060         els = selector;
10061     }else{
10062         throw "Invalid selector";
10063     }
10064     if(unique === true){
10065         return new Roo.CompositeElement(els);
10066     }else{
10067         return new Roo.CompositeElementLite(els);
10068     }
10069 };
10070 /**
10071  * Selects elements based on the passed CSS selector to enable working on them as 1.
10072  * @param {String/Array} selector The CSS selector or an array of elements
10073  * @param {Boolean} unique (optional) true to create a unique Roo.Element for each element (defaults to a shared flyweight object)
10074  * @param {HTMLElement/String} root (optional) The root element of the query or id of the root
10075  * @return {CompositeElementLite/CompositeElement}
10076  * @member Roo
10077  * @method select
10078  */
10079 Roo.select = Roo.Element.select;
10080
10081
10082
10083
10084
10085
10086
10087
10088
10089
10090
10091
10092
10093
10094 /*
10095  * Based on:
10096  * Ext JS Library 1.1.1
10097  * Copyright(c) 2006-2007, Ext JS, LLC.
10098  *
10099  * Originally Released Under LGPL - original licence link has changed is not relivant.
10100  *
10101  * Fork - LGPL
10102  * <script type="text/javascript">
10103  */
10104
10105
10106
10107 //Notifies Element that fx methods are available
10108 Roo.enableFx = true;
10109
10110 /**
10111  * @class Roo.Fx
10112  * <p>A class to provide basic animation and visual effects support.  <b>Note:</b> This class is automatically applied
10113  * to the {@link Roo.Element} interface when included, so all effects calls should be performed via Element.
10114  * Conversely, since the effects are not actually defined in Element, Roo.Fx <b>must</b> be included in order for the 
10115  * Element effects to work.</p><br/>
10116  *
10117  * <p>It is important to note that although the Fx methods and many non-Fx Element methods support "method chaining" in that
10118  * they return the Element object itself as the method return value, it is not always possible to mix the two in a single
10119  * method chain.  The Fx methods use an internal effects queue so that each effect can be properly timed and sequenced.
10120  * Non-Fx methods, on the other hand, have no such internal queueing and will always execute immediately.  For this reason,
10121  * while it may be possible to mix certain Fx and non-Fx method calls in a single chain, it may not always provide the
10122  * expected results and should be done with care.</p><br/>
10123  *
10124  * <p>Motion effects support 8-way anchoring, meaning that you can choose one of 8 different anchor points on the Element
10125  * that will serve as either the start or end point of the animation.  Following are all of the supported anchor positions:</p>
10126 <pre>
10127 Value  Description
10128 -----  -----------------------------
10129 tl     The top left corner
10130 t      The center of the top edge
10131 tr     The top right corner
10132 l      The center of the left edge
10133 r      The center of the right edge
10134 bl     The bottom left corner
10135 b      The center of the bottom edge
10136 br     The bottom right corner
10137 </pre>
10138  * <b>Although some Fx methods accept specific custom config parameters, the ones shown in the Config Options section
10139  * below are common options that can be passed to any Fx method.</b>
10140  * @cfg {Function} callback A function called when the effect is finished
10141  * @cfg {Object} scope The scope of the effect function
10142  * @cfg {String} easing A valid Easing value for the effect
10143  * @cfg {String} afterCls A css class to apply after the effect
10144  * @cfg {Number} duration The length of time (in seconds) that the effect should last
10145  * @cfg {Boolean} remove Whether the Element should be removed from the DOM and destroyed after the effect finishes
10146  * @cfg {Boolean} useDisplay Whether to use the <i>display</i> CSS property instead of <i>visibility</i> when hiding Elements (only applies to 
10147  * effects that end with the element being visually hidden, ignored otherwise)
10148  * @cfg {String/Object/Function} afterStyle A style specification string, e.g. "width:100px", or an object in the form {width:"100px"}, or
10149  * a function which returns such a specification that will be applied to the Element after the effect finishes
10150  * @cfg {Boolean} block Whether the effect should block other effects from queueing while it runs
10151  * @cfg {Boolean} concurrent Whether to allow subsequently-queued effects to run at the same time as the current effect, or to ensure that they run in sequence
10152  * @cfg {Boolean} stopFx Whether subsequent effects should be stopped and removed after the current effect finishes
10153  */
10154 Roo.Fx = {
10155         /**
10156          * Slides the element into view.  An anchor point can be optionally passed to set the point of
10157          * origin for the slide effect.  This function automatically handles wrapping the element with
10158          * a fixed-size container if needed.  See the Fx class overview for valid anchor point options.
10159          * Usage:
10160          *<pre><code>
10161 // default: slide the element in from the top
10162 el.slideIn();
10163
10164 // custom: slide the element in from the right with a 2-second duration
10165 el.slideIn('r', { duration: 2 });
10166
10167 // common config options shown with default values
10168 el.slideIn('t', {
10169     easing: 'easeOut',
10170     duration: .5
10171 });
10172 </code></pre>
10173          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to top: 't')
10174          * @param {Object} options (optional) Object literal with any of the Fx config options
10175          * @return {Roo.Element} The Element
10176          */
10177     slideIn : function(anchor, o){
10178         var el = this.getFxEl();
10179         o = o || {};
10180
10181         el.queueFx(o, function(){
10182
10183             anchor = anchor || "t";
10184
10185             // fix display to visibility
10186             this.fixDisplay();
10187
10188             // restore values after effect
10189             var r = this.getFxRestore();
10190             var b = this.getBox();
10191             // fixed size for slide
10192             this.setSize(b);
10193
10194             // wrap if needed
10195             var wrap = this.fxWrap(r.pos, o, "hidden");
10196
10197             var st = this.dom.style;
10198             st.visibility = "visible";
10199             st.position = "absolute";
10200
10201             // clear out temp styles after slide and unwrap
10202             var after = function(){
10203                 el.fxUnwrap(wrap, r.pos, o);
10204                 st.width = r.width;
10205                 st.height = r.height;
10206                 el.afterFx(o);
10207             };
10208             // time to calc the positions
10209             var a, pt = {to: [b.x, b.y]}, bw = {to: b.width}, bh = {to: b.height};
10210
10211             switch(anchor.toLowerCase()){
10212                 case "t":
10213                     wrap.setSize(b.width, 0);
10214                     st.left = st.bottom = "0";
10215                     a = {height: bh};
10216                 break;
10217                 case "l":
10218                     wrap.setSize(0, b.height);
10219                     st.right = st.top = "0";
10220                     a = {width: bw};
10221                 break;
10222                 case "r":
10223                     wrap.setSize(0, b.height);
10224                     wrap.setX(b.right);
10225                     st.left = st.top = "0";
10226                     a = {width: bw, points: pt};
10227                 break;
10228                 case "b":
10229                     wrap.setSize(b.width, 0);
10230                     wrap.setY(b.bottom);
10231                     st.left = st.top = "0";
10232                     a = {height: bh, points: pt};
10233                 break;
10234                 case "tl":
10235                     wrap.setSize(0, 0);
10236                     st.right = st.bottom = "0";
10237                     a = {width: bw, height: bh};
10238                 break;
10239                 case "bl":
10240                     wrap.setSize(0, 0);
10241                     wrap.setY(b.y+b.height);
10242                     st.right = st.top = "0";
10243                     a = {width: bw, height: bh, points: pt};
10244                 break;
10245                 case "br":
10246                     wrap.setSize(0, 0);
10247                     wrap.setXY([b.right, b.bottom]);
10248                     st.left = st.top = "0";
10249                     a = {width: bw, height: bh, points: pt};
10250                 break;
10251                 case "tr":
10252                     wrap.setSize(0, 0);
10253                     wrap.setX(b.x+b.width);
10254                     st.left = st.bottom = "0";
10255                     a = {width: bw, height: bh, points: pt};
10256                 break;
10257             }
10258             this.dom.style.visibility = "visible";
10259             wrap.show();
10260
10261             arguments.callee.anim = wrap.fxanim(a,
10262                 o,
10263                 'motion',
10264                 .5,
10265                 'easeOut', after);
10266         });
10267         return this;
10268     },
10269     
10270         /**
10271          * Slides the element out of view.  An anchor point can be optionally passed to set the end point
10272          * for the slide effect.  When the effect is completed, the element will be hidden (visibility = 
10273          * 'hidden') but block elements will still take up space in the document.  The element must be removed
10274          * from the DOM using the 'remove' config option if desired.  This function automatically handles 
10275          * wrapping the element with a fixed-size container if needed.  See the Fx class overview for valid anchor point options.
10276          * Usage:
10277          *<pre><code>
10278 // default: slide the element out to the top
10279 el.slideOut();
10280
10281 // custom: slide the element out to the right with a 2-second duration
10282 el.slideOut('r', { duration: 2 });
10283
10284 // common config options shown with default values
10285 el.slideOut('t', {
10286     easing: 'easeOut',
10287     duration: .5,
10288     remove: false,
10289     useDisplay: false
10290 });
10291 </code></pre>
10292          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to top: 't')
10293          * @param {Object} options (optional) Object literal with any of the Fx config options
10294          * @return {Roo.Element} The Element
10295          */
10296     slideOut : function(anchor, o){
10297         var el = this.getFxEl();
10298         o = o || {};
10299
10300         el.queueFx(o, function(){
10301
10302             anchor = anchor || "t";
10303
10304             // restore values after effect
10305             var r = this.getFxRestore();
10306             
10307             var b = this.getBox();
10308             // fixed size for slide
10309             this.setSize(b);
10310
10311             // wrap if needed
10312             var wrap = this.fxWrap(r.pos, o, "visible");
10313
10314             var st = this.dom.style;
10315             st.visibility = "visible";
10316             st.position = "absolute";
10317
10318             wrap.setSize(b);
10319
10320             var after = function(){
10321                 if(o.useDisplay){
10322                     el.setDisplayed(false);
10323                 }else{
10324                     el.hide();
10325                 }
10326
10327                 el.fxUnwrap(wrap, r.pos, o);
10328
10329                 st.width = r.width;
10330                 st.height = r.height;
10331
10332                 el.afterFx(o);
10333             };
10334
10335             var a, zero = {to: 0};
10336             switch(anchor.toLowerCase()){
10337                 case "t":
10338                     st.left = st.bottom = "0";
10339                     a = {height: zero};
10340                 break;
10341                 case "l":
10342                     st.right = st.top = "0";
10343                     a = {width: zero};
10344                 break;
10345                 case "r":
10346                     st.left = st.top = "0";
10347                     a = {width: zero, points: {to:[b.right, b.y]}};
10348                 break;
10349                 case "b":
10350                     st.left = st.top = "0";
10351                     a = {height: zero, points: {to:[b.x, b.bottom]}};
10352                 break;
10353                 case "tl":
10354                     st.right = st.bottom = "0";
10355                     a = {width: zero, height: zero};
10356                 break;
10357                 case "bl":
10358                     st.right = st.top = "0";
10359                     a = {width: zero, height: zero, points: {to:[b.x, b.bottom]}};
10360                 break;
10361                 case "br":
10362                     st.left = st.top = "0";
10363                     a = {width: zero, height: zero, points: {to:[b.x+b.width, b.bottom]}};
10364                 break;
10365                 case "tr":
10366                     st.left = st.bottom = "0";
10367                     a = {width: zero, height: zero, points: {to:[b.right, b.y]}};
10368                 break;
10369             }
10370
10371             arguments.callee.anim = wrap.fxanim(a,
10372                 o,
10373                 'motion',
10374                 .5,
10375                 "easeOut", after);
10376         });
10377         return this;
10378     },
10379
10380         /**
10381          * Fades the element out while slowly expanding it in all directions.  When the effect is completed, the 
10382          * element will be hidden (visibility = 'hidden') but block elements will still take up space in the document. 
10383          * The element must be removed from the DOM using the 'remove' config option if desired.
10384          * Usage:
10385          *<pre><code>
10386 // default
10387 el.puff();
10388
10389 // common config options shown with default values
10390 el.puff({
10391     easing: 'easeOut',
10392     duration: .5,
10393     remove: false,
10394     useDisplay: false
10395 });
10396 </code></pre>
10397          * @param {Object} options (optional) Object literal with any of the Fx config options
10398          * @return {Roo.Element} The Element
10399          */
10400     puff : function(o){
10401         var el = this.getFxEl();
10402         o = o || {};
10403
10404         el.queueFx(o, function(){
10405             this.clearOpacity();
10406             this.show();
10407
10408             // restore values after effect
10409             var r = this.getFxRestore();
10410             var st = this.dom.style;
10411
10412             var after = function(){
10413                 if(o.useDisplay){
10414                     el.setDisplayed(false);
10415                 }else{
10416                     el.hide();
10417                 }
10418
10419                 el.clearOpacity();
10420
10421                 el.setPositioning(r.pos);
10422                 st.width = r.width;
10423                 st.height = r.height;
10424                 st.fontSize = '';
10425                 el.afterFx(o);
10426             };
10427
10428             var width = this.getWidth();
10429             var height = this.getHeight();
10430
10431             arguments.callee.anim = this.fxanim({
10432                     width : {to: this.adjustWidth(width * 2)},
10433                     height : {to: this.adjustHeight(height * 2)},
10434                     points : {by: [-(width * .5), -(height * .5)]},
10435                     opacity : {to: 0},
10436                     fontSize: {to:200, unit: "%"}
10437                 },
10438                 o,
10439                 'motion',
10440                 .5,
10441                 "easeOut", after);
10442         });
10443         return this;
10444     },
10445
10446         /**
10447          * Blinks the element as if it was clicked and then collapses on its center (similar to switching off a television).
10448          * When the effect is completed, the element will be hidden (visibility = 'hidden') but block elements will still 
10449          * take up space in the document. The element must be removed from the DOM using the 'remove' config option if desired.
10450          * Usage:
10451          *<pre><code>
10452 // default
10453 el.switchOff();
10454
10455 // all config options shown with default values
10456 el.switchOff({
10457     easing: 'easeIn',
10458     duration: .3,
10459     remove: false,
10460     useDisplay: false
10461 });
10462 </code></pre>
10463          * @param {Object} options (optional) Object literal with any of the Fx config options
10464          * @return {Roo.Element} The Element
10465          */
10466     switchOff : function(o){
10467         var el = this.getFxEl();
10468         o = o || {};
10469
10470         el.queueFx(o, function(){
10471             this.clearOpacity();
10472             this.clip();
10473
10474             // restore values after effect
10475             var r = this.getFxRestore();
10476             var st = this.dom.style;
10477
10478             var after = function(){
10479                 if(o.useDisplay){
10480                     el.setDisplayed(false);
10481                 }else{
10482                     el.hide();
10483                 }
10484
10485                 el.clearOpacity();
10486                 el.setPositioning(r.pos);
10487                 st.width = r.width;
10488                 st.height = r.height;
10489
10490                 el.afterFx(o);
10491             };
10492
10493             this.fxanim({opacity:{to:0.3}}, null, null, .1, null, function(){
10494                 this.clearOpacity();
10495                 (function(){
10496                     this.fxanim({
10497                         height:{to:1},
10498                         points:{by:[0, this.getHeight() * .5]}
10499                     }, o, 'motion', 0.3, 'easeIn', after);
10500                 }).defer(100, this);
10501             });
10502         });
10503         return this;
10504     },
10505
10506     /**
10507      * Highlights the Element by setting a color (applies to the background-color by default, but can be
10508      * changed using the "attr" config option) and then fading back to the original color. If no original
10509      * color is available, you should provide the "endColor" config option which will be cleared after the animation.
10510      * Usage:
10511 <pre><code>
10512 // default: highlight background to yellow
10513 el.highlight();
10514
10515 // custom: highlight foreground text to blue for 2 seconds
10516 el.highlight("0000ff", { attr: 'color', duration: 2 });
10517
10518 // common config options shown with default values
10519 el.highlight("ffff9c", {
10520     attr: "background-color", //can be any valid CSS property (attribute) that supports a color value
10521     endColor: (current color) or "ffffff",
10522     easing: 'easeIn',
10523     duration: 1
10524 });
10525 </code></pre>
10526      * @param {String} color (optional) The highlight color. Should be a 6 char hex color without the leading # (defaults to yellow: 'ffff9c')
10527      * @param {Object} options (optional) Object literal with any of the Fx config options
10528      * @return {Roo.Element} The Element
10529      */ 
10530     highlight : function(color, o){
10531         var el = this.getFxEl();
10532         o = o || {};
10533
10534         el.queueFx(o, function(){
10535             color = color || "ffff9c";
10536             attr = o.attr || "backgroundColor";
10537
10538             this.clearOpacity();
10539             this.show();
10540
10541             var origColor = this.getColor(attr);
10542             var restoreColor = this.dom.style[attr];
10543             endColor = (o.endColor || origColor) || "ffffff";
10544
10545             var after = function(){
10546                 el.dom.style[attr] = restoreColor;
10547                 el.afterFx(o);
10548             };
10549
10550             var a = {};
10551             a[attr] = {from: color, to: endColor};
10552             arguments.callee.anim = this.fxanim(a,
10553                 o,
10554                 'color',
10555                 1,
10556                 'easeIn', after);
10557         });
10558         return this;
10559     },
10560
10561    /**
10562     * Shows a ripple of exploding, attenuating borders to draw attention to an Element.
10563     * Usage:
10564 <pre><code>
10565 // default: a single light blue ripple
10566 el.frame();
10567
10568 // custom: 3 red ripples lasting 3 seconds total
10569 el.frame("ff0000", 3, { duration: 3 });
10570
10571 // common config options shown with default values
10572 el.frame("C3DAF9", 1, {
10573     duration: 1 //duration of entire animation (not each individual ripple)
10574     // Note: Easing is not configurable and will be ignored if included
10575 });
10576 </code></pre>
10577     * @param {String} color (optional) The color of the border.  Should be a 6 char hex color without the leading # (defaults to light blue: 'C3DAF9').
10578     * @param {Number} count (optional) The number of ripples to display (defaults to 1)
10579     * @param {Object} options (optional) Object literal with any of the Fx config options
10580     * @return {Roo.Element} The Element
10581     */
10582     frame : function(color, count, o){
10583         var el = this.getFxEl();
10584         o = o || {};
10585
10586         el.queueFx(o, function(){
10587             color = color || "#C3DAF9";
10588             if(color.length == 6){
10589                 color = "#" + color;
10590             }
10591             count = count || 1;
10592             duration = o.duration || 1;
10593             this.show();
10594
10595             var b = this.getBox();
10596             var animFn = function(){
10597                 var proxy = this.createProxy({
10598
10599                      style:{
10600                         visbility:"hidden",
10601                         position:"absolute",
10602                         "z-index":"35000", // yee haw
10603                         border:"0px solid " + color
10604                      }
10605                   });
10606                 var scale = Roo.isBorderBox ? 2 : 1;
10607                 proxy.animate({
10608                     top:{from:b.y, to:b.y - 20},
10609                     left:{from:b.x, to:b.x - 20},
10610                     borderWidth:{from:0, to:10},
10611                     opacity:{from:1, to:0},
10612                     height:{from:b.height, to:(b.height + (20*scale))},
10613                     width:{from:b.width, to:(b.width + (20*scale))}
10614                 }, duration, function(){
10615                     proxy.remove();
10616                 });
10617                 if(--count > 0){
10618                      animFn.defer((duration/2)*1000, this);
10619                 }else{
10620                     el.afterFx(o);
10621                 }
10622             };
10623             animFn.call(this);
10624         });
10625         return this;
10626     },
10627
10628    /**
10629     * Creates a pause before any subsequent queued effects begin.  If there are
10630     * no effects queued after the pause it will have no effect.
10631     * Usage:
10632 <pre><code>
10633 el.pause(1);
10634 </code></pre>
10635     * @param {Number} seconds The length of time to pause (in seconds)
10636     * @return {Roo.Element} The Element
10637     */
10638     pause : function(seconds){
10639         var el = this.getFxEl();
10640         var o = {};
10641
10642         el.queueFx(o, function(){
10643             setTimeout(function(){
10644                 el.afterFx(o);
10645             }, seconds * 1000);
10646         });
10647         return this;
10648     },
10649
10650    /**
10651     * Fade an element in (from transparent to opaque).  The ending opacity can be specified
10652     * using the "endOpacity" config option.
10653     * Usage:
10654 <pre><code>
10655 // default: fade in from opacity 0 to 100%
10656 el.fadeIn();
10657
10658 // custom: fade in from opacity 0 to 75% over 2 seconds
10659 el.fadeIn({ endOpacity: .75, duration: 2});
10660
10661 // common config options shown with default values
10662 el.fadeIn({
10663     endOpacity: 1, //can be any value between 0 and 1 (e.g. .5)
10664     easing: 'easeOut',
10665     duration: .5
10666 });
10667 </code></pre>
10668     * @param {Object} options (optional) Object literal with any of the Fx config options
10669     * @return {Roo.Element} The Element
10670     */
10671     fadeIn : function(o){
10672         var el = this.getFxEl();
10673         o = o || {};
10674         el.queueFx(o, function(){
10675             this.setOpacity(0);
10676             this.fixDisplay();
10677             this.dom.style.visibility = 'visible';
10678             var to = o.endOpacity || 1;
10679             arguments.callee.anim = this.fxanim({opacity:{to:to}},
10680                 o, null, .5, "easeOut", function(){
10681                 if(to == 1){
10682                     this.clearOpacity();
10683                 }
10684                 el.afterFx(o);
10685             });
10686         });
10687         return this;
10688     },
10689
10690    /**
10691     * Fade an element out (from opaque to transparent).  The ending opacity can be specified
10692     * using the "endOpacity" config option.
10693     * Usage:
10694 <pre><code>
10695 // default: fade out from the element's current opacity to 0
10696 el.fadeOut();
10697
10698 // custom: fade out from the element's current opacity to 25% over 2 seconds
10699 el.fadeOut({ endOpacity: .25, duration: 2});
10700
10701 // common config options shown with default values
10702 el.fadeOut({
10703     endOpacity: 0, //can be any value between 0 and 1 (e.g. .5)
10704     easing: 'easeOut',
10705     duration: .5
10706     remove: false,
10707     useDisplay: false
10708 });
10709 </code></pre>
10710     * @param {Object} options (optional) Object literal with any of the Fx config options
10711     * @return {Roo.Element} The Element
10712     */
10713     fadeOut : function(o){
10714         var el = this.getFxEl();
10715         o = o || {};
10716         el.queueFx(o, function(){
10717             arguments.callee.anim = this.fxanim({opacity:{to:o.endOpacity || 0}},
10718                 o, null, .5, "easeOut", function(){
10719                 if(this.visibilityMode == Roo.Element.DISPLAY || o.useDisplay){
10720                      this.dom.style.display = "none";
10721                 }else{
10722                      this.dom.style.visibility = "hidden";
10723                 }
10724                 this.clearOpacity();
10725                 el.afterFx(o);
10726             });
10727         });
10728         return this;
10729     },
10730
10731    /**
10732     * Animates the transition of an element's dimensions from a starting height/width
10733     * to an ending height/width.
10734     * Usage:
10735 <pre><code>
10736 // change height and width to 100x100 pixels
10737 el.scale(100, 100);
10738
10739 // common config options shown with default values.  The height and width will default to
10740 // the element's existing values if passed as null.
10741 el.scale(
10742     [element's width],
10743     [element's height], {
10744     easing: 'easeOut',
10745     duration: .35
10746 });
10747 </code></pre>
10748     * @param {Number} width  The new width (pass undefined to keep the original width)
10749     * @param {Number} height  The new height (pass undefined to keep the original height)
10750     * @param {Object} options (optional) Object literal with any of the Fx config options
10751     * @return {Roo.Element} The Element
10752     */
10753     scale : function(w, h, o){
10754         this.shift(Roo.apply({}, o, {
10755             width: w,
10756             height: h
10757         }));
10758         return this;
10759     },
10760
10761    /**
10762     * Animates the transition of any combination of an element's dimensions, xy position and/or opacity.
10763     * Any of these properties not specified in the config object will not be changed.  This effect 
10764     * requires that at least one new dimension, position or opacity setting must be passed in on
10765     * the config object in order for the function to have any effect.
10766     * Usage:
10767 <pre><code>
10768 // slide the element horizontally to x position 200 while changing the height and opacity
10769 el.shift({ x: 200, height: 50, opacity: .8 });
10770
10771 // common config options shown with default values.
10772 el.shift({
10773     width: [element's width],
10774     height: [element's height],
10775     x: [element's x position],
10776     y: [element's y position],
10777     opacity: [element's opacity],
10778     easing: 'easeOut',
10779     duration: .35
10780 });
10781 </code></pre>
10782     * @param {Object} options  Object literal with any of the Fx config options
10783     * @return {Roo.Element} The Element
10784     */
10785     shift : function(o){
10786         var el = this.getFxEl();
10787         o = o || {};
10788         el.queueFx(o, function(){
10789             var a = {}, w = o.width, h = o.height, x = o.x, y = o.y,  op = o.opacity;
10790             if(w !== undefined){
10791                 a.width = {to: this.adjustWidth(w)};
10792             }
10793             if(h !== undefined){
10794                 a.height = {to: this.adjustHeight(h)};
10795             }
10796             if(x !== undefined || y !== undefined){
10797                 a.points = {to: [
10798                     x !== undefined ? x : this.getX(),
10799                     y !== undefined ? y : this.getY()
10800                 ]};
10801             }
10802             if(op !== undefined){
10803                 a.opacity = {to: op};
10804             }
10805             if(o.xy !== undefined){
10806                 a.points = {to: o.xy};
10807             }
10808             arguments.callee.anim = this.fxanim(a,
10809                 o, 'motion', .35, "easeOut", function(){
10810                 el.afterFx(o);
10811             });
10812         });
10813         return this;
10814     },
10815
10816         /**
10817          * Slides the element while fading it out of view.  An anchor point can be optionally passed to set the 
10818          * ending point of the effect.
10819          * Usage:
10820          *<pre><code>
10821 // default: slide the element downward while fading out
10822 el.ghost();
10823
10824 // custom: slide the element out to the right with a 2-second duration
10825 el.ghost('r', { duration: 2 });
10826
10827 // common config options shown with default values
10828 el.ghost('b', {
10829     easing: 'easeOut',
10830     duration: .5
10831     remove: false,
10832     useDisplay: false
10833 });
10834 </code></pre>
10835          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to bottom: 'b')
10836          * @param {Object} options (optional) Object literal with any of the Fx config options
10837          * @return {Roo.Element} The Element
10838          */
10839     ghost : function(anchor, o){
10840         var el = this.getFxEl();
10841         o = o || {};
10842
10843         el.queueFx(o, function(){
10844             anchor = anchor || "b";
10845
10846             // restore values after effect
10847             var r = this.getFxRestore();
10848             var w = this.getWidth(),
10849                 h = this.getHeight();
10850
10851             var st = this.dom.style;
10852
10853             var after = function(){
10854                 if(o.useDisplay){
10855                     el.setDisplayed(false);
10856                 }else{
10857                     el.hide();
10858                 }
10859
10860                 el.clearOpacity();
10861                 el.setPositioning(r.pos);
10862                 st.width = r.width;
10863                 st.height = r.height;
10864
10865                 el.afterFx(o);
10866             };
10867
10868             var a = {opacity: {to: 0}, points: {}}, pt = a.points;
10869             switch(anchor.toLowerCase()){
10870                 case "t":
10871                     pt.by = [0, -h];
10872                 break;
10873                 case "l":
10874                     pt.by = [-w, 0];
10875                 break;
10876                 case "r":
10877                     pt.by = [w, 0];
10878                 break;
10879                 case "b":
10880                     pt.by = [0, h];
10881                 break;
10882                 case "tl":
10883                     pt.by = [-w, -h];
10884                 break;
10885                 case "bl":
10886                     pt.by = [-w, h];
10887                 break;
10888                 case "br":
10889                     pt.by = [w, h];
10890                 break;
10891                 case "tr":
10892                     pt.by = [w, -h];
10893                 break;
10894             }
10895
10896             arguments.callee.anim = this.fxanim(a,
10897                 o,
10898                 'motion',
10899                 .5,
10900                 "easeOut", after);
10901         });
10902         return this;
10903     },
10904
10905         /**
10906          * Ensures that all effects queued after syncFx is called on the element are
10907          * run concurrently.  This is the opposite of {@link #sequenceFx}.
10908          * @return {Roo.Element} The Element
10909          */
10910     syncFx : function(){
10911         this.fxDefaults = Roo.apply(this.fxDefaults || {}, {
10912             block : false,
10913             concurrent : true,
10914             stopFx : false
10915         });
10916         return this;
10917     },
10918
10919         /**
10920          * Ensures that all effects queued after sequenceFx is called on the element are
10921          * run in sequence.  This is the opposite of {@link #syncFx}.
10922          * @return {Roo.Element} The Element
10923          */
10924     sequenceFx : function(){
10925         this.fxDefaults = Roo.apply(this.fxDefaults || {}, {
10926             block : false,
10927             concurrent : false,
10928             stopFx : false
10929         });
10930         return this;
10931     },
10932
10933         /* @private */
10934     nextFx : function(){
10935         var ef = this.fxQueue[0];
10936         if(ef){
10937             ef.call(this);
10938         }
10939     },
10940
10941         /**
10942          * Returns true if the element has any effects actively running or queued, else returns false.
10943          * @return {Boolean} True if element has active effects, else false
10944          */
10945     hasActiveFx : function(){
10946         return this.fxQueue && this.fxQueue[0];
10947     },
10948
10949         /**
10950          * Stops any running effects and clears the element's internal effects queue if it contains
10951          * any additional effects that haven't started yet.
10952          * @return {Roo.Element} The Element
10953          */
10954     stopFx : function(){
10955         if(this.hasActiveFx()){
10956             var cur = this.fxQueue[0];
10957             if(cur && cur.anim && cur.anim.isAnimated()){
10958                 this.fxQueue = [cur]; // clear out others
10959                 cur.anim.stop(true);
10960             }
10961         }
10962         return this;
10963     },
10964
10965         /* @private */
10966     beforeFx : function(o){
10967         if(this.hasActiveFx() && !o.concurrent){
10968            if(o.stopFx){
10969                this.stopFx();
10970                return true;
10971            }
10972            return false;
10973         }
10974         return true;
10975     },
10976
10977         /**
10978          * Returns true if the element is currently blocking so that no other effect can be queued
10979          * until this effect is finished, else returns false if blocking is not set.  This is commonly
10980          * used to ensure that an effect initiated by a user action runs to completion prior to the
10981          * same effect being restarted (e.g., firing only one effect even if the user clicks several times).
10982          * @return {Boolean} True if blocking, else false
10983          */
10984     hasFxBlock : function(){
10985         var q = this.fxQueue;
10986         return q && q[0] && q[0].block;
10987     },
10988
10989         /* @private */
10990     queueFx : function(o, fn){
10991         if(!this.fxQueue){
10992             this.fxQueue = [];
10993         }
10994         if(!this.hasFxBlock()){
10995             Roo.applyIf(o, this.fxDefaults);
10996             if(!o.concurrent){
10997                 var run = this.beforeFx(o);
10998                 fn.block = o.block;
10999                 this.fxQueue.push(fn);
11000                 if(run){
11001                     this.nextFx();
11002                 }
11003             }else{
11004                 fn.call(this);
11005             }
11006         }
11007         return this;
11008     },
11009
11010         /* @private */
11011     fxWrap : function(pos, o, vis){
11012         var wrap;
11013         if(!o.wrap || !(wrap = Roo.get(o.wrap))){
11014             var wrapXY;
11015             if(o.fixPosition){
11016                 wrapXY = this.getXY();
11017             }
11018             var div = document.createElement("div");
11019             div.style.visibility = vis;
11020             wrap = Roo.get(this.dom.parentNode.insertBefore(div, this.dom));
11021             wrap.setPositioning(pos);
11022             if(wrap.getStyle("position") == "static"){
11023                 wrap.position("relative");
11024             }
11025             this.clearPositioning('auto');
11026             wrap.clip();
11027             wrap.dom.appendChild(this.dom);
11028             if(wrapXY){
11029                 wrap.setXY(wrapXY);
11030             }
11031         }
11032         return wrap;
11033     },
11034
11035         /* @private */
11036     fxUnwrap : function(wrap, pos, o){
11037         this.clearPositioning();
11038         this.setPositioning(pos);
11039         if(!o.wrap){
11040             wrap.dom.parentNode.insertBefore(this.dom, wrap.dom);
11041             wrap.remove();
11042         }
11043     },
11044
11045         /* @private */
11046     getFxRestore : function(){
11047         var st = this.dom.style;
11048         return {pos: this.getPositioning(), width: st.width, height : st.height};
11049     },
11050
11051         /* @private */
11052     afterFx : function(o){
11053         if(o.afterStyle){
11054             this.applyStyles(o.afterStyle);
11055         }
11056         if(o.afterCls){
11057             this.addClass(o.afterCls);
11058         }
11059         if(o.remove === true){
11060             this.remove();
11061         }
11062         Roo.callback(o.callback, o.scope, [this]);
11063         if(!o.concurrent){
11064             this.fxQueue.shift();
11065             this.nextFx();
11066         }
11067     },
11068
11069         /* @private */
11070     getFxEl : function(){ // support for composite element fx
11071         return Roo.get(this.dom);
11072     },
11073
11074         /* @private */
11075     fxanim : function(args, opt, animType, defaultDur, defaultEase, cb){
11076         animType = animType || 'run';
11077         opt = opt || {};
11078         var anim = Roo.lib.Anim[animType](
11079             this.dom, args,
11080             (opt.duration || defaultDur) || .35,
11081             (opt.easing || defaultEase) || 'easeOut',
11082             function(){
11083                 Roo.callback(cb, this);
11084             },
11085             this
11086         );
11087         opt.anim = anim;
11088         return anim;
11089     }
11090 };
11091
11092 // backwords compat
11093 Roo.Fx.resize = Roo.Fx.scale;
11094
11095 //When included, Roo.Fx is automatically applied to Element so that all basic
11096 //effects are available directly via the Element API
11097 Roo.apply(Roo.Element.prototype, Roo.Fx);/*
11098  * Based on:
11099  * Ext JS Library 1.1.1
11100  * Copyright(c) 2006-2007, Ext JS, LLC.
11101  *
11102  * Originally Released Under LGPL - original licence link has changed is not relivant.
11103  *
11104  * Fork - LGPL
11105  * <script type="text/javascript">
11106  */
11107
11108
11109 /**
11110  * @class Roo.CompositeElement
11111  * Standard composite class. Creates a Roo.Element for every element in the collection.
11112  * <br><br>
11113  * <b>NOTE: Although they are not listed, this class supports all of the set/update methods of Roo.Element. All Roo.Element
11114  * actions will be performed on all the elements in this collection.</b>
11115  * <br><br>
11116  * All methods return <i>this</i> and can be chained.
11117  <pre><code>
11118  var els = Roo.select("#some-el div.some-class", true);
11119  // or select directly from an existing element
11120  var el = Roo.get('some-el');
11121  el.select('div.some-class', true);
11122
11123  els.setWidth(100); // all elements become 100 width
11124  els.hide(true); // all elements fade out and hide
11125  // or
11126  els.setWidth(100).hide(true);
11127  </code></pre>
11128  */
11129 Roo.CompositeElement = function(els){
11130     this.elements = [];
11131     this.addElements(els);
11132 };
11133 Roo.CompositeElement.prototype = {
11134     isComposite: true,
11135     addElements : function(els){
11136         if(!els) {
11137             return this;
11138         }
11139         if(typeof els == "string"){
11140             els = Roo.Element.selectorFunction(els);
11141         }
11142         var yels = this.elements;
11143         var index = yels.length-1;
11144         for(var i = 0, len = els.length; i < len; i++) {
11145                 yels[++index] = Roo.get(els[i]);
11146         }
11147         return this;
11148     },
11149
11150     /**
11151     * Clears this composite and adds the elements returned by the passed selector.
11152     * @param {String/Array} els A string CSS selector, an array of elements or an element
11153     * @return {CompositeElement} this
11154     */
11155     fill : function(els){
11156         this.elements = [];
11157         this.add(els);
11158         return this;
11159     },
11160
11161     /**
11162     * Filters this composite to only elements that match the passed selector.
11163     * @param {String} selector A string CSS selector
11164     * @param {Boolean} inverse return inverse filter (not matches)
11165     * @return {CompositeElement} this
11166     */
11167     filter : function(selector, inverse){
11168         var els = [];
11169         inverse = inverse || false;
11170         this.each(function(el){
11171             var match = inverse ? !el.is(selector) : el.is(selector);
11172             if(match){
11173                 els[els.length] = el.dom;
11174             }
11175         });
11176         this.fill(els);
11177         return this;
11178     },
11179
11180     invoke : function(fn, args){
11181         var els = this.elements;
11182         for(var i = 0, len = els.length; i < len; i++) {
11183                 Roo.Element.prototype[fn].apply(els[i], args);
11184         }
11185         return this;
11186     },
11187     /**
11188     * Adds elements to this composite.
11189     * @param {String/Array} els A string CSS selector, an array of elements or an element
11190     * @return {CompositeElement} this
11191     */
11192     add : function(els){
11193         if(typeof els == "string"){
11194             this.addElements(Roo.Element.selectorFunction(els));
11195         }else if(els.length !== undefined){
11196             this.addElements(els);
11197         }else{
11198             this.addElements([els]);
11199         }
11200         return this;
11201     },
11202     /**
11203     * Calls the passed function passing (el, this, index) for each element in this composite.
11204     * @param {Function} fn The function to call
11205     * @param {Object} scope (optional) The <i>this</i> object (defaults to the element)
11206     * @return {CompositeElement} this
11207     */
11208     each : function(fn, scope){
11209         var els = this.elements;
11210         for(var i = 0, len = els.length; i < len; i++){
11211             if(fn.call(scope || els[i], els[i], this, i) === false) {
11212                 break;
11213             }
11214         }
11215         return this;
11216     },
11217
11218     /**
11219      * Returns the Element object at the specified index
11220      * @param {Number} index
11221      * @return {Roo.Element}
11222      */
11223     item : function(index){
11224         return this.elements[index] || null;
11225     },
11226
11227     /**
11228      * Returns the first Element
11229      * @return {Roo.Element}
11230      */
11231     first : function(){
11232         return this.item(0);
11233     },
11234
11235     /**
11236      * Returns the last Element
11237      * @return {Roo.Element}
11238      */
11239     last : function(){
11240         return this.item(this.elements.length-1);
11241     },
11242
11243     /**
11244      * Returns the number of elements in this composite
11245      * @return Number
11246      */
11247     getCount : function(){
11248         return this.elements.length;
11249     },
11250
11251     /**
11252      * Returns true if this composite contains the passed element
11253      * @return Boolean
11254      */
11255     contains : function(el){
11256         return this.indexOf(el) !== -1;
11257     },
11258
11259     /**
11260      * Returns true if this composite contains the passed element
11261      * @return Boolean
11262      */
11263     indexOf : function(el){
11264         return this.elements.indexOf(Roo.get(el));
11265     },
11266
11267
11268     /**
11269     * Removes the specified element(s).
11270     * @param {Mixed} el The id of an element, the Element itself, the index of the element in this composite
11271     * or an array of any of those.
11272     * @param {Boolean} removeDom (optional) True to also remove the element from the document
11273     * @return {CompositeElement} this
11274     */
11275     removeElement : function(el, removeDom){
11276         if(el instanceof Array){
11277             for(var i = 0, len = el.length; i < len; i++){
11278                 this.removeElement(el[i]);
11279             }
11280             return this;
11281         }
11282         var index = typeof el == 'number' ? el : this.indexOf(el);
11283         if(index !== -1){
11284             if(removeDom){
11285                 var d = this.elements[index];
11286                 if(d.dom){
11287                     d.remove();
11288                 }else{
11289                     d.parentNode.removeChild(d);
11290                 }
11291             }
11292             this.elements.splice(index, 1);
11293         }
11294         return this;
11295     },
11296
11297     /**
11298     * Replaces the specified element with the passed element.
11299     * @param {String/HTMLElement/Element/Number} el The id of an element, the Element itself, the index of the element in this composite
11300     * to replace.
11301     * @param {String/HTMLElement/Element} replacement The id of an element or the Element itself.
11302     * @param {Boolean} domReplace (Optional) True to remove and replace the element in the document too.
11303     * @return {CompositeElement} this
11304     */
11305     replaceElement : function(el, replacement, domReplace){
11306         var index = typeof el == 'number' ? el : this.indexOf(el);
11307         if(index !== -1){
11308             if(domReplace){
11309                 this.elements[index].replaceWith(replacement);
11310             }else{
11311                 this.elements.splice(index, 1, Roo.get(replacement))
11312             }
11313         }
11314         return this;
11315     },
11316
11317     /**
11318      * Removes all elements.
11319      */
11320     clear : function(){
11321         this.elements = [];
11322     }
11323 };
11324 (function(){
11325     Roo.CompositeElement.createCall = function(proto, fnName){
11326         if(!proto[fnName]){
11327             proto[fnName] = function(){
11328                 return this.invoke(fnName, arguments);
11329             };
11330         }
11331     };
11332     for(var fnName in Roo.Element.prototype){
11333         if(typeof Roo.Element.prototype[fnName] == "function"){
11334             Roo.CompositeElement.createCall(Roo.CompositeElement.prototype, fnName);
11335         }
11336     };
11337 })();
11338 /*
11339  * Based on:
11340  * Ext JS Library 1.1.1
11341  * Copyright(c) 2006-2007, Ext JS, LLC.
11342  *
11343  * Originally Released Under LGPL - original licence link has changed is not relivant.
11344  *
11345  * Fork - LGPL
11346  * <script type="text/javascript">
11347  */
11348
11349 /**
11350  * @class Roo.CompositeElementLite
11351  * @extends Roo.CompositeElement
11352  * Flyweight composite class. Reuses the same Roo.Element for element operations.
11353  <pre><code>
11354  var els = Roo.select("#some-el div.some-class");
11355  // or select directly from an existing element
11356  var el = Roo.get('some-el');
11357  el.select('div.some-class');
11358
11359  els.setWidth(100); // all elements become 100 width
11360  els.hide(true); // all elements fade out and hide
11361  // or
11362  els.setWidth(100).hide(true);
11363  </code></pre><br><br>
11364  * <b>NOTE: Although they are not listed, this class supports all of the set/update methods of Roo.Element. All Roo.Element
11365  * actions will be performed on all the elements in this collection.</b>
11366  */
11367 Roo.CompositeElementLite = function(els){
11368     Roo.CompositeElementLite.superclass.constructor.call(this, els);
11369     this.el = new Roo.Element.Flyweight();
11370 };
11371 Roo.extend(Roo.CompositeElementLite, Roo.CompositeElement, {
11372     addElements : function(els){
11373         if(els){
11374             if(els instanceof Array){
11375                 this.elements = this.elements.concat(els);
11376             }else{
11377                 var yels = this.elements;
11378                 var index = yels.length-1;
11379                 for(var i = 0, len = els.length; i < len; i++) {
11380                     yels[++index] = els[i];
11381                 }
11382             }
11383         }
11384         return this;
11385     },
11386     invoke : function(fn, args){
11387         var els = this.elements;
11388         var el = this.el;
11389         for(var i = 0, len = els.length; i < len; i++) {
11390             el.dom = els[i];
11391                 Roo.Element.prototype[fn].apply(el, args);
11392         }
11393         return this;
11394     },
11395     /**
11396      * Returns a flyweight Element of the dom element object at the specified index
11397      * @param {Number} index
11398      * @return {Roo.Element}
11399      */
11400     item : function(index){
11401         if(!this.elements[index]){
11402             return null;
11403         }
11404         this.el.dom = this.elements[index];
11405         return this.el;
11406     },
11407
11408     // fixes scope with flyweight
11409     addListener : function(eventName, handler, scope, opt){
11410         var els = this.elements;
11411         for(var i = 0, len = els.length; i < len; i++) {
11412             Roo.EventManager.on(els[i], eventName, handler, scope || els[i], opt);
11413         }
11414         return this;
11415     },
11416
11417     /**
11418     * Calls the passed function passing (el, this, index) for each element in this composite. <b>The element
11419     * passed is the flyweight (shared) Roo.Element instance, so if you require a
11420     * a reference to the dom node, use el.dom.</b>
11421     * @param {Function} fn The function to call
11422     * @param {Object} scope (optional) The <i>this</i> object (defaults to the element)
11423     * @return {CompositeElement} this
11424     */
11425     each : function(fn, scope){
11426         var els = this.elements;
11427         var el = this.el;
11428         for(var i = 0, len = els.length; i < len; i++){
11429             el.dom = els[i];
11430                 if(fn.call(scope || el, el, this, i) === false){
11431                 break;
11432             }
11433         }
11434         return this;
11435     },
11436
11437     indexOf : function(el){
11438         return this.elements.indexOf(Roo.getDom(el));
11439     },
11440
11441     replaceElement : function(el, replacement, domReplace){
11442         var index = typeof el == 'number' ? el : this.indexOf(el);
11443         if(index !== -1){
11444             replacement = Roo.getDom(replacement);
11445             if(domReplace){
11446                 var d = this.elements[index];
11447                 d.parentNode.insertBefore(replacement, d);
11448                 d.parentNode.removeChild(d);
11449             }
11450             this.elements.splice(index, 1, replacement);
11451         }
11452         return this;
11453     }
11454 });
11455 Roo.CompositeElementLite.prototype.on = Roo.CompositeElementLite.prototype.addListener;
11456
11457 /*
11458  * Based on:
11459  * Ext JS Library 1.1.1
11460  * Copyright(c) 2006-2007, Ext JS, LLC.
11461  *
11462  * Originally Released Under LGPL - original licence link has changed is not relivant.
11463  *
11464  * Fork - LGPL
11465  * <script type="text/javascript">
11466  */
11467
11468  
11469
11470 /**
11471  * @class Roo.data.Connection
11472  * @extends Roo.util.Observable
11473  * The class encapsulates a connection to the page's originating domain, allowing requests to be made
11474  * either to a configured URL, or to a URL specified at request time. 
11475  * 
11476  * Requests made by this class are asynchronous, and will return immediately. No data from
11477  * the server will be available to the statement immediately following the {@link #request} call.
11478  * To process returned data, use a callback in the request options object, or an event listener.
11479  * 
11480  * Note: If you are doing a file upload, you will not get a normal response object sent back to
11481  * your callback or event handler.  Since the upload is handled via in IFRAME, there is no XMLHttpRequest.
11482  * The response object is created using the innerHTML of the IFRAME's document as the responseText
11483  * property and, if present, the IFRAME's XML document as the responseXML property.
11484  * 
11485  * This means that a valid XML or HTML document must be returned. If JSON data is required, it is suggested
11486  * that it be placed either inside a &lt;textarea> in an HTML document and retrieved from the responseText
11487  * using a regex, or inside a CDATA section in an XML document and retrieved from the responseXML using
11488  * standard DOM methods.
11489  * @constructor
11490  * @param {Object} config a configuration object.
11491  */
11492 Roo.data.Connection = function(config){
11493     Roo.apply(this, config);
11494     this.addEvents({
11495         /**
11496          * @event beforerequest
11497          * Fires before a network request is made to retrieve a data object.
11498          * @param {Connection} conn This Connection object.
11499          * @param {Object} options The options config object passed to the {@link #request} method.
11500          */
11501         "beforerequest" : true,
11502         /**
11503          * @event requestcomplete
11504          * Fires if the request was successfully completed.
11505          * @param {Connection} conn This Connection object.
11506          * @param {Object} response The XHR object containing the response data.
11507          * See {@link http://www.w3.org/TR/XMLHttpRequest/} for details.
11508          * @param {Object} options The options config object passed to the {@link #request} method.
11509          */
11510         "requestcomplete" : true,
11511         /**
11512          * @event requestexception
11513          * Fires if an error HTTP status was returned from the server.
11514          * See {@link http://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html} for details of HTTP status codes.
11515          * @param {Connection} conn This Connection object.
11516          * @param {Object} response The XHR object containing the response data.
11517          * See {@link http://www.w3.org/TR/XMLHttpRequest/} for details.
11518          * @param {Object} options The options config object passed to the {@link #request} method.
11519          */
11520         "requestexception" : true
11521     });
11522     Roo.data.Connection.superclass.constructor.call(this);
11523 };
11524
11525 Roo.extend(Roo.data.Connection, Roo.util.Observable, {
11526     /**
11527      * @cfg {String} url (Optional) The default URL to be used for requests to the server. (defaults to undefined)
11528      */
11529     /**
11530      * @cfg {Object} extraParams (Optional) An object containing properties which are used as
11531      * extra parameters to each request made by this object. (defaults to undefined)
11532      */
11533     /**
11534      * @cfg {Object} defaultHeaders (Optional) An object containing request headers which are added
11535      *  to each request made by this object. (defaults to undefined)
11536      */
11537     /**
11538      * @cfg {String} method (Optional) The default HTTP method to be used for requests. (defaults to undefined; if not set but parms are present will use POST, otherwise GET)
11539      */
11540     /**
11541      * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
11542      */
11543     timeout : 30000,
11544     /**
11545      * @cfg {Boolean} autoAbort (Optional) Whether this request should abort any pending requests. (defaults to false)
11546      * @type Boolean
11547      */
11548     autoAbort:false,
11549
11550     /**
11551      * @cfg {Boolean} disableCaching (Optional) True to add a unique cache-buster param to GET requests. (defaults to true)
11552      * @type Boolean
11553      */
11554     disableCaching: true,
11555
11556     /**
11557      * Sends an HTTP request to a remote server.
11558      * @param {Object} options An object which may contain the following properties:<ul>
11559      * <li><b>url</b> {String} (Optional) The URL to which to send the request. Defaults to configured URL</li>
11560      * <li><b>params</b> {Object/String/Function} (Optional) An object containing properties which are used as parameters to the
11561      * request, a url encoded string or a function to call to get either.</li>
11562      * <li><b>method</b> {String} (Optional) The HTTP method to use for the request. Defaults to the configured method, or
11563      * if no method was configured, "GET" if no parameters are being sent, and "POST" if parameters are being sent.</li>
11564      * <li><b>callback</b> {Function} (Optional) The function to be called upon receipt of the HTTP response.
11565      * The callback is called regardless of success or failure and is passed the following parameters:<ul>
11566      * <li>options {Object} The parameter to the request call.</li>
11567      * <li>success {Boolean} True if the request succeeded.</li>
11568      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11569      * </ul></li>
11570      * <li><b>success</b> {Function} (Optional) The function to be called upon success of the request.
11571      * The callback is passed the following parameters:<ul>
11572      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11573      * <li>options {Object} The parameter to the request call.</li>
11574      * </ul></li>
11575      * <li><b>failure</b> {Function} (Optional) The function to be called upon failure of the request.
11576      * The callback is passed the following parameters:<ul>
11577      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11578      * <li>options {Object} The parameter to the request call.</li>
11579      * </ul></li>
11580      * <li><b>scope</b> {Object} (Optional) The scope in which to execute the callbacks: The "this" object
11581      * for the callback function. Defaults to the browser window.</li>
11582      * <li><b>form</b> {Object/String} (Optional) A form object or id to pull parameters from.</li>
11583      * <li><b>isUpload</b> {Boolean} (Optional) True if the form object is a file upload (will usually be automatically detected).</li>
11584      * <li><b>headers</b> {Object} (Optional) Request headers to set for the request.</li>
11585      * <li><b>xmlData</b> {Object} (Optional) XML document to use for the post. Note: This will be used instead of
11586      * params for the post data. Any params will be appended to the URL.</li>
11587      * <li><b>disableCaching</b> {Boolean} (Optional) True to add a unique cache-buster param to GET requests.</li>
11588      * </ul>
11589      * @return {Number} transactionId
11590      */
11591     request : function(o){
11592         if(this.fireEvent("beforerequest", this, o) !== false){
11593             var p = o.params;
11594
11595             if(typeof p == "function"){
11596                 p = p.call(o.scope||window, o);
11597             }
11598             if(typeof p == "object"){
11599                 p = Roo.urlEncode(o.params);
11600             }
11601             if(this.extraParams){
11602                 var extras = Roo.urlEncode(this.extraParams);
11603                 p = p ? (p + '&' + extras) : extras;
11604             }
11605
11606             var url = o.url || this.url;
11607             if(typeof url == 'function'){
11608                 url = url.call(o.scope||window, o);
11609             }
11610
11611             if(o.form){
11612                 var form = Roo.getDom(o.form);
11613                 url = url || form.action;
11614
11615                 var enctype = form.getAttribute("enctype");
11616                 
11617                 if (o.formData) {
11618                     return this.doFormDataUpload(o,p,url);
11619                 }
11620                 
11621                 if(o.isUpload || (enctype && enctype.toLowerCase() == 'multipart/form-data')){
11622                     return this.doFormUpload(o, p, url);
11623                 }
11624                 var f = Roo.lib.Ajax.serializeForm(form);
11625                 p = p ? (p + '&' + f) : f;
11626             }
11627
11628             var hs = o.headers;
11629             if(this.defaultHeaders){
11630                 hs = Roo.apply(hs || {}, this.defaultHeaders);
11631                 if(!o.headers){
11632                     o.headers = hs;
11633                 }
11634             }
11635
11636             var cb = {
11637                 success: this.handleResponse,
11638                 failure: this.handleFailure,
11639                 scope: this,
11640                 argument: {options: o},
11641                 timeout : o.timeout || this.timeout
11642             };
11643
11644             var method = o.method||this.method||(p ? "POST" : "GET");
11645
11646             if(method == 'GET' && (this.disableCaching && o.disableCaching !== false) || o.disableCaching === true){
11647                 url += (url.indexOf('?') != -1 ? '&' : '?') + '_dc=' + (new Date().getTime());
11648             }
11649
11650             if(typeof o.autoAbort == 'boolean'){ // options gets top priority
11651                 if(o.autoAbort){
11652                     this.abort();
11653                 }
11654             }else if(this.autoAbort !== false){
11655                 this.abort();
11656             }
11657
11658             if((method == 'GET' && p) || o.xmlData){
11659                 url += (url.indexOf('?') != -1 ? '&' : '?') + p;
11660                 p = '';
11661             }
11662             this.transId = Roo.lib.Ajax.request(method, url, cb, p, o);
11663             return this.transId;
11664         }else{
11665             Roo.callback(o.callback, o.scope, [o, null, null]);
11666             return null;
11667         }
11668     },
11669
11670     /**
11671      * Determine whether this object has a request outstanding.
11672      * @param {Number} transactionId (Optional) defaults to the last transaction
11673      * @return {Boolean} True if there is an outstanding request.
11674      */
11675     isLoading : function(transId){
11676         if(transId){
11677             return Roo.lib.Ajax.isCallInProgress(transId);
11678         }else{
11679             return this.transId ? true : false;
11680         }
11681     },
11682
11683     /**
11684      * Aborts any outstanding request.
11685      * @param {Number} transactionId (Optional) defaults to the last transaction
11686      */
11687     abort : function(transId){
11688         if(transId || this.isLoading()){
11689             Roo.lib.Ajax.abort(transId || this.transId);
11690         }
11691     },
11692
11693     // private
11694     handleResponse : function(response){
11695         this.transId = false;
11696         var options = response.argument.options;
11697         response.argument = options ? options.argument : null;
11698         this.fireEvent("requestcomplete", this, response, options);
11699         Roo.callback(options.success, options.scope, [response, options]);
11700         Roo.callback(options.callback, options.scope, [options, true, response]);
11701     },
11702
11703     // private
11704     handleFailure : function(response, e){
11705         this.transId = false;
11706         var options = response.argument.options;
11707         response.argument = options ? options.argument : null;
11708         this.fireEvent("requestexception", this, response, options, e);
11709         Roo.callback(options.failure, options.scope, [response, options]);
11710         Roo.callback(options.callback, options.scope, [options, false, response]);
11711     },
11712
11713     // private
11714     doFormUpload : function(o, ps, url){
11715         var id = Roo.id();
11716         var frame = document.createElement('iframe');
11717         frame.id = id;
11718         frame.name = id;
11719         frame.className = 'x-hidden';
11720         if(Roo.isIE){
11721             frame.src = Roo.SSL_SECURE_URL;
11722         }
11723         document.body.appendChild(frame);
11724
11725         if(Roo.isIE){
11726            document.frames[id].name = id;
11727         }
11728
11729         var form = Roo.getDom(o.form);
11730         form.target = id;
11731         form.method = 'POST';
11732         form.enctype = form.encoding = 'multipart/form-data';
11733         if(url){
11734             form.action = url;
11735         }
11736
11737         var hiddens, hd;
11738         if(ps){ // add dynamic params
11739             hiddens = [];
11740             ps = Roo.urlDecode(ps, false);
11741             for(var k in ps){
11742                 if(ps.hasOwnProperty(k)){
11743                     hd = document.createElement('input');
11744                     hd.type = 'hidden';
11745                     hd.name = k;
11746                     hd.value = ps[k];
11747                     form.appendChild(hd);
11748                     hiddens.push(hd);
11749                 }
11750             }
11751         }
11752
11753         function cb(){
11754             var r = {  // bogus response object
11755                 responseText : '',
11756                 responseXML : null
11757             };
11758
11759             r.argument = o ? o.argument : null;
11760
11761             try { //
11762                 var doc;
11763                 if(Roo.isIE){
11764                     doc = frame.contentWindow.document;
11765                 }else {
11766                     doc = (frame.contentDocument || window.frames[id].document);
11767                 }
11768                 if(doc && doc.body){
11769                     r.responseText = doc.body.innerHTML;
11770                 }
11771                 if(doc && doc.XMLDocument){
11772                     r.responseXML = doc.XMLDocument;
11773                 }else {
11774                     r.responseXML = doc;
11775                 }
11776             }
11777             catch(e) {
11778                 // ignore
11779             }
11780
11781             Roo.EventManager.removeListener(frame, 'load', cb, this);
11782
11783             this.fireEvent("requestcomplete", this, r, o);
11784             Roo.callback(o.success, o.scope, [r, o]);
11785             Roo.callback(o.callback, o.scope, [o, true, r]);
11786
11787             setTimeout(function(){document.body.removeChild(frame);}, 100);
11788         }
11789
11790         Roo.EventManager.on(frame, 'load', cb, this);
11791         form.submit();
11792
11793         if(hiddens){ // remove dynamic params
11794             for(var i = 0, len = hiddens.length; i < len; i++){
11795                 form.removeChild(hiddens[i]);
11796             }
11797         }
11798     },
11799     // this is a 'formdata version???'
11800     
11801     
11802     doFormDataUpload : function(o, ps, url)
11803     {
11804         var form = Roo.getDom(o.form);
11805         form.enctype = form.encoding = 'multipart/form-data';
11806         var formData = o.formData === true ? new FormData(form) : o.formData;
11807       
11808         var cb = {
11809             success: this.handleResponse,
11810             failure: this.handleFailure,
11811             scope: this,
11812             argument: {options: o},
11813             timeout : o.timeout || this.timeout
11814         };
11815  
11816         if(typeof o.autoAbort == 'boolean'){ // options gets top priority
11817             if(o.autoAbort){
11818                 this.abort();
11819             }
11820         }else if(this.autoAbort !== false){
11821             this.abort();
11822         }
11823
11824         //Roo.lib.Ajax.defaultPostHeader = null;
11825         Roo.lib.Ajax.useDefaultHeader = false;
11826         this.transId = Roo.lib.Ajax.request( "POST", url, cb, o.formData, o);
11827         Roo.lib.Ajax.useDefaultHeader = true;
11828  
11829          
11830     }
11831     
11832 });
11833 /*
11834  * Based on:
11835  * Ext JS Library 1.1.1
11836  * Copyright(c) 2006-2007, Ext JS, LLC.
11837  *
11838  * Originally Released Under LGPL - original licence link has changed is not relivant.
11839  *
11840  * Fork - LGPL
11841  * <script type="text/javascript">
11842  */
11843  
11844 /**
11845  * Global Ajax request class.
11846  * 
11847  * @class Roo.Ajax
11848  * @extends Roo.data.Connection
11849  * @static
11850  * 
11851  * @cfg {String} url  The default URL to be used for requests to the server. (defaults to undefined)
11852  * @cfg {Object} extraParams  An object containing properties which are used as extra parameters to each request made by this object. (defaults to undefined)
11853  * @cfg {Object} defaultHeaders  An object containing request headers which are added to each request made by this object. (defaults to undefined)
11854  * @cfg {String} method (Optional)  The default HTTP method to be used for requests. (defaults to undefined; if not set but parms are present will use POST, otherwise GET)
11855  * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
11856  * @cfg {Boolean} autoAbort (Optional) Whether a new request should abort any pending requests. (defaults to false)
11857  * @cfg {Boolean} disableCaching (Optional)   True to add a unique cache-buster param to GET requests. (defaults to true)
11858  */
11859 Roo.Ajax = new Roo.data.Connection({
11860     // fix up the docs
11861     /**
11862      * @scope Roo.Ajax
11863      * @type {Boolear} 
11864      */
11865     autoAbort : false,
11866
11867     /**
11868      * Serialize the passed form into a url encoded string
11869      * @scope Roo.Ajax
11870      * @param {String/HTMLElement} form
11871      * @return {String}
11872      */
11873     serializeForm : function(form){
11874         return Roo.lib.Ajax.serializeForm(form);
11875     }
11876 });/*
11877  * Based on:
11878  * Ext JS Library 1.1.1
11879  * Copyright(c) 2006-2007, Ext JS, LLC.
11880  *
11881  * Originally Released Under LGPL - original licence link has changed is not relivant.
11882  *
11883  * Fork - LGPL
11884  * <script type="text/javascript">
11885  */
11886
11887  
11888 /**
11889  * @class Roo.UpdateManager
11890  * @extends Roo.util.Observable
11891  * Provides AJAX-style update for Element object.<br><br>
11892  * Usage:<br>
11893  * <pre><code>
11894  * // Get it from a Roo.Element object
11895  * var el = Roo.get("foo");
11896  * var mgr = el.getUpdateManager();
11897  * mgr.update("http://myserver.com/index.php", "param1=1&amp;param2=2");
11898  * ...
11899  * mgr.formUpdate("myFormId", "http://myserver.com/index.php");
11900  * <br>
11901  * // or directly (returns the same UpdateManager instance)
11902  * var mgr = new Roo.UpdateManager("myElementId");
11903  * mgr.startAutoRefresh(60, "http://myserver.com/index.php");
11904  * mgr.on("update", myFcnNeedsToKnow);
11905  * <br>
11906    // short handed call directly from the element object
11907    Roo.get("foo").load({
11908         url: "bar.php",
11909         scripts:true,
11910         params: "for=bar",
11911         text: "Loading Foo..."
11912    });
11913  * </code></pre>
11914  * @constructor
11915  * Create new UpdateManager directly.
11916  * @param {String/HTMLElement/Roo.Element} el The element to update
11917  * @param {Boolean} forceNew (optional) By default the constructor checks to see if the passed element already has an UpdateManager and if it does it returns the same instance. This will skip that check (useful for extending this class).
11918  */
11919 Roo.UpdateManager = function(el, forceNew){
11920     el = Roo.get(el);
11921     if(!forceNew && el.updateManager){
11922         return el.updateManager;
11923     }
11924     /**
11925      * The Element object
11926      * @type Roo.Element
11927      */
11928     this.el = el;
11929     /**
11930      * Cached url to use for refreshes. Overwritten every time update() is called unless "discardUrl" param is set to true.
11931      * @type String
11932      */
11933     this.defaultUrl = null;
11934
11935     this.addEvents({
11936         /**
11937          * @event beforeupdate
11938          * Fired before an update is made, return false from your handler and the update is cancelled.
11939          * @param {Roo.Element} el
11940          * @param {String/Object/Function} url
11941          * @param {String/Object} params
11942          */
11943         "beforeupdate": true,
11944         /**
11945          * @event update
11946          * Fired after successful update is made.
11947          * @param {Roo.Element} el
11948          * @param {Object} oResponseObject The response Object
11949          */
11950         "update": true,
11951         /**
11952          * @event failure
11953          * Fired on update failure.
11954          * @param {Roo.Element} el
11955          * @param {Object} oResponseObject The response Object
11956          */
11957         "failure": true
11958     });
11959     var d = Roo.UpdateManager.defaults;
11960     /**
11961      * Blank page URL to use with SSL file uploads (Defaults to Roo.UpdateManager.defaults.sslBlankUrl or "about:blank").
11962      * @type String
11963      */
11964     this.sslBlankUrl = d.sslBlankUrl;
11965     /**
11966      * Whether to append unique parameter on get request to disable caching (Defaults to Roo.UpdateManager.defaults.disableCaching or false).
11967      * @type Boolean
11968      */
11969     this.disableCaching = d.disableCaching;
11970     /**
11971      * Text for loading indicator (Defaults to Roo.UpdateManager.defaults.indicatorText or '&lt;div class="loading-indicator"&gt;Loading...&lt;/div&gt;').
11972      * @type String
11973      */
11974     this.indicatorText = d.indicatorText;
11975     /**
11976      * Whether to show indicatorText when loading (Defaults to Roo.UpdateManager.defaults.showLoadIndicator or true).
11977      * @type String
11978      */
11979     this.showLoadIndicator = d.showLoadIndicator;
11980     /**
11981      * Timeout for requests or form posts in seconds (Defaults to Roo.UpdateManager.defaults.timeout or 30 seconds).
11982      * @type Number
11983      */
11984     this.timeout = d.timeout;
11985
11986     /**
11987      * True to process scripts in the output (Defaults to Roo.UpdateManager.defaults.loadScripts (false)).
11988      * @type Boolean
11989      */
11990     this.loadScripts = d.loadScripts;
11991
11992     /**
11993      * Transaction object of current executing transaction
11994      */
11995     this.transaction = null;
11996
11997     /**
11998      * @private
11999      */
12000     this.autoRefreshProcId = null;
12001     /**
12002      * Delegate for refresh() prebound to "this", use myUpdater.refreshDelegate.createCallback(arg1, arg2) to bind arguments
12003      * @type Function
12004      */
12005     this.refreshDelegate = this.refresh.createDelegate(this);
12006     /**
12007      * Delegate for update() prebound to "this", use myUpdater.updateDelegate.createCallback(arg1, arg2) to bind arguments
12008      * @type Function
12009      */
12010     this.updateDelegate = this.update.createDelegate(this);
12011     /**
12012      * Delegate for formUpdate() prebound to "this", use myUpdater.formUpdateDelegate.createCallback(arg1, arg2) to bind arguments
12013      * @type Function
12014      */
12015     this.formUpdateDelegate = this.formUpdate.createDelegate(this);
12016     /**
12017      * @private
12018      */
12019     this.successDelegate = this.processSuccess.createDelegate(this);
12020     /**
12021      * @private
12022      */
12023     this.failureDelegate = this.processFailure.createDelegate(this);
12024
12025     if(!this.renderer){
12026      /**
12027       * The renderer for this UpdateManager. Defaults to {@link Roo.UpdateManager.BasicRenderer}.
12028       */
12029     this.renderer = new Roo.UpdateManager.BasicRenderer();
12030     }
12031     
12032     Roo.UpdateManager.superclass.constructor.call(this);
12033 };
12034
12035 Roo.extend(Roo.UpdateManager, Roo.util.Observable, {
12036     /**
12037      * Get the Element this UpdateManager is bound to
12038      * @return {Roo.Element} The element
12039      */
12040     getEl : function(){
12041         return this.el;
12042     },
12043     /**
12044      * Performs an async request, updating this element with the response. If params are specified it uses POST, otherwise it uses GET.
12045      * @param {Object/String/Function} url The url for this request or a function to call to get the url or a config object containing any of the following options:
12046 <pre><code>
12047 um.update({<br/>
12048     url: "your-url.php",<br/>
12049     params: {param1: "foo", param2: "bar"}, // or a URL encoded string<br/>
12050     callback: yourFunction,<br/>
12051     scope: yourObject, //(optional scope)  <br/>
12052     discardUrl: false, <br/>
12053     nocache: false,<br/>
12054     text: "Loading...",<br/>
12055     timeout: 30,<br/>
12056     scripts: false<br/>
12057 });
12058 </code></pre>
12059      * The only required property is url. The optional properties nocache, text and scripts
12060      * are shorthand for disableCaching, indicatorText and loadScripts and are used to set their associated property on this UpdateManager instance.
12061      * @param {String/Object} params (optional) The parameters to pass as either a url encoded string "param1=1&amp;param2=2" or an object {param1: 1, param2: 2}
12062      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess, oResponse)
12063      * @param {Boolean} discardUrl (optional) By default when you execute an update the defaultUrl is changed to the last used url. If true, it will not store the url.
12064      */
12065     update : function(url, params, callback, discardUrl){
12066         if(this.fireEvent("beforeupdate", this.el, url, params) !== false){
12067             var method = this.method,
12068                 cfg;
12069             if(typeof url == "object"){ // must be config object
12070                 cfg = url;
12071                 url = cfg.url;
12072                 params = params || cfg.params;
12073                 callback = callback || cfg.callback;
12074                 discardUrl = discardUrl || cfg.discardUrl;
12075                 if(callback && cfg.scope){
12076                     callback = callback.createDelegate(cfg.scope);
12077                 }
12078                 if(typeof cfg.method != "undefined"){method = cfg.method;};
12079                 if(typeof cfg.nocache != "undefined"){this.disableCaching = cfg.nocache;};
12080                 if(typeof cfg.text != "undefined"){this.indicatorText = '<div class="loading-indicator">'+cfg.text+"</div>";};
12081                 if(typeof cfg.scripts != "undefined"){this.loadScripts = cfg.scripts;};
12082                 if(typeof cfg.timeout != "undefined"){this.timeout = cfg.timeout;};
12083             }
12084             this.showLoading();
12085             if(!discardUrl){
12086                 this.defaultUrl = url;
12087             }
12088             if(typeof url == "function"){
12089                 url = url.call(this);
12090             }
12091
12092             method = method || (params ? "POST" : "GET");
12093             if(method == "GET"){
12094                 url = this.prepareUrl(url);
12095             }
12096
12097             var o = Roo.apply(cfg ||{}, {
12098                 url : url,
12099                 params: params,
12100                 success: this.successDelegate,
12101                 failure: this.failureDelegate,
12102                 callback: undefined,
12103                 timeout: (this.timeout*1000),
12104                 argument: {"url": url, "form": null, "callback": callback, "params": params}
12105             });
12106             Roo.log("updated manager called with timeout of " + o.timeout);
12107             this.transaction = Roo.Ajax.request(o);
12108         }
12109     },
12110
12111     /**
12112      * Performs an async form post, updating this element with the response. If the form has the attribute enctype="multipart/form-data", it assumes it's a file upload.
12113      * Uses this.sslBlankUrl for SSL file uploads to prevent IE security warning.
12114      * @param {String/HTMLElement} form The form Id or form element
12115      * @param {String} url (optional) The url to pass the form to. If omitted the action attribute on the form will be used.
12116      * @param {Boolean} reset (optional) Whether to try to reset the form after the update
12117      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess, oResponse)
12118      */
12119     formUpdate : function(form, url, reset, callback){
12120         if(this.fireEvent("beforeupdate", this.el, form, url) !== false){
12121             if(typeof url == "function"){
12122                 url = url.call(this);
12123             }
12124             form = Roo.getDom(form);
12125             this.transaction = Roo.Ajax.request({
12126                 form: form,
12127                 url:url,
12128                 success: this.successDelegate,
12129                 failure: this.failureDelegate,
12130                 timeout: (this.timeout*1000),
12131                 argument: {"url": url, "form": form, "callback": callback, "reset": reset}
12132             });
12133             this.showLoading.defer(1, this);
12134         }
12135     },
12136
12137     /**
12138      * Refresh the element with the last used url or defaultUrl. If there is no url, it returns immediately
12139      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
12140      */
12141     refresh : function(callback){
12142         if(this.defaultUrl == null){
12143             return;
12144         }
12145         this.update(this.defaultUrl, null, callback, true);
12146     },
12147
12148     /**
12149      * Set this element to auto refresh.
12150      * @param {Number} interval How often to update (in seconds).
12151      * @param {String/Function} url (optional) The url for this request or a function to call to get the url (Defaults to the last used url)
12152      * @param {String/Object} params (optional) The parameters to pass as either a url encoded string "&param1=1&param2=2" or as an object {param1: 1, param2: 2}
12153      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
12154      * @param {Boolean} refreshNow (optional) Whether to execute the refresh now, or wait the interval
12155      */
12156     startAutoRefresh : function(interval, url, params, callback, refreshNow){
12157         if(refreshNow){
12158             this.update(url || this.defaultUrl, params, callback, true);
12159         }
12160         if(this.autoRefreshProcId){
12161             clearInterval(this.autoRefreshProcId);
12162         }
12163         this.autoRefreshProcId = setInterval(this.update.createDelegate(this, [url || this.defaultUrl, params, callback, true]), interval*1000);
12164     },
12165
12166     /**
12167      * Stop auto refresh on this element.
12168      */
12169      stopAutoRefresh : function(){
12170         if(this.autoRefreshProcId){
12171             clearInterval(this.autoRefreshProcId);
12172             delete this.autoRefreshProcId;
12173         }
12174     },
12175
12176     isAutoRefreshing : function(){
12177        return this.autoRefreshProcId ? true : false;
12178     },
12179     /**
12180      * Called to update the element to "Loading" state. Override to perform custom action.
12181      */
12182     showLoading : function(){
12183         if(this.showLoadIndicator){
12184             this.el.update(this.indicatorText);
12185         }
12186     },
12187
12188     /**
12189      * Adds unique parameter to query string if disableCaching = true
12190      * @private
12191      */
12192     prepareUrl : function(url){
12193         if(this.disableCaching){
12194             var append = "_dc=" + (new Date().getTime());
12195             if(url.indexOf("?") !== -1){
12196                 url += "&" + append;
12197             }else{
12198                 url += "?" + append;
12199             }
12200         }
12201         return url;
12202     },
12203
12204     /**
12205      * @private
12206      */
12207     processSuccess : function(response){
12208         this.transaction = null;
12209         if(response.argument.form && response.argument.reset){
12210             try{ // put in try/catch since some older FF releases had problems with this
12211                 response.argument.form.reset();
12212             }catch(e){}
12213         }
12214         if(this.loadScripts){
12215             this.renderer.render(this.el, response, this,
12216                 this.updateComplete.createDelegate(this, [response]));
12217         }else{
12218             this.renderer.render(this.el, response, this);
12219             this.updateComplete(response);
12220         }
12221     },
12222
12223     updateComplete : function(response){
12224         this.fireEvent("update", this.el, response);
12225         if(typeof response.argument.callback == "function"){
12226             response.argument.callback(this.el, true, response);
12227         }
12228     },
12229
12230     /**
12231      * @private
12232      */
12233     processFailure : function(response){
12234         this.transaction = null;
12235         this.fireEvent("failure", this.el, response);
12236         if(typeof response.argument.callback == "function"){
12237             response.argument.callback(this.el, false, response);
12238         }
12239     },
12240
12241     /**
12242      * Set the content renderer for this UpdateManager. See {@link Roo.UpdateManager.BasicRenderer#render} for more details.
12243      * @param {Object} renderer The object implementing the render() method
12244      */
12245     setRenderer : function(renderer){
12246         this.renderer = renderer;
12247     },
12248
12249     getRenderer : function(){
12250        return this.renderer;
12251     },
12252
12253     /**
12254      * Set the defaultUrl used for updates
12255      * @param {String/Function} defaultUrl The url or a function to call to get the url
12256      */
12257     setDefaultUrl : function(defaultUrl){
12258         this.defaultUrl = defaultUrl;
12259     },
12260
12261     /**
12262      * Aborts the executing transaction
12263      */
12264     abort : function(){
12265         if(this.transaction){
12266             Roo.Ajax.abort(this.transaction);
12267         }
12268     },
12269
12270     /**
12271      * Returns true if an update is in progress
12272      * @return {Boolean}
12273      */
12274     isUpdating : function(){
12275         if(this.transaction){
12276             return Roo.Ajax.isLoading(this.transaction);
12277         }
12278         return false;
12279     }
12280 });
12281
12282 /**
12283  * @class Roo.UpdateManager.defaults
12284  * @static (not really - but it helps the doc tool)
12285  * The defaults collection enables customizing the default properties of UpdateManager
12286  */
12287    Roo.UpdateManager.defaults = {
12288        /**
12289          * Timeout for requests or form posts in seconds (Defaults 30 seconds).
12290          * @type Number
12291          */
12292          timeout : 30,
12293
12294          /**
12295          * True to process scripts by default (Defaults to false).
12296          * @type Boolean
12297          */
12298         loadScripts : false,
12299
12300         /**
12301         * Blank page URL to use with SSL file uploads (Defaults to "javascript:false").
12302         * @type String
12303         */
12304         sslBlankUrl : (Roo.SSL_SECURE_URL || "javascript:false"),
12305         /**
12306          * Whether to append unique parameter on get request to disable caching (Defaults to false).
12307          * @type Boolean
12308          */
12309         disableCaching : false,
12310         /**
12311          * Whether to show indicatorText when loading (Defaults to true).
12312          * @type Boolean
12313          */
12314         showLoadIndicator : true,
12315         /**
12316          * Text for loading indicator (Defaults to '&lt;div class="loading-indicator"&gt;Loading...&lt;/div&gt;').
12317          * @type String
12318          */
12319         indicatorText : '<div class="loading-indicator">Loading...</div>'
12320    };
12321
12322 /**
12323  * Static convenience method. This method is deprecated in favor of el.load({url:'foo.php', ...}).
12324  *Usage:
12325  * <pre><code>Roo.UpdateManager.updateElement("my-div", "stuff.php");</code></pre>
12326  * @param {String/HTMLElement/Roo.Element} el The element to update
12327  * @param {String} url The url
12328  * @param {String/Object} params (optional) Url encoded param string or an object of name/value pairs
12329  * @param {Object} options (optional) A config object with any of the UpdateManager properties you want to set - for example: {disableCaching:true, indicatorText: "Loading data..."}
12330  * @static
12331  * @deprecated
12332  * @member Roo.UpdateManager
12333  */
12334 Roo.UpdateManager.updateElement = function(el, url, params, options){
12335     var um = Roo.get(el, true).getUpdateManager();
12336     Roo.apply(um, options);
12337     um.update(url, params, options ? options.callback : null);
12338 };
12339 // alias for backwards compat
12340 Roo.UpdateManager.update = Roo.UpdateManager.updateElement;
12341 /**
12342  * @class Roo.UpdateManager.BasicRenderer
12343  * Default Content renderer. Updates the elements innerHTML with the responseText.
12344  */
12345 Roo.UpdateManager.BasicRenderer = function(){};
12346
12347 Roo.UpdateManager.BasicRenderer.prototype = {
12348     /**
12349      * This is called when the transaction is completed and it's time to update the element - The BasicRenderer
12350      * updates the elements innerHTML with the responseText - To perform a custom render (i.e. XML or JSON processing),
12351      * create an object with a "render(el, response)" method and pass it to setRenderer on the UpdateManager.
12352      * @param {Roo.Element} el The element being rendered
12353      * @param {Object} response The YUI Connect response object
12354      * @param {UpdateManager} updateManager The calling update manager
12355      * @param {Function} callback A callback that will need to be called if loadScripts is true on the UpdateManager
12356      */
12357      render : function(el, response, updateManager, callback){
12358         el.update(response.responseText, updateManager.loadScripts, callback);
12359     }
12360 };
12361 /*
12362  * Based on:
12363  * Roo JS
12364  * (c)) Alan Knowles
12365  * Licence : LGPL
12366  */
12367
12368
12369 /**
12370  * @class Roo.DomTemplate
12371  * @extends Roo.Template
12372  * An effort at a dom based template engine..
12373  *
12374  * Similar to XTemplate, except it uses dom parsing to create the template..
12375  *
12376  * Supported features:
12377  *
12378  *  Tags:
12379
12380 <pre><code>
12381       {a_variable} - output encoded.
12382       {a_variable.format:("Y-m-d")} - call a method on the variable
12383       {a_variable:raw} - unencoded output
12384       {a_variable:toFixed(1,2)} - Roo.util.Format."toFixed"
12385       {a_variable:this.method_on_template(...)} - call a method on the template object.
12386  
12387 </code></pre>
12388  *  The tpl tag:
12389 <pre><code>
12390         &lt;div roo-for="a_variable or condition.."&gt;&lt;/div&gt;
12391         &lt;div roo-if="a_variable or condition"&gt;&lt;/div&gt;
12392         &lt;div roo-exec="some javascript"&gt;&lt;/div&gt;
12393         &lt;div roo-name="named_template"&gt;&lt;/div&gt; 
12394   
12395 </code></pre>
12396  *      
12397  */
12398 Roo.DomTemplate = function()
12399 {
12400      Roo.DomTemplate.superclass.constructor.apply(this, arguments);
12401      if (this.html) {
12402         this.compile();
12403      }
12404 };
12405
12406
12407 Roo.extend(Roo.DomTemplate, Roo.Template, {
12408     /**
12409      * id counter for sub templates.
12410      */
12411     id : 0,
12412     /**
12413      * flag to indicate if dom parser is inside a pre,
12414      * it will strip whitespace if not.
12415      */
12416     inPre : false,
12417     
12418     /**
12419      * The various sub templates
12420      */
12421     tpls : false,
12422     
12423     
12424     
12425     /**
12426      *
12427      * basic tag replacing syntax
12428      * WORD:WORD()
12429      *
12430      * // you can fake an object call by doing this
12431      *  x.t:(test,tesT) 
12432      * 
12433      */
12434     re : /(\{|\%7B)([\w-\.]+)(?:\:([\w\.]*)(?:\(([^)]*?)?\))?)?(\}|\%7D)/g,
12435     //re : /\{([\w-\.]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
12436     
12437     iterChild : function (node, method) {
12438         
12439         var oldPre = this.inPre;
12440         if (node.tagName == 'PRE') {
12441             this.inPre = true;
12442         }
12443         for( var i = 0; i < node.childNodes.length; i++) {
12444             method.call(this, node.childNodes[i]);
12445         }
12446         this.inPre = oldPre;
12447     },
12448     
12449     
12450     
12451     /**
12452      * compile the template
12453      *
12454      * This is not recursive, so I'm not sure how nested templates are really going to be handled..
12455      *
12456      */
12457     compile: function()
12458     {
12459         var s = this.html;
12460         
12461         // covert the html into DOM...
12462         var doc = false;
12463         var div =false;
12464         try {
12465             doc = document.implementation.createHTMLDocument("");
12466             doc.documentElement.innerHTML =   this.html  ;
12467             div = doc.documentElement;
12468         } catch (e) {
12469             // old IE... - nasty -- it causes all sorts of issues.. with
12470             // images getting pulled from server..
12471             div = document.createElement('div');
12472             div.innerHTML = this.html;
12473         }
12474         //doc.documentElement.innerHTML = htmlBody
12475          
12476         
12477         
12478         this.tpls = [];
12479         var _t = this;
12480         this.iterChild(div, function(n) {_t.compileNode(n, true); });
12481         
12482         var tpls = this.tpls;
12483         
12484         // create a top level template from the snippet..
12485         
12486         //Roo.log(div.innerHTML);
12487         
12488         var tpl = {
12489             uid : 'master',
12490             id : this.id++,
12491             attr : false,
12492             value : false,
12493             body : div.innerHTML,
12494             
12495             forCall : false,
12496             execCall : false,
12497             dom : div,
12498             isTop : true
12499             
12500         };
12501         tpls.unshift(tpl);
12502         
12503         
12504         // compile them...
12505         this.tpls = [];
12506         Roo.each(tpls, function(tp){
12507             this.compileTpl(tp);
12508             this.tpls[tp.id] = tp;
12509         }, this);
12510         
12511         this.master = tpls[0];
12512         return this;
12513         
12514         
12515     },
12516     
12517     compileNode : function(node, istop) {
12518         // test for
12519         //Roo.log(node);
12520         
12521         
12522         // skip anything not a tag..
12523         if (node.nodeType != 1) {
12524             if (node.nodeType == 3 && !this.inPre) {
12525                 // reduce white space..
12526                 node.nodeValue = node.nodeValue.replace(/\s+/g, ' '); 
12527                 
12528             }
12529             return;
12530         }
12531         
12532         var tpl = {
12533             uid : false,
12534             id : false,
12535             attr : false,
12536             value : false,
12537             body : '',
12538             
12539             forCall : false,
12540             execCall : false,
12541             dom : false,
12542             isTop : istop
12543             
12544             
12545         };
12546         
12547         
12548         switch(true) {
12549             case (node.hasAttribute('roo-for')): tpl.attr = 'for'; break;
12550             case (node.hasAttribute('roo-if')): tpl.attr = 'if'; break;
12551             case (node.hasAttribute('roo-name')): tpl.attr = 'name'; break;
12552             case (node.hasAttribute('roo-exec')): tpl.attr = 'exec'; break;
12553             // no default..
12554         }
12555         
12556         
12557         if (!tpl.attr) {
12558             // just itterate children..
12559             this.iterChild(node,this.compileNode);
12560             return;
12561         }
12562         tpl.uid = this.id++;
12563         tpl.value = node.getAttribute('roo-' +  tpl.attr);
12564         node.removeAttribute('roo-'+ tpl.attr);
12565         if (tpl.attr != 'name') {
12566             var placeholder = document.createTextNode('{domtpl' + tpl.uid + '}');
12567             node.parentNode.replaceChild(placeholder,  node);
12568         } else {
12569             
12570             var placeholder =  document.createElement('span');
12571             placeholder.className = 'roo-tpl-' + tpl.value;
12572             node.parentNode.replaceChild(placeholder,  node);
12573         }
12574         
12575         // parent now sees '{domtplXXXX}
12576         this.iterChild(node,this.compileNode);
12577         
12578         // we should now have node body...
12579         var div = document.createElement('div');
12580         div.appendChild(node);
12581         tpl.dom = node;
12582         // this has the unfortunate side effect of converting tagged attributes
12583         // eg. href="{...}" into %7C...%7D
12584         // this has been fixed by searching for those combo's although it's a bit hacky..
12585         
12586         
12587         tpl.body = div.innerHTML;
12588         
12589         
12590          
12591         tpl.id = tpl.uid;
12592         switch(tpl.attr) {
12593             case 'for' :
12594                 switch (tpl.value) {
12595                     case '.':  tpl.forCall = new Function('values', 'parent', 'with(values){ return values; }'); break;
12596                     case '..': tpl.forCall= new Function('values', 'parent', 'with(values){ return parent; }'); break;
12597                     default:   tpl.forCall= new Function('values', 'parent', 'with(values){ return '+tpl.value+'; }');
12598                 }
12599                 break;
12600             
12601             case 'exec':
12602                 tpl.execCall = new Function('values', 'parent', 'with(values){ '+(Roo.util.Format.htmlDecode(tpl.value))+'; }');
12603                 break;
12604             
12605             case 'if':     
12606                 tpl.ifCall = new Function('values', 'parent', 'with(values){ return '+(Roo.util.Format.htmlDecode(tpl.value))+'; }');
12607                 break;
12608             
12609             case 'name':
12610                 tpl.id  = tpl.value; // replace non characters???
12611                 break;
12612             
12613         }
12614         
12615         
12616         this.tpls.push(tpl);
12617         
12618         
12619         
12620     },
12621     
12622     
12623     
12624     
12625     /**
12626      * Compile a segment of the template into a 'sub-template'
12627      *
12628      * 
12629      * 
12630      *
12631      */
12632     compileTpl : function(tpl)
12633     {
12634         var fm = Roo.util.Format;
12635         var useF = this.disableFormats !== true;
12636         
12637         var sep = Roo.isGecko ? "+\n" : ",\n";
12638         
12639         var undef = function(str) {
12640             Roo.debug && Roo.log("Property not found :"  + str);
12641             return '';
12642         };
12643           
12644         //Roo.log(tpl.body);
12645         
12646         
12647         
12648         var fn = function(m, lbrace, name, format, args)
12649         {
12650             //Roo.log("ARGS");
12651             //Roo.log(arguments);
12652             args = args ? args.replace(/\\'/g,"'") : args;
12653             //["{TEST:(a,b,c)}", "TEST", "", "a,b,c", 0, "{TEST:(a,b,c)}"]
12654             if (typeof(format) == 'undefined') {
12655                 format =  'htmlEncode'; 
12656             }
12657             if (format == 'raw' ) {
12658                 format = false;
12659             }
12660             
12661             if(name.substr(0, 6) == 'domtpl'){
12662                 return "'"+ sep +'this.applySubTemplate('+name.substr(6)+', values, parent)'+sep+"'";
12663             }
12664             
12665             // build an array of options to determine if value is undefined..
12666             
12667             // basically get 'xxxx.yyyy' then do
12668             // (typeof(xxxx) == 'undefined' || typeof(xxx.yyyy) == 'undefined') ?
12669             //    (function () { Roo.log("Property not found"); return ''; })() :
12670             //    ......
12671             
12672             var udef_ar = [];
12673             var lookfor = '';
12674             Roo.each(name.split('.'), function(st) {
12675                 lookfor += (lookfor.length ? '.': '') + st;
12676                 udef_ar.push(  "(typeof(" + lookfor + ") == 'undefined')"  );
12677             });
12678             
12679             var udef_st = '((' + udef_ar.join(" || ") +") ? undef('" + name + "') : "; // .. needs )
12680             
12681             
12682             if(format && useF){
12683                 
12684                 args = args ? ',' + args : "";
12685                  
12686                 if(format.substr(0, 5) != "this."){
12687                     format = "fm." + format + '(';
12688                 }else{
12689                     format = 'this.call("'+ format.substr(5) + '", ';
12690                     args = ", values";
12691                 }
12692                 
12693                 return "'"+ sep +   udef_st   +    format + name + args + "))"+sep+"'";
12694             }
12695              
12696             if (args && args.length) {
12697                 // called with xxyx.yuu:(test,test)
12698                 // change to ()
12699                 return "'"+ sep + udef_st  + name + '(' +  args + "))"+sep+"'";
12700             }
12701             // raw.. - :raw modifier..
12702             return "'"+ sep + udef_st  + name + ")"+sep+"'";
12703             
12704         };
12705         var body;
12706         // branched to use + in gecko and [].join() in others
12707         if(Roo.isGecko){
12708             body = "tpl.compiled = function(values, parent){  with(values) { return '" +
12709                    tpl.body.replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
12710                     "';};};";
12711         }else{
12712             body = ["tpl.compiled = function(values, parent){  with (values) { return ['"];
12713             body.push(tpl.body.replace(/(\r\n|\n)/g,
12714                             '\\n').replace(/'/g, "\\'").replace(this.re, fn));
12715             body.push("'].join('');};};");
12716             body = body.join('');
12717         }
12718         
12719         Roo.debug && Roo.log(body.replace(/\\n/,'\n'));
12720        
12721         /** eval:var:tpl eval:var:fm eval:var:useF eval:var:undef  */
12722         eval(body);
12723         
12724         return this;
12725     },
12726      
12727     /**
12728      * same as applyTemplate, except it's done to one of the subTemplates
12729      * when using named templates, you can do:
12730      *
12731      * var str = pl.applySubTemplate('your-name', values);
12732      *
12733      * 
12734      * @param {Number} id of the template
12735      * @param {Object} values to apply to template
12736      * @param {Object} parent (normaly the instance of this object)
12737      */
12738     applySubTemplate : function(id, values, parent)
12739     {
12740         
12741         
12742         var t = this.tpls[id];
12743         
12744         
12745         try { 
12746             if(t.ifCall && !t.ifCall.call(this, values, parent)){
12747                 Roo.debug && Roo.log('if call on ' + t.value + ' return false');
12748                 return '';
12749             }
12750         } catch(e) {
12751             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-if="' + t.value + '" - ' + e.toString());
12752             Roo.log(values);
12753           
12754             return '';
12755         }
12756         try { 
12757             
12758             if(t.execCall && t.execCall.call(this, values, parent)){
12759                 return '';
12760             }
12761         } catch(e) {
12762             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-for="' + t.value + '" - ' + e.toString());
12763             Roo.log(values);
12764             return '';
12765         }
12766         
12767         try {
12768             var vs = t.forCall ? t.forCall.call(this, values, parent) : values;
12769             parent = t.target ? values : parent;
12770             if(t.forCall && vs instanceof Array){
12771                 var buf = [];
12772                 for(var i = 0, len = vs.length; i < len; i++){
12773                     try {
12774                         buf[buf.length] = t.compiled.call(this, vs[i], parent);
12775                     } catch (e) {
12776                         Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on body="' + t.value + '" - ' + e.toString());
12777                         Roo.log(e.body);
12778                         //Roo.log(t.compiled);
12779                         Roo.log(vs[i]);
12780                     }   
12781                 }
12782                 return buf.join('');
12783             }
12784         } catch (e) {
12785             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-for="' + t.value + '" - ' + e.toString());
12786             Roo.log(values);
12787             return '';
12788         }
12789         try {
12790             return t.compiled.call(this, vs, parent);
12791         } catch (e) {
12792             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on body="' + t.value + '" - ' + e.toString());
12793             Roo.log(e.body);
12794             //Roo.log(t.compiled);
12795             Roo.log(values);
12796             return '';
12797         }
12798     },
12799
12800    
12801
12802     applyTemplate : function(values){
12803         return this.master.compiled.call(this, values, {});
12804         //var s = this.subs;
12805     },
12806
12807     apply : function(){
12808         return this.applyTemplate.apply(this, arguments);
12809     }
12810
12811  });
12812
12813 Roo.DomTemplate.from = function(el){
12814     el = Roo.getDom(el);
12815     return new Roo.Domtemplate(el.value || el.innerHTML);
12816 };/*
12817  * Based on:
12818  * Ext JS Library 1.1.1
12819  * Copyright(c) 2006-2007, Ext JS, LLC.
12820  *
12821  * Originally Released Under LGPL - original licence link has changed is not relivant.
12822  *
12823  * Fork - LGPL
12824  * <script type="text/javascript">
12825  */
12826
12827 /**
12828  * @class Roo.util.DelayedTask
12829  * Provides a convenient method of performing setTimeout where a new
12830  * timeout cancels the old timeout. An example would be performing validation on a keypress.
12831  * You can use this class to buffer
12832  * the keypress events for a certain number of milliseconds, and perform only if they stop
12833  * for that amount of time.
12834  * @constructor The parameters to this constructor serve as defaults and are not required.
12835  * @param {Function} fn (optional) The default function to timeout
12836  * @param {Object} scope (optional) The default scope of that timeout
12837  * @param {Array} args (optional) The default Array of arguments
12838  */
12839 Roo.util.DelayedTask = function(fn, scope, args){
12840     var id = null, d, t;
12841
12842     var call = function(){
12843         var now = new Date().getTime();
12844         if(now - t >= d){
12845             clearInterval(id);
12846             id = null;
12847             fn.apply(scope, args || []);
12848         }
12849     };
12850     /**
12851      * Cancels any pending timeout and queues a new one
12852      * @param {Number} delay The milliseconds to delay
12853      * @param {Function} newFn (optional) Overrides function passed to constructor
12854      * @param {Object} newScope (optional) Overrides scope passed to constructor
12855      * @param {Array} newArgs (optional) Overrides args passed to constructor
12856      */
12857     this.delay = function(delay, newFn, newScope, newArgs){
12858         if(id && delay != d){
12859             this.cancel();
12860         }
12861         d = delay;
12862         t = new Date().getTime();
12863         fn = newFn || fn;
12864         scope = newScope || scope;
12865         args = newArgs || args;
12866         if(!id){
12867             id = setInterval(call, d);
12868         }
12869     };
12870
12871     /**
12872      * Cancel the last queued timeout
12873      */
12874     this.cancel = function(){
12875         if(id){
12876             clearInterval(id);
12877             id = null;
12878         }
12879     };
12880 };/*
12881  * Based on:
12882  * Ext JS Library 1.1.1
12883  * Copyright(c) 2006-2007, Ext JS, LLC.
12884  *
12885  * Originally Released Under LGPL - original licence link has changed is not relivant.
12886  *
12887  * Fork - LGPL
12888  * <script type="text/javascript">
12889  */
12890  
12891  
12892 Roo.util.TaskRunner = function(interval){
12893     interval = interval || 10;
12894     var tasks = [], removeQueue = [];
12895     var id = 0;
12896     var running = false;
12897
12898     var stopThread = function(){
12899         running = false;
12900         clearInterval(id);
12901         id = 0;
12902     };
12903
12904     var startThread = function(){
12905         if(!running){
12906             running = true;
12907             id = setInterval(runTasks, interval);
12908         }
12909     };
12910
12911     var removeTask = function(task){
12912         removeQueue.push(task);
12913         if(task.onStop){
12914             task.onStop();
12915         }
12916     };
12917
12918     var runTasks = function(){
12919         if(removeQueue.length > 0){
12920             for(var i = 0, len = removeQueue.length; i < len; i++){
12921                 tasks.remove(removeQueue[i]);
12922             }
12923             removeQueue = [];
12924             if(tasks.length < 1){
12925                 stopThread();
12926                 return;
12927             }
12928         }
12929         var now = new Date().getTime();
12930         for(var i = 0, len = tasks.length; i < len; ++i){
12931             var t = tasks[i];
12932             var itime = now - t.taskRunTime;
12933             if(t.interval <= itime){
12934                 var rt = t.run.apply(t.scope || t, t.args || [++t.taskRunCount]);
12935                 t.taskRunTime = now;
12936                 if(rt === false || t.taskRunCount === t.repeat){
12937                     removeTask(t);
12938                     return;
12939                 }
12940             }
12941             if(t.duration && t.duration <= (now - t.taskStartTime)){
12942                 removeTask(t);
12943             }
12944         }
12945     };
12946
12947     /**
12948      * Queues a new task.
12949      * @param {Object} task
12950      */
12951     this.start = function(task){
12952         tasks.push(task);
12953         task.taskStartTime = new Date().getTime();
12954         task.taskRunTime = 0;
12955         task.taskRunCount = 0;
12956         startThread();
12957         return task;
12958     };
12959
12960     this.stop = function(task){
12961         removeTask(task);
12962         return task;
12963     };
12964
12965     this.stopAll = function(){
12966         stopThread();
12967         for(var i = 0, len = tasks.length; i < len; i++){
12968             if(tasks[i].onStop){
12969                 tasks[i].onStop();
12970             }
12971         }
12972         tasks = [];
12973         removeQueue = [];
12974     };
12975 };
12976
12977 Roo.TaskMgr = new Roo.util.TaskRunner();/*
12978  * Based on:
12979  * Ext JS Library 1.1.1
12980  * Copyright(c) 2006-2007, Ext JS, LLC.
12981  *
12982  * Originally Released Under LGPL - original licence link has changed is not relivant.
12983  *
12984  * Fork - LGPL
12985  * <script type="text/javascript">
12986  */
12987
12988  
12989 /**
12990  * @class Roo.util.MixedCollection
12991  * @extends Roo.util.Observable
12992  * A Collection class that maintains both numeric indexes and keys and exposes events.
12993  * @constructor
12994  * @param {Boolean} allowFunctions True if the addAll function should add function references to the
12995  * collection (defaults to false)
12996  * @param {Function} keyFn A function that can accept an item of the type(s) stored in this MixedCollection
12997  * and return the key value for that item.  This is used when available to look up the key on items that
12998  * were passed without an explicit key parameter to a MixedCollection method.  Passing this parameter is
12999  * equivalent to providing an implementation for the {@link #getKey} method.
13000  */
13001 Roo.util.MixedCollection = function(allowFunctions, keyFn){
13002     this.items = [];
13003     this.map = {};
13004     this.keys = [];
13005     this.length = 0;
13006     this.addEvents({
13007         /**
13008          * @event clear
13009          * Fires when the collection is cleared.
13010          */
13011         "clear" : true,
13012         /**
13013          * @event add
13014          * Fires when an item is added to the collection.
13015          * @param {Number} index The index at which the item was added.
13016          * @param {Object} o The item added.
13017          * @param {String} key The key associated with the added item.
13018          */
13019         "add" : true,
13020         /**
13021          * @event replace
13022          * Fires when an item is replaced in the collection.
13023          * @param {String} key he key associated with the new added.
13024          * @param {Object} old The item being replaced.
13025          * @param {Object} new The new item.
13026          */
13027         "replace" : true,
13028         /**
13029          * @event remove
13030          * Fires when an item is removed from the collection.
13031          * @param {Object} o The item being removed.
13032          * @param {String} key (optional) The key associated with the removed item.
13033          */
13034         "remove" : true,
13035         "sort" : true
13036     });
13037     this.allowFunctions = allowFunctions === true;
13038     if(keyFn){
13039         this.getKey = keyFn;
13040     }
13041     Roo.util.MixedCollection.superclass.constructor.call(this);
13042 };
13043
13044 Roo.extend(Roo.util.MixedCollection, Roo.util.Observable, {
13045     allowFunctions : false,
13046     
13047 /**
13048  * Adds an item to the collection.
13049  * @param {String} key The key to associate with the item
13050  * @param {Object} o The item to add.
13051  * @return {Object} The item added.
13052  */
13053     add : function(key, o){
13054         if(arguments.length == 1){
13055             o = arguments[0];
13056             key = this.getKey(o);
13057         }
13058         if(typeof key == "undefined" || key === null){
13059             this.length++;
13060             this.items.push(o);
13061             this.keys.push(null);
13062         }else{
13063             var old = this.map[key];
13064             if(old){
13065                 return this.replace(key, o);
13066             }
13067             this.length++;
13068             this.items.push(o);
13069             this.map[key] = o;
13070             this.keys.push(key);
13071         }
13072         this.fireEvent("add", this.length-1, o, key);
13073         return o;
13074     },
13075        
13076 /**
13077   * MixedCollection has a generic way to fetch keys if you implement getKey.
13078 <pre><code>
13079 // normal way
13080 var mc = new Roo.util.MixedCollection();
13081 mc.add(someEl.dom.id, someEl);
13082 mc.add(otherEl.dom.id, otherEl);
13083 //and so on
13084
13085 // using getKey
13086 var mc = new Roo.util.MixedCollection();
13087 mc.getKey = function(el){
13088    return el.dom.id;
13089 };
13090 mc.add(someEl);
13091 mc.add(otherEl);
13092
13093 // or via the constructor
13094 var mc = new Roo.util.MixedCollection(false, function(el){
13095    return el.dom.id;
13096 });
13097 mc.add(someEl);
13098 mc.add(otherEl);
13099 </code></pre>
13100  * @param o {Object} The item for which to find the key.
13101  * @return {Object} The key for the passed item.
13102  */
13103     getKey : function(o){
13104          return o.id; 
13105     },
13106    
13107 /**
13108  * Replaces an item in the collection.
13109  * @param {String} key The key associated with the item to replace, or the item to replace.
13110  * @param o {Object} o (optional) If the first parameter passed was a key, the item to associate with that key.
13111  * @return {Object}  The new item.
13112  */
13113     replace : function(key, o){
13114         if(arguments.length == 1){
13115             o = arguments[0];
13116             key = this.getKey(o);
13117         }
13118         var old = this.item(key);
13119         if(typeof key == "undefined" || key === null || typeof old == "undefined"){
13120              return this.add(key, o);
13121         }
13122         var index = this.indexOfKey(key);
13123         this.items[index] = o;
13124         this.map[key] = o;
13125         this.fireEvent("replace", key, old, o);
13126         return o;
13127     },
13128    
13129 /**
13130  * Adds all elements of an Array or an Object to the collection.
13131  * @param {Object/Array} objs An Object containing properties which will be added to the collection, or
13132  * an Array of values, each of which are added to the collection.
13133  */
13134     addAll : function(objs){
13135         if(arguments.length > 1 || objs instanceof Array){
13136             var args = arguments.length > 1 ? arguments : objs;
13137             for(var i = 0, len = args.length; i < len; i++){
13138                 this.add(args[i]);
13139             }
13140         }else{
13141             for(var key in objs){
13142                 if(this.allowFunctions || typeof objs[key] != "function"){
13143                     this.add(key, objs[key]);
13144                 }
13145             }
13146         }
13147     },
13148    
13149 /**
13150  * Executes the specified function once for every item in the collection, passing each
13151  * item as the first and only parameter. returning false from the function will stop the iteration.
13152  * @param {Function} fn The function to execute for each item.
13153  * @param {Object} scope (optional) The scope in which to execute the function.
13154  */
13155     each : function(fn, scope){
13156         var items = [].concat(this.items); // each safe for removal
13157         for(var i = 0, len = items.length; i < len; i++){
13158             if(fn.call(scope || items[i], items[i], i, len) === false){
13159                 break;
13160             }
13161         }
13162     },
13163    
13164 /**
13165  * Executes the specified function once for every key in the collection, passing each
13166  * key, and its associated item as the first two parameters.
13167  * @param {Function} fn The function to execute for each item.
13168  * @param {Object} scope (optional) The scope in which to execute the function.
13169  */
13170     eachKey : function(fn, scope){
13171         for(var i = 0, len = this.keys.length; i < len; i++){
13172             fn.call(scope || window, this.keys[i], this.items[i], i, len);
13173         }
13174     },
13175    
13176 /**
13177  * Returns the first item in the collection which elicits a true return value from the
13178  * passed selection function.
13179  * @param {Function} fn The selection function to execute for each item.
13180  * @param {Object} scope (optional) The scope in which to execute the function.
13181  * @return {Object} The first item in the collection which returned true from the selection function.
13182  */
13183     find : function(fn, scope){
13184         for(var i = 0, len = this.items.length; i < len; i++){
13185             if(fn.call(scope || window, this.items[i], this.keys[i])){
13186                 return this.items[i];
13187             }
13188         }
13189         return null;
13190     },
13191    
13192 /**
13193  * Inserts an item at the specified index in the collection.
13194  * @param {Number} index The index to insert the item at.
13195  * @param {String} key The key to associate with the new item, or the item itself.
13196  * @param {Object} o  (optional) If the second parameter was a key, the new item.
13197  * @return {Object} The item inserted.
13198  */
13199     insert : function(index, key, o){
13200         if(arguments.length == 2){
13201             o = arguments[1];
13202             key = this.getKey(o);
13203         }
13204         if(index >= this.length){
13205             return this.add(key, o);
13206         }
13207         this.length++;
13208         this.items.splice(index, 0, o);
13209         if(typeof key != "undefined" && key != null){
13210             this.map[key] = o;
13211         }
13212         this.keys.splice(index, 0, key);
13213         this.fireEvent("add", index, o, key);
13214         return o;
13215     },
13216    
13217 /**
13218  * Removed an item from the collection.
13219  * @param {Object} o The item to remove.
13220  * @return {Object} The item removed.
13221  */
13222     remove : function(o){
13223         return this.removeAt(this.indexOf(o));
13224     },
13225    
13226 /**
13227  * Remove an item from a specified index in the collection.
13228  * @param {Number} index The index within the collection of the item to remove.
13229  */
13230     removeAt : function(index){
13231         if(index < this.length && index >= 0){
13232             this.length--;
13233             var o = this.items[index];
13234             this.items.splice(index, 1);
13235             var key = this.keys[index];
13236             if(typeof key != "undefined"){
13237                 delete this.map[key];
13238             }
13239             this.keys.splice(index, 1);
13240             this.fireEvent("remove", o, key);
13241         }
13242     },
13243    
13244 /**
13245  * Removed an item associated with the passed key fom the collection.
13246  * @param {String} key The key of the item to remove.
13247  */
13248     removeKey : function(key){
13249         return this.removeAt(this.indexOfKey(key));
13250     },
13251    
13252 /**
13253  * Returns the number of items in the collection.
13254  * @return {Number} the number of items in the collection.
13255  */
13256     getCount : function(){
13257         return this.length; 
13258     },
13259    
13260 /**
13261  * Returns index within the collection of the passed Object.
13262  * @param {Object} o The item to find the index of.
13263  * @return {Number} index of the item.
13264  */
13265     indexOf : function(o){
13266         if(!this.items.indexOf){
13267             for(var i = 0, len = this.items.length; i < len; i++){
13268                 if(this.items[i] == o) {
13269                     return i;
13270                 }
13271             }
13272             return -1;
13273         }else{
13274             return this.items.indexOf(o);
13275         }
13276     },
13277    
13278 /**
13279  * Returns index within the collection of the passed key.
13280  * @param {String} key The key to find the index of.
13281  * @return {Number} index of the key.
13282  */
13283     indexOfKey : function(key){
13284         if(!this.keys.indexOf){
13285             for(var i = 0, len = this.keys.length; i < len; i++){
13286                 if(this.keys[i] == key) {
13287                     return i;
13288                 }
13289             }
13290             return -1;
13291         }else{
13292             return this.keys.indexOf(key);
13293         }
13294     },
13295    
13296 /**
13297  * Returns the item associated with the passed key OR index. Key has priority over index.
13298  * @param {String/Number} key The key or index of the item.
13299  * @return {Object} The item associated with the passed key.
13300  */
13301     item : function(key){
13302         var item = typeof this.map[key] != "undefined" ? this.map[key] : this.items[key];
13303         return typeof item != 'function' || this.allowFunctions ? item : null; // for prototype!
13304     },
13305     
13306 /**
13307  * Returns the item at the specified index.
13308  * @param {Number} index The index of the item.
13309  * @return {Object}
13310  */
13311     itemAt : function(index){
13312         return this.items[index];
13313     },
13314     
13315 /**
13316  * Returns the item associated with the passed key.
13317  * @param {String/Number} key The key of the item.
13318  * @return {Object} The item associated with the passed key.
13319  */
13320     key : function(key){
13321         return this.map[key];
13322     },
13323    
13324 /**
13325  * Returns true if the collection contains the passed Object as an item.
13326  * @param {Object} o  The Object to look for in the collection.
13327  * @return {Boolean} True if the collection contains the Object as an item.
13328  */
13329     contains : function(o){
13330         return this.indexOf(o) != -1;
13331     },
13332    
13333 /**
13334  * Returns true if the collection contains the passed Object as a key.
13335  * @param {String} key The key to look for in the collection.
13336  * @return {Boolean} True if the collection contains the Object as a key.
13337  */
13338     containsKey : function(key){
13339         return typeof this.map[key] != "undefined";
13340     },
13341    
13342 /**
13343  * Removes all items from the collection.
13344  */
13345     clear : function(){
13346         this.length = 0;
13347         this.items = [];
13348         this.keys = [];
13349         this.map = {};
13350         this.fireEvent("clear");
13351     },
13352    
13353 /**
13354  * Returns the first item in the collection.
13355  * @return {Object} the first item in the collection..
13356  */
13357     first : function(){
13358         return this.items[0]; 
13359     },
13360    
13361 /**
13362  * Returns the last item in the collection.
13363  * @return {Object} the last item in the collection..
13364  */
13365     last : function(){
13366         return this.items[this.length-1];   
13367     },
13368     
13369     _sort : function(property, dir, fn){
13370         var dsc = String(dir).toUpperCase() == "DESC" ? -1 : 1;
13371         fn = fn || function(a, b){
13372             return a-b;
13373         };
13374         var c = [], k = this.keys, items = this.items;
13375         for(var i = 0, len = items.length; i < len; i++){
13376             c[c.length] = {key: k[i], value: items[i], index: i};
13377         }
13378         c.sort(function(a, b){
13379             var v = fn(a[property], b[property]) * dsc;
13380             if(v == 0){
13381                 v = (a.index < b.index ? -1 : 1);
13382             }
13383             return v;
13384         });
13385         for(var i = 0, len = c.length; i < len; i++){
13386             items[i] = c[i].value;
13387             k[i] = c[i].key;
13388         }
13389         this.fireEvent("sort", this);
13390     },
13391     
13392     /**
13393      * Sorts this collection with the passed comparison function
13394      * @param {String} direction (optional) "ASC" or "DESC"
13395      * @param {Function} fn (optional) comparison function
13396      */
13397     sort : function(dir, fn){
13398         this._sort("value", dir, fn);
13399     },
13400     
13401     /**
13402      * Sorts this collection by keys
13403      * @param {String} direction (optional) "ASC" or "DESC"
13404      * @param {Function} fn (optional) a comparison function (defaults to case insensitive string)
13405      */
13406     keySort : function(dir, fn){
13407         this._sort("key", dir, fn || function(a, b){
13408             return String(a).toUpperCase()-String(b).toUpperCase();
13409         });
13410     },
13411     
13412     /**
13413      * Returns a range of items in this collection
13414      * @param {Number} startIndex (optional) defaults to 0
13415      * @param {Number} endIndex (optional) default to the last item
13416      * @return {Array} An array of items
13417      */
13418     getRange : function(start, end){
13419         var items = this.items;
13420         if(items.length < 1){
13421             return [];
13422         }
13423         start = start || 0;
13424         end = Math.min(typeof end == "undefined" ? this.length-1 : end, this.length-1);
13425         var r = [];
13426         if(start <= end){
13427             for(var i = start; i <= end; i++) {
13428                     r[r.length] = items[i];
13429             }
13430         }else{
13431             for(var i = start; i >= end; i--) {
13432                     r[r.length] = items[i];
13433             }
13434         }
13435         return r;
13436     },
13437         
13438     /**
13439      * Filter the <i>objects</i> in this collection by a specific property. 
13440      * Returns a new collection that has been filtered.
13441      * @param {String} property A property on your objects
13442      * @param {String/RegExp} value Either string that the property values 
13443      * should start with or a RegExp to test against the property
13444      * @return {MixedCollection} The new filtered collection
13445      */
13446     filter : function(property, value){
13447         if(!value.exec){ // not a regex
13448             value = String(value);
13449             if(value.length == 0){
13450                 return this.clone();
13451             }
13452             value = new RegExp("^" + Roo.escapeRe(value), "i");
13453         }
13454         return this.filterBy(function(o){
13455             return o && value.test(o[property]);
13456         });
13457         },
13458     
13459     /**
13460      * Filter by a function. * Returns a new collection that has been filtered.
13461      * The passed function will be called with each 
13462      * object in the collection. If the function returns true, the value is included 
13463      * otherwise it is filtered.
13464      * @param {Function} fn The function to be called, it will receive the args o (the object), k (the key)
13465      * @param {Object} scope (optional) The scope of the function (defaults to this) 
13466      * @return {MixedCollection} The new filtered collection
13467      */
13468     filterBy : function(fn, scope){
13469         var r = new Roo.util.MixedCollection();
13470         r.getKey = this.getKey;
13471         var k = this.keys, it = this.items;
13472         for(var i = 0, len = it.length; i < len; i++){
13473             if(fn.call(scope||this, it[i], k[i])){
13474                                 r.add(k[i], it[i]);
13475                         }
13476         }
13477         return r;
13478     },
13479     
13480     /**
13481      * Creates a duplicate of this collection
13482      * @return {MixedCollection}
13483      */
13484     clone : function(){
13485         var r = new Roo.util.MixedCollection();
13486         var k = this.keys, it = this.items;
13487         for(var i = 0, len = it.length; i < len; i++){
13488             r.add(k[i], it[i]);
13489         }
13490         r.getKey = this.getKey;
13491         return r;
13492     }
13493 });
13494 /**
13495  * Returns the item associated with the passed key or index.
13496  * @method
13497  * @param {String/Number} key The key or index of the item.
13498  * @return {Object} The item associated with the passed key.
13499  */
13500 Roo.util.MixedCollection.prototype.get = Roo.util.MixedCollection.prototype.item;/*
13501  * Based on:
13502  * Ext JS Library 1.1.1
13503  * Copyright(c) 2006-2007, Ext JS, LLC.
13504  *
13505  * Originally Released Under LGPL - original licence link has changed is not relivant.
13506  *
13507  * Fork - LGPL
13508  * <script type="text/javascript">
13509  */
13510 /**
13511  * @class Roo.util.JSON
13512  * Modified version of Douglas Crockford"s json.js that doesn"t
13513  * mess with the Object prototype 
13514  * http://www.json.org/js.html
13515  * @singleton
13516  */
13517 Roo.util.JSON = new (function(){
13518     var useHasOwn = {}.hasOwnProperty ? true : false;
13519     
13520     // crashes Safari in some instances
13521     //var validRE = /^("(\\.|[^"\\\n\r])*?"|[,:{}\[\]0-9.\-+Eaeflnr-u \n\r\t])+?$/;
13522     
13523     var pad = function(n) {
13524         return n < 10 ? "0" + n : n;
13525     };
13526     
13527     var m = {
13528         "\b": '\\b',
13529         "\t": '\\t',
13530         "\n": '\\n',
13531         "\f": '\\f',
13532         "\r": '\\r',
13533         '"' : '\\"',
13534         "\\": '\\\\'
13535     };
13536
13537     var encodeString = function(s){
13538         if (/["\\\x00-\x1f]/.test(s)) {
13539             return '"' + s.replace(/([\x00-\x1f\\"])/g, function(a, b) {
13540                 var c = m[b];
13541                 if(c){
13542                     return c;
13543                 }
13544                 c = b.charCodeAt();
13545                 return "\\u00" +
13546                     Math.floor(c / 16).toString(16) +
13547                     (c % 16).toString(16);
13548             }) + '"';
13549         }
13550         return '"' + s + '"';
13551     };
13552     
13553     var encodeArray = function(o){
13554         var a = ["["], b, i, l = o.length, v;
13555             for (i = 0; i < l; i += 1) {
13556                 v = o[i];
13557                 switch (typeof v) {
13558                     case "undefined":
13559                     case "function":
13560                     case "unknown":
13561                         break;
13562                     default:
13563                         if (b) {
13564                             a.push(',');
13565                         }
13566                         a.push(v === null ? "null" : Roo.util.JSON.encode(v));
13567                         b = true;
13568                 }
13569             }
13570             a.push("]");
13571             return a.join("");
13572     };
13573     
13574     var encodeDate = function(o){
13575         return '"' + o.getFullYear() + "-" +
13576                 pad(o.getMonth() + 1) + "-" +
13577                 pad(o.getDate()) + "T" +
13578                 pad(o.getHours()) + ":" +
13579                 pad(o.getMinutes()) + ":" +
13580                 pad(o.getSeconds()) + '"';
13581     };
13582     
13583     /**
13584      * Encodes an Object, Array or other value
13585      * @param {Mixed} o The variable to encode
13586      * @return {String} The JSON string
13587      */
13588     this.encode = function(o)
13589     {
13590         // should this be extended to fully wrap stringify..
13591         
13592         if(typeof o == "undefined" || o === null){
13593             return "null";
13594         }else if(o instanceof Array){
13595             return encodeArray(o);
13596         }else if(o instanceof Date){
13597             return encodeDate(o);
13598         }else if(typeof o == "string"){
13599             return encodeString(o);
13600         }else if(typeof o == "number"){
13601             return isFinite(o) ? String(o) : "null";
13602         }else if(typeof o == "boolean"){
13603             return String(o);
13604         }else {
13605             var a = ["{"], b, i, v;
13606             for (i in o) {
13607                 if(!useHasOwn || o.hasOwnProperty(i)) {
13608                     v = o[i];
13609                     switch (typeof v) {
13610                     case "undefined":
13611                     case "function":
13612                     case "unknown":
13613                         break;
13614                     default:
13615                         if(b){
13616                             a.push(',');
13617                         }
13618                         a.push(this.encode(i), ":",
13619                                 v === null ? "null" : this.encode(v));
13620                         b = true;
13621                     }
13622                 }
13623             }
13624             a.push("}");
13625             return a.join("");
13626         }
13627     };
13628     
13629     /**
13630      * Decodes (parses) a JSON string to an object. If the JSON is invalid, this function throws a SyntaxError.
13631      * @param {String} json The JSON string
13632      * @return {Object} The resulting object
13633      */
13634     this.decode = function(json){
13635         
13636         return  /** eval:var:json */ eval("(" + json + ')');
13637     };
13638 })();
13639 /** 
13640  * Shorthand for {@link Roo.util.JSON#encode}
13641  * @member Roo encode 
13642  * @method */
13643 Roo.encode = typeof(JSON) != 'undefined' && JSON.stringify ? JSON.stringify : Roo.util.JSON.encode;
13644 /** 
13645  * Shorthand for {@link Roo.util.JSON#decode}
13646  * @member Roo decode 
13647  * @method */
13648 Roo.decode = typeof(JSON) != 'undefined' && JSON.parse ? JSON.parse : Roo.util.JSON.decode;
13649 /*
13650  * Based on:
13651  * Ext JS Library 1.1.1
13652  * Copyright(c) 2006-2007, Ext JS, LLC.
13653  *
13654  * Originally Released Under LGPL - original licence link has changed is not relivant.
13655  *
13656  * Fork - LGPL
13657  * <script type="text/javascript">
13658  */
13659  
13660 /**
13661  * @class Roo.util.Format
13662  * Reusable data formatting functions
13663  * @singleton
13664  */
13665 Roo.util.Format = function(){
13666     var trimRe = /^\s+|\s+$/g;
13667     return {
13668         /**
13669          * Truncate a string and add an ellipsis ('...') to the end if it exceeds the specified length
13670          * @param {String} value The string to truncate
13671          * @param {Number} length The maximum length to allow before truncating
13672          * @return {String} The converted text
13673          */
13674         ellipsis : function(value, len){
13675             if(value && value.length > len){
13676                 return value.substr(0, len-3)+"...";
13677             }
13678             return value;
13679         },
13680
13681         /**
13682          * Checks a reference and converts it to empty string if it is undefined
13683          * @param {Mixed} value Reference to check
13684          * @return {Mixed} Empty string if converted, otherwise the original value
13685          */
13686         undef : function(value){
13687             return typeof value != "undefined" ? value : "";
13688         },
13689
13690         /**
13691          * Convert certain characters (&, <, >, and ') to their HTML character equivalents for literal display in web pages.
13692          * @param {String} value The string to encode
13693          * @return {String} The encoded text
13694          */
13695         htmlEncode : function(value){
13696             return !value ? value : String(value).replace(/&/g, "&amp;").replace(/>/g, "&gt;").replace(/</g, "&lt;").replace(/"/g, "&quot;");
13697         },
13698
13699         /**
13700          * Convert certain characters (&, <, >, and ') from their HTML character equivalents.
13701          * @param {String} value The string to decode
13702          * @return {String} The decoded text
13703          */
13704         htmlDecode : function(value){
13705             return !value ? value : String(value).replace(/&amp;/g, "&").replace(/&gt;/g, ">").replace(/&lt;/g, "<").replace(/&quot;/g, '"');
13706         },
13707
13708         /**
13709          * Trims any whitespace from either side of a string
13710          * @param {String} value The text to trim
13711          * @return {String} The trimmed text
13712          */
13713         trim : function(value){
13714             return String(value).replace(trimRe, "");
13715         },
13716
13717         /**
13718          * Returns a substring from within an original string
13719          * @param {String} value The original text
13720          * @param {Number} start The start index of the substring
13721          * @param {Number} length The length of the substring
13722          * @return {String} The substring
13723          */
13724         substr : function(value, start, length){
13725             return String(value).substr(start, length);
13726         },
13727
13728         /**
13729          * Converts a string to all lower case letters
13730          * @param {String} value The text to convert
13731          * @return {String} The converted text
13732          */
13733         lowercase : function(value){
13734             return String(value).toLowerCase();
13735         },
13736
13737         /**
13738          * Converts a string to all upper case letters
13739          * @param {String} value The text to convert
13740          * @return {String} The converted text
13741          */
13742         uppercase : function(value){
13743             return String(value).toUpperCase();
13744         },
13745
13746         /**
13747          * Converts the first character only of a string to upper case
13748          * @param {String} value The text to convert
13749          * @return {String} The converted text
13750          */
13751         capitalize : function(value){
13752             return !value ? value : value.charAt(0).toUpperCase() + value.substr(1).toLowerCase();
13753         },
13754
13755         // private
13756         call : function(value, fn){
13757             if(arguments.length > 2){
13758                 var args = Array.prototype.slice.call(arguments, 2);
13759                 args.unshift(value);
13760                  
13761                 return /** eval:var:value */  eval(fn).apply(window, args);
13762             }else{
13763                 /** eval:var:value */
13764                 return /** eval:var:value */ eval(fn).call(window, value);
13765             }
13766         },
13767
13768        
13769         /**
13770          * safer version of Math.toFixed..??/
13771          * @param {Number/String} value The numeric value to format
13772          * @param {Number/String} value Decimal places 
13773          * @return {String} The formatted currency string
13774          */
13775         toFixed : function(v, n)
13776         {
13777             // why not use to fixed - precision is buggered???
13778             if (!n) {
13779                 return Math.round(v-0);
13780             }
13781             var fact = Math.pow(10,n+1);
13782             v = (Math.round((v-0)*fact))/fact;
13783             var z = (''+fact).substring(2);
13784             if (v == Math.floor(v)) {
13785                 return Math.floor(v) + '.' + z;
13786             }
13787             
13788             // now just padd decimals..
13789             var ps = String(v).split('.');
13790             var fd = (ps[1] + z);
13791             var r = fd.substring(0,n); 
13792             var rm = fd.substring(n); 
13793             if (rm < 5) {
13794                 return ps[0] + '.' + r;
13795             }
13796             r*=1; // turn it into a number;
13797             r++;
13798             if (String(r).length != n) {
13799                 ps[0]*=1;
13800                 ps[0]++;
13801                 r = String(r).substring(1); // chop the end off.
13802             }
13803             
13804             return ps[0] + '.' + r;
13805              
13806         },
13807         
13808         /**
13809          * Format a number as US currency
13810          * @param {Number/String} value The numeric value to format
13811          * @return {String} The formatted currency string
13812          */
13813         usMoney : function(v){
13814             return '$' + Roo.util.Format.number(v);
13815         },
13816         
13817         /**
13818          * Format a number
13819          * eventually this should probably emulate php's number_format
13820          * @param {Number/String} value The numeric value to format
13821          * @param {Number} decimals number of decimal places
13822          * @param {String} delimiter for thousands (default comma)
13823          * @return {String} The formatted currency string
13824          */
13825         number : function(v, decimals, thousandsDelimiter)
13826         {
13827             // multiply and round.
13828             decimals = typeof(decimals) == 'undefined' ? 2 : decimals;
13829             thousandsDelimiter = typeof(thousandsDelimiter) == 'undefined' ? ',' : thousandsDelimiter;
13830             
13831             var mul = Math.pow(10, decimals);
13832             var zero = String(mul).substring(1);
13833             v = (Math.round((v-0)*mul))/mul;
13834             
13835             // if it's '0' number.. then
13836             
13837             //v = (v == Math.floor(v)) ? v + "." + zero : ((v*10 == Math.floor(v*10)) ? v + "0" : v);
13838             v = String(v);
13839             var ps = v.split('.');
13840             var whole = ps[0];
13841             
13842             var r = /(\d+)(\d{3})/;
13843             // add comma's
13844             
13845             if(thousandsDelimiter.length != 0) {
13846                 whole = whole.replace(/\B(?=(\d{3})+(?!\d))/g, thousandsDelimiter );
13847             } 
13848             
13849             var sub = ps[1] ?
13850                     // has decimals..
13851                     (decimals ?  ('.'+ ps[1] + zero.substring(ps[1].length)) : '') :
13852                     // does not have decimals
13853                     (decimals ? ('.' + zero) : '');
13854             
13855             
13856             return whole + sub ;
13857         },
13858         
13859         /**
13860          * Parse a value into a formatted date using the specified format pattern.
13861          * @param {Mixed} value The value to format
13862          * @param {String} format (optional) Any valid date format string (defaults to 'm/d/Y')
13863          * @return {String} The formatted date string
13864          */
13865         date : function(v, format){
13866             if(!v){
13867                 return "";
13868             }
13869             if(!(v instanceof Date)){
13870                 v = new Date(Date.parse(v));
13871             }
13872             return v.dateFormat(format || Roo.util.Format.defaults.date);
13873         },
13874
13875         /**
13876          * Returns a date rendering function that can be reused to apply a date format multiple times efficiently
13877          * @param {String} format Any valid date format string
13878          * @return {Function} The date formatting function
13879          */
13880         dateRenderer : function(format){
13881             return function(v){
13882                 return Roo.util.Format.date(v, format);  
13883             };
13884         },
13885
13886         // private
13887         stripTagsRE : /<\/?[^>]+>/gi,
13888         
13889         /**
13890          * Strips all HTML tags
13891          * @param {Mixed} value The text from which to strip tags
13892          * @return {String} The stripped text
13893          */
13894         stripTags : function(v){
13895             return !v ? v : String(v).replace(this.stripTagsRE, "");
13896         }
13897     };
13898 }();
13899 Roo.util.Format.defaults = {
13900     date : 'd/M/Y'
13901 };/*
13902  * Based on:
13903  * Ext JS Library 1.1.1
13904  * Copyright(c) 2006-2007, Ext JS, LLC.
13905  *
13906  * Originally Released Under LGPL - original licence link has changed is not relivant.
13907  *
13908  * Fork - LGPL
13909  * <script type="text/javascript">
13910  */
13911
13912
13913  
13914
13915 /**
13916  * @class Roo.MasterTemplate
13917  * @extends Roo.Template
13918  * Provides a template that can have child templates. The syntax is:
13919 <pre><code>
13920 var t = new Roo.MasterTemplate(
13921         '&lt;select name="{name}"&gt;',
13922                 '&lt;tpl name="options"&gt;&lt;option value="{value:trim}"&gt;{text:ellipsis(10)}&lt;/option&gt;&lt;/tpl&gt;',
13923         '&lt;/select&gt;'
13924 );
13925 t.add('options', {value: 'foo', text: 'bar'});
13926 // or you can add multiple child elements in one shot
13927 t.addAll('options', [
13928     {value: 'foo', text: 'bar'},
13929     {value: 'foo2', text: 'bar2'},
13930     {value: 'foo3', text: 'bar3'}
13931 ]);
13932 // then append, applying the master template values
13933 t.append('my-form', {name: 'my-select'});
13934 </code></pre>
13935 * A name attribute for the child template is not required if you have only one child
13936 * template or you want to refer to them by index.
13937  */
13938 Roo.MasterTemplate = function(){
13939     Roo.MasterTemplate.superclass.constructor.apply(this, arguments);
13940     this.originalHtml = this.html;
13941     var st = {};
13942     var m, re = this.subTemplateRe;
13943     re.lastIndex = 0;
13944     var subIndex = 0;
13945     while(m = re.exec(this.html)){
13946         var name = m[1], content = m[2];
13947         st[subIndex] = {
13948             name: name,
13949             index: subIndex,
13950             buffer: [],
13951             tpl : new Roo.Template(content)
13952         };
13953         if(name){
13954             st[name] = st[subIndex];
13955         }
13956         st[subIndex].tpl.compile();
13957         st[subIndex].tpl.call = this.call.createDelegate(this);
13958         subIndex++;
13959     }
13960     this.subCount = subIndex;
13961     this.subs = st;
13962 };
13963 Roo.extend(Roo.MasterTemplate, Roo.Template, {
13964     /**
13965     * The regular expression used to match sub templates
13966     * @type RegExp
13967     * @property
13968     */
13969     subTemplateRe : /<tpl(?:\sname="([\w-]+)")?>((?:.|\n)*?)<\/tpl>/gi,
13970
13971     /**
13972      * Applies the passed values to a child template.
13973      * @param {String/Number} name (optional) The name or index of the child template
13974      * @param {Array/Object} values The values to be applied to the template
13975      * @return {MasterTemplate} this
13976      */
13977      add : function(name, values){
13978         if(arguments.length == 1){
13979             values = arguments[0];
13980             name = 0;
13981         }
13982         var s = this.subs[name];
13983         s.buffer[s.buffer.length] = s.tpl.apply(values);
13984         return this;
13985     },
13986
13987     /**
13988      * Applies all the passed values to a child template.
13989      * @param {String/Number} name (optional) The name or index of the child template
13990      * @param {Array} values The values to be applied to the template, this should be an array of objects.
13991      * @param {Boolean} reset (optional) True to reset the template first
13992      * @return {MasterTemplate} this
13993      */
13994     fill : function(name, values, reset){
13995         var a = arguments;
13996         if(a.length == 1 || (a.length == 2 && typeof a[1] == "boolean")){
13997             values = a[0];
13998             name = 0;
13999             reset = a[1];
14000         }
14001         if(reset){
14002             this.reset();
14003         }
14004         for(var i = 0, len = values.length; i < len; i++){
14005             this.add(name, values[i]);
14006         }
14007         return this;
14008     },
14009
14010     /**
14011      * Resets the template for reuse
14012      * @return {MasterTemplate} this
14013      */
14014      reset : function(){
14015         var s = this.subs;
14016         for(var i = 0; i < this.subCount; i++){
14017             s[i].buffer = [];
14018         }
14019         return this;
14020     },
14021
14022     applyTemplate : function(values){
14023         var s = this.subs;
14024         var replaceIndex = -1;
14025         this.html = this.originalHtml.replace(this.subTemplateRe, function(m, name){
14026             return s[++replaceIndex].buffer.join("");
14027         });
14028         return Roo.MasterTemplate.superclass.applyTemplate.call(this, values);
14029     },
14030
14031     apply : function(){
14032         return this.applyTemplate.apply(this, arguments);
14033     },
14034
14035     compile : function(){return this;}
14036 });
14037
14038 /**
14039  * Alias for fill().
14040  * @method
14041  */
14042 Roo.MasterTemplate.prototype.addAll = Roo.MasterTemplate.prototype.fill;
14043  /**
14044  * Creates a template from the passed element's value (display:none textarea, preferred) or innerHTML. e.g.
14045  * var tpl = Roo.MasterTemplate.from('element-id');
14046  * @param {String/HTMLElement} el
14047  * @param {Object} config
14048  * @static
14049  */
14050 Roo.MasterTemplate.from = function(el, config){
14051     el = Roo.getDom(el);
14052     return new Roo.MasterTemplate(el.value || el.innerHTML, config || '');
14053 };/*
14054  * Based on:
14055  * Ext JS Library 1.1.1
14056  * Copyright(c) 2006-2007, Ext JS, LLC.
14057  *
14058  * Originally Released Under LGPL - original licence link has changed is not relivant.
14059  *
14060  * Fork - LGPL
14061  * <script type="text/javascript">
14062  */
14063
14064  
14065 /**
14066  * @class Roo.util.CSS
14067  * Utility class for manipulating CSS rules
14068  * @singleton
14069  */
14070 Roo.util.CSS = function(){
14071         var rules = null;
14072         var doc = document;
14073
14074     var camelRe = /(-[a-z])/gi;
14075     var camelFn = function(m, a){ return a.charAt(1).toUpperCase(); };
14076
14077    return {
14078    /**
14079     * Very simple dynamic creation of stylesheets from a text blob of rules.  The text will wrapped in a style
14080     * tag and appended to the HEAD of the document.
14081     * @param {String|Object} cssText The text containing the css rules
14082     * @param {String} id An id to add to the stylesheet for later removal
14083     * @return {StyleSheet}
14084     */
14085     createStyleSheet : function(cssText, id){
14086         var ss;
14087         var head = doc.getElementsByTagName("head")[0];
14088         var nrules = doc.createElement("style");
14089         nrules.setAttribute("type", "text/css");
14090         if(id){
14091             nrules.setAttribute("id", id);
14092         }
14093         if (typeof(cssText) != 'string') {
14094             // support object maps..
14095             // not sure if this a good idea.. 
14096             // perhaps it should be merged with the general css handling
14097             // and handle js style props.
14098             var cssTextNew = [];
14099             for(var n in cssText) {
14100                 var citems = [];
14101                 for(var k in cssText[n]) {
14102                     citems.push( k + ' : ' +cssText[n][k] + ';' );
14103                 }
14104                 cssTextNew.push( n + ' { ' + citems.join(' ') + '} ');
14105                 
14106             }
14107             cssText = cssTextNew.join("\n");
14108             
14109         }
14110        
14111        
14112        if(Roo.isIE){
14113            head.appendChild(nrules);
14114            ss = nrules.styleSheet;
14115            ss.cssText = cssText;
14116        }else{
14117            try{
14118                 nrules.appendChild(doc.createTextNode(cssText));
14119            }catch(e){
14120                nrules.cssText = cssText; 
14121            }
14122            head.appendChild(nrules);
14123            ss = nrules.styleSheet ? nrules.styleSheet : (nrules.sheet || doc.styleSheets[doc.styleSheets.length-1]);
14124        }
14125        this.cacheStyleSheet(ss);
14126        return ss;
14127    },
14128
14129    /**
14130     * Removes a style or link tag by id
14131     * @param {String} id The id of the tag
14132     */
14133    removeStyleSheet : function(id){
14134        var existing = doc.getElementById(id);
14135        if(existing){
14136            existing.parentNode.removeChild(existing);
14137        }
14138    },
14139
14140    /**
14141     * Dynamically swaps an existing stylesheet reference for a new one
14142     * @param {String} id The id of an existing link tag to remove
14143     * @param {String} url The href of the new stylesheet to include
14144     */
14145    swapStyleSheet : function(id, url){
14146        this.removeStyleSheet(id);
14147        var ss = doc.createElement("link");
14148        ss.setAttribute("rel", "stylesheet");
14149        ss.setAttribute("type", "text/css");
14150        ss.setAttribute("id", id);
14151        ss.setAttribute("href", url);
14152        doc.getElementsByTagName("head")[0].appendChild(ss);
14153    },
14154    
14155    /**
14156     * Refresh the rule cache if you have dynamically added stylesheets
14157     * @return {Object} An object (hash) of rules indexed by selector
14158     */
14159    refreshCache : function(){
14160        return this.getRules(true);
14161    },
14162
14163    // private
14164    cacheStyleSheet : function(stylesheet){
14165        if(!rules){
14166            rules = {};
14167        }
14168        try{// try catch for cross domain access issue
14169            var ssRules = stylesheet.cssRules || stylesheet.rules;
14170            for(var j = ssRules.length-1; j >= 0; --j){
14171                rules[ssRules[j].selectorText] = ssRules[j];
14172            }
14173        }catch(e){}
14174    },
14175    
14176    /**
14177     * Gets all css rules for the document
14178     * @param {Boolean} refreshCache true to refresh the internal cache
14179     * @return {Object} An object (hash) of rules indexed by selector
14180     */
14181    getRules : function(refreshCache){
14182                 if(rules == null || refreshCache){
14183                         rules = {};
14184                         var ds = doc.styleSheets;
14185                         for(var i =0, len = ds.length; i < len; i++){
14186                             try{
14187                         this.cacheStyleSheet(ds[i]);
14188                     }catch(e){} 
14189                 }
14190                 }
14191                 return rules;
14192         },
14193         
14194         /**
14195     * Gets an an individual CSS rule by selector(s)
14196     * @param {String/Array} selector The CSS selector or an array of selectors to try. The first selector that is found is returned.
14197     * @param {Boolean} refreshCache true to refresh the internal cache if you have recently updated any rules or added styles dynamically
14198     * @return {CSSRule} The CSS rule or null if one is not found
14199     */
14200    getRule : function(selector, refreshCache){
14201                 var rs = this.getRules(refreshCache);
14202                 if(!(selector instanceof Array)){
14203                     return rs[selector];
14204                 }
14205                 for(var i = 0; i < selector.length; i++){
14206                         if(rs[selector[i]]){
14207                                 return rs[selector[i]];
14208                         }
14209                 }
14210                 return null;
14211         },
14212         
14213         
14214         /**
14215     * Updates a rule property
14216     * @param {String/Array} selector If it's an array it tries each selector until it finds one. Stops immediately once one is found.
14217     * @param {String} property The css property
14218     * @param {String} value The new value for the property
14219     * @return {Boolean} true If a rule was found and updated
14220     */
14221    updateRule : function(selector, property, value){
14222                 if(!(selector instanceof Array)){
14223                         var rule = this.getRule(selector);
14224                         if(rule){
14225                                 rule.style[property.replace(camelRe, camelFn)] = value;
14226                                 return true;
14227                         }
14228                 }else{
14229                         for(var i = 0; i < selector.length; i++){
14230                                 if(this.updateRule(selector[i], property, value)){
14231                                         return true;
14232                                 }
14233                         }
14234                 }
14235                 return false;
14236         }
14237    };   
14238 }();/*
14239  * Based on:
14240  * Ext JS Library 1.1.1
14241  * Copyright(c) 2006-2007, Ext JS, LLC.
14242  *
14243  * Originally Released Under LGPL - original licence link has changed is not relivant.
14244  *
14245  * Fork - LGPL
14246  * <script type="text/javascript">
14247  */
14248
14249  
14250
14251 /**
14252  * @class Roo.util.ClickRepeater
14253  * @extends Roo.util.Observable
14254  * 
14255  * A wrapper class which can be applied to any element. Fires a "click" event while the
14256  * mouse is pressed. The interval between firings may be specified in the config but
14257  * defaults to 10 milliseconds.
14258  * 
14259  * Optionally, a CSS class may be applied to the element during the time it is pressed.
14260  * 
14261  * @cfg {String/HTMLElement/Element} el The element to act as a button.
14262  * @cfg {Number} delay The initial delay before the repeating event begins firing.
14263  * Similar to an autorepeat key delay.
14264  * @cfg {Number} interval The interval between firings of the "click" event. Default 10 ms.
14265  * @cfg {String} pressClass A CSS class name to be applied to the element while pressed.
14266  * @cfg {Boolean} accelerate True if autorepeating should start slowly and accelerate.
14267  *           "interval" and "delay" are ignored. "immediate" is honored.
14268  * @cfg {Boolean} preventDefault True to prevent the default click event
14269  * @cfg {Boolean} stopDefault True to stop the default click event
14270  * 
14271  * @history
14272  *     2007-02-02 jvs Original code contributed by Nige "Animal" White
14273  *     2007-02-02 jvs Renamed to ClickRepeater
14274  *   2007-02-03 jvs Modifications for FF Mac and Safari 
14275  *
14276  *  @constructor
14277  * @param {String/HTMLElement/Element} el The element to listen on
14278  * @param {Object} config
14279  **/
14280 Roo.util.ClickRepeater = function(el, config)
14281 {
14282     this.el = Roo.get(el);
14283     this.el.unselectable();
14284
14285     Roo.apply(this, config);
14286
14287     this.addEvents({
14288     /**
14289      * @event mousedown
14290      * Fires when the mouse button is depressed.
14291      * @param {Roo.util.ClickRepeater} this
14292      */
14293         "mousedown" : true,
14294     /**
14295      * @event click
14296      * Fires on a specified interval during the time the element is pressed.
14297      * @param {Roo.util.ClickRepeater} this
14298      */
14299         "click" : true,
14300     /**
14301      * @event mouseup
14302      * Fires when the mouse key is released.
14303      * @param {Roo.util.ClickRepeater} this
14304      */
14305         "mouseup" : true
14306     });
14307
14308     this.el.on("mousedown", this.handleMouseDown, this);
14309     if(this.preventDefault || this.stopDefault){
14310         this.el.on("click", function(e){
14311             if(this.preventDefault){
14312                 e.preventDefault();
14313             }
14314             if(this.stopDefault){
14315                 e.stopEvent();
14316             }
14317         }, this);
14318     }
14319
14320     // allow inline handler
14321     if(this.handler){
14322         this.on("click", this.handler,  this.scope || this);
14323     }
14324
14325     Roo.util.ClickRepeater.superclass.constructor.call(this);
14326 };
14327
14328 Roo.extend(Roo.util.ClickRepeater, Roo.util.Observable, {
14329     interval : 20,
14330     delay: 250,
14331     preventDefault : true,
14332     stopDefault : false,
14333     timer : 0,
14334
14335     // private
14336     handleMouseDown : function(){
14337         clearTimeout(this.timer);
14338         this.el.blur();
14339         if(this.pressClass){
14340             this.el.addClass(this.pressClass);
14341         }
14342         this.mousedownTime = new Date();
14343
14344         Roo.get(document).on("mouseup", this.handleMouseUp, this);
14345         this.el.on("mouseout", this.handleMouseOut, this);
14346
14347         this.fireEvent("mousedown", this);
14348         this.fireEvent("click", this);
14349         
14350         this.timer = this.click.defer(this.delay || this.interval, this);
14351     },
14352
14353     // private
14354     click : function(){
14355         this.fireEvent("click", this);
14356         this.timer = this.click.defer(this.getInterval(), this);
14357     },
14358
14359     // private
14360     getInterval: function(){
14361         if(!this.accelerate){
14362             return this.interval;
14363         }
14364         var pressTime = this.mousedownTime.getElapsed();
14365         if(pressTime < 500){
14366             return 400;
14367         }else if(pressTime < 1700){
14368             return 320;
14369         }else if(pressTime < 2600){
14370             return 250;
14371         }else if(pressTime < 3500){
14372             return 180;
14373         }else if(pressTime < 4400){
14374             return 140;
14375         }else if(pressTime < 5300){
14376             return 80;
14377         }else if(pressTime < 6200){
14378             return 50;
14379         }else{
14380             return 10;
14381         }
14382     },
14383
14384     // private
14385     handleMouseOut : function(){
14386         clearTimeout(this.timer);
14387         if(this.pressClass){
14388             this.el.removeClass(this.pressClass);
14389         }
14390         this.el.on("mouseover", this.handleMouseReturn, this);
14391     },
14392
14393     // private
14394     handleMouseReturn : function(){
14395         this.el.un("mouseover", this.handleMouseReturn);
14396         if(this.pressClass){
14397             this.el.addClass(this.pressClass);
14398         }
14399         this.click();
14400     },
14401
14402     // private
14403     handleMouseUp : function(){
14404         clearTimeout(this.timer);
14405         this.el.un("mouseover", this.handleMouseReturn);
14406         this.el.un("mouseout", this.handleMouseOut);
14407         Roo.get(document).un("mouseup", this.handleMouseUp);
14408         this.el.removeClass(this.pressClass);
14409         this.fireEvent("mouseup", this);
14410     }
14411 });/*
14412  * Based on:
14413  * Ext JS Library 1.1.1
14414  * Copyright(c) 2006-2007, Ext JS, LLC.
14415  *
14416  * Originally Released Under LGPL - original licence link has changed is not relivant.
14417  *
14418  * Fork - LGPL
14419  * <script type="text/javascript">
14420  */
14421
14422  
14423 /**
14424  * @class Roo.KeyNav
14425  * <p>Provides a convenient wrapper for normalized keyboard navigation.  KeyNav allows you to bind
14426  * navigation keys to function calls that will get called when the keys are pressed, providing an easy
14427  * way to implement custom navigation schemes for any UI component.</p>
14428  * <p>The following are all of the possible keys that can be implemented: enter, left, right, up, down, tab, esc,
14429  * pageUp, pageDown, del, home, end.  Usage:</p>
14430  <pre><code>
14431 var nav = new Roo.KeyNav("my-element", {
14432     "left" : function(e){
14433         this.moveLeft(e.ctrlKey);
14434     },
14435     "right" : function(e){
14436         this.moveRight(e.ctrlKey);
14437     },
14438     "enter" : function(e){
14439         this.save();
14440     },
14441     scope : this
14442 });
14443 </code></pre>
14444  * @constructor
14445  * @param {String/HTMLElement/Roo.Element} el The element to bind to
14446  * @param {Object} config The config
14447  */
14448 Roo.KeyNav = function(el, config){
14449     this.el = Roo.get(el);
14450     Roo.apply(this, config);
14451     if(!this.disabled){
14452         this.disabled = true;
14453         this.enable();
14454     }
14455 };
14456
14457 Roo.KeyNav.prototype = {
14458     /**
14459      * @cfg {Boolean} disabled
14460      * True to disable this KeyNav instance (defaults to false)
14461      */
14462     disabled : false,
14463     /**
14464      * @cfg {String} defaultEventAction
14465      * The method to call on the {@link Roo.EventObject} after this KeyNav intercepts a key.  Valid values are
14466      * {@link Roo.EventObject#stopEvent}, {@link Roo.EventObject#preventDefault} and
14467      * {@link Roo.EventObject#stopPropagation} (defaults to 'stopEvent')
14468      */
14469     defaultEventAction: "stopEvent",
14470     /**
14471      * @cfg {Boolean} forceKeyDown
14472      * Handle the keydown event instead of keypress (defaults to false).  KeyNav automatically does this for IE since
14473      * IE does not propagate special keys on keypress, but setting this to true will force other browsers to also
14474      * handle keydown instead of keypress.
14475      */
14476     forceKeyDown : false,
14477
14478     // private
14479     prepareEvent : function(e){
14480         var k = e.getKey();
14481         var h = this.keyToHandler[k];
14482         //if(h && this[h]){
14483         //    e.stopPropagation();
14484         //}
14485         if(Roo.isSafari && h && k >= 37 && k <= 40){
14486             e.stopEvent();
14487         }
14488     },
14489
14490     // private
14491     relay : function(e){
14492         var k = e.getKey();
14493         var h = this.keyToHandler[k];
14494         if(h && this[h]){
14495             if(this.doRelay(e, this[h], h) !== true){
14496                 e[this.defaultEventAction]();
14497             }
14498         }
14499     },
14500
14501     // private
14502     doRelay : function(e, h, hname){
14503         return h.call(this.scope || this, e);
14504     },
14505
14506     // possible handlers
14507     enter : false,
14508     left : false,
14509     right : false,
14510     up : false,
14511     down : false,
14512     tab : false,
14513     esc : false,
14514     pageUp : false,
14515     pageDown : false,
14516     del : false,
14517     home : false,
14518     end : false,
14519
14520     // quick lookup hash
14521     keyToHandler : {
14522         37 : "left",
14523         39 : "right",
14524         38 : "up",
14525         40 : "down",
14526         33 : "pageUp",
14527         34 : "pageDown",
14528         46 : "del",
14529         36 : "home",
14530         35 : "end",
14531         13 : "enter",
14532         27 : "esc",
14533         9  : "tab"
14534     },
14535
14536         /**
14537          * Enable this KeyNav
14538          */
14539         enable: function(){
14540                 if(this.disabled){
14541             // ie won't do special keys on keypress, no one else will repeat keys with keydown
14542             // the EventObject will normalize Safari automatically
14543             if(this.forceKeyDown || Roo.isIE || Roo.isAir){
14544                 this.el.on("keydown", this.relay,  this);
14545             }else{
14546                 this.el.on("keydown", this.prepareEvent,  this);
14547                 this.el.on("keypress", this.relay,  this);
14548             }
14549                     this.disabled = false;
14550                 }
14551         },
14552
14553         /**
14554          * Disable this KeyNav
14555          */
14556         disable: function(){
14557                 if(!this.disabled){
14558                     if(this.forceKeyDown || Roo.isIE || Roo.isAir){
14559                 this.el.un("keydown", this.relay);
14560             }else{
14561                 this.el.un("keydown", this.prepareEvent);
14562                 this.el.un("keypress", this.relay);
14563             }
14564                     this.disabled = true;
14565                 }
14566         }
14567 };/*
14568  * Based on:
14569  * Ext JS Library 1.1.1
14570  * Copyright(c) 2006-2007, Ext JS, LLC.
14571  *
14572  * Originally Released Under LGPL - original licence link has changed is not relivant.
14573  *
14574  * Fork - LGPL
14575  * <script type="text/javascript">
14576  */
14577
14578  
14579 /**
14580  * @class Roo.KeyMap
14581  * Handles mapping keys to actions for an element. One key map can be used for multiple actions.
14582  * The constructor accepts the same config object as defined by {@link #addBinding}.
14583  * If you bind a callback function to a KeyMap, anytime the KeyMap handles an expected key
14584  * combination it will call the function with this signature (if the match is a multi-key
14585  * combination the callback will still be called only once): (String key, Roo.EventObject e)
14586  * A KeyMap can also handle a string representation of keys.<br />
14587  * Usage:
14588  <pre><code>
14589 // map one key by key code
14590 var map = new Roo.KeyMap("my-element", {
14591     key: 13, // or Roo.EventObject.ENTER
14592     fn: myHandler,
14593     scope: myObject
14594 });
14595
14596 // map multiple keys to one action by string
14597 var map = new Roo.KeyMap("my-element", {
14598     key: "a\r\n\t",
14599     fn: myHandler,
14600     scope: myObject
14601 });
14602
14603 // map multiple keys to multiple actions by strings and array of codes
14604 var map = new Roo.KeyMap("my-element", [
14605     {
14606         key: [10,13],
14607         fn: function(){ alert("Return was pressed"); }
14608     }, {
14609         key: "abc",
14610         fn: function(){ alert('a, b or c was pressed'); }
14611     }, {
14612         key: "\t",
14613         ctrl:true,
14614         shift:true,
14615         fn: function(){ alert('Control + shift + tab was pressed.'); }
14616     }
14617 ]);
14618 </code></pre>
14619  * <b>Note: A KeyMap starts enabled</b>
14620  * @constructor
14621  * @param {String/HTMLElement/Roo.Element} el The element to bind to
14622  * @param {Object} config The config (see {@link #addBinding})
14623  * @param {String} eventName (optional) The event to bind to (defaults to "keydown")
14624  */
14625 Roo.KeyMap = function(el, config, eventName){
14626     this.el  = Roo.get(el);
14627     this.eventName = eventName || "keydown";
14628     this.bindings = [];
14629     if(config){
14630         this.addBinding(config);
14631     }
14632     this.enable();
14633 };
14634
14635 Roo.KeyMap.prototype = {
14636     /**
14637      * True to stop the event from bubbling and prevent the default browser action if the
14638      * key was handled by the KeyMap (defaults to false)
14639      * @type Boolean
14640      */
14641     stopEvent : false,
14642
14643     /**
14644      * Add a new binding to this KeyMap. The following config object properties are supported:
14645      * <pre>
14646 Property    Type             Description
14647 ----------  ---------------  ----------------------------------------------------------------------
14648 key         String/Array     A single keycode or an array of keycodes to handle
14649 shift       Boolean          True to handle key only when shift is pressed (defaults to false)
14650 ctrl        Boolean          True to handle key only when ctrl is pressed (defaults to false)
14651 alt         Boolean          True to handle key only when alt is pressed (defaults to false)
14652 fn          Function         The function to call when KeyMap finds the expected key combination
14653 scope       Object           The scope of the callback function
14654 </pre>
14655      *
14656      * Usage:
14657      * <pre><code>
14658 // Create a KeyMap
14659 var map = new Roo.KeyMap(document, {
14660     key: Roo.EventObject.ENTER,
14661     fn: handleKey,
14662     scope: this
14663 });
14664
14665 //Add a new binding to the existing KeyMap later
14666 map.addBinding({
14667     key: 'abc',
14668     shift: true,
14669     fn: handleKey,
14670     scope: this
14671 });
14672 </code></pre>
14673      * @param {Object/Array} config A single KeyMap config or an array of configs
14674      */
14675         addBinding : function(config){
14676         if(config instanceof Array){
14677             for(var i = 0, len = config.length; i < len; i++){
14678                 this.addBinding(config[i]);
14679             }
14680             return;
14681         }
14682         var keyCode = config.key,
14683             shift = config.shift, 
14684             ctrl = config.ctrl, 
14685             alt = config.alt,
14686             fn = config.fn,
14687             scope = config.scope;
14688         if(typeof keyCode == "string"){
14689             var ks = [];
14690             var keyString = keyCode.toUpperCase();
14691             for(var j = 0, len = keyString.length; j < len; j++){
14692                 ks.push(keyString.charCodeAt(j));
14693             }
14694             keyCode = ks;
14695         }
14696         var keyArray = keyCode instanceof Array;
14697         var handler = function(e){
14698             if((!shift || e.shiftKey) && (!ctrl || e.ctrlKey) &&  (!alt || e.altKey)){
14699                 var k = e.getKey();
14700                 if(keyArray){
14701                     for(var i = 0, len = keyCode.length; i < len; i++){
14702                         if(keyCode[i] == k){
14703                           if(this.stopEvent){
14704                               e.stopEvent();
14705                           }
14706                           fn.call(scope || window, k, e);
14707                           return;
14708                         }
14709                     }
14710                 }else{
14711                     if(k == keyCode){
14712                         if(this.stopEvent){
14713                            e.stopEvent();
14714                         }
14715                         fn.call(scope || window, k, e);
14716                     }
14717                 }
14718             }
14719         };
14720         this.bindings.push(handler);  
14721         },
14722
14723     /**
14724      * Shorthand for adding a single key listener
14725      * @param {Number/Array/Object} key Either the numeric key code, array of key codes or an object with the
14726      * following options:
14727      * {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
14728      * @param {Function} fn The function to call
14729      * @param {Object} scope (optional) The scope of the function
14730      */
14731     on : function(key, fn, scope){
14732         var keyCode, shift, ctrl, alt;
14733         if(typeof key == "object" && !(key instanceof Array)){
14734             keyCode = key.key;
14735             shift = key.shift;
14736             ctrl = key.ctrl;
14737             alt = key.alt;
14738         }else{
14739             keyCode = key;
14740         }
14741         this.addBinding({
14742             key: keyCode,
14743             shift: shift,
14744             ctrl: ctrl,
14745             alt: alt,
14746             fn: fn,
14747             scope: scope
14748         })
14749     },
14750
14751     // private
14752     handleKeyDown : function(e){
14753             if(this.enabled){ //just in case
14754             var b = this.bindings;
14755             for(var i = 0, len = b.length; i < len; i++){
14756                 b[i].call(this, e);
14757             }
14758             }
14759         },
14760         
14761         /**
14762          * Returns true if this KeyMap is enabled
14763          * @return {Boolean} 
14764          */
14765         isEnabled : function(){
14766             return this.enabled;  
14767         },
14768         
14769         /**
14770          * Enables this KeyMap
14771          */
14772         enable: function(){
14773                 if(!this.enabled){
14774                     this.el.on(this.eventName, this.handleKeyDown, this);
14775                     this.enabled = true;
14776                 }
14777         },
14778
14779         /**
14780          * Disable this KeyMap
14781          */
14782         disable: function(){
14783                 if(this.enabled){
14784                     this.el.removeListener(this.eventName, this.handleKeyDown, this);
14785                     this.enabled = false;
14786                 }
14787         }
14788 };/*
14789  * Based on:
14790  * Ext JS Library 1.1.1
14791  * Copyright(c) 2006-2007, Ext JS, LLC.
14792  *
14793  * Originally Released Under LGPL - original licence link has changed is not relivant.
14794  *
14795  * Fork - LGPL
14796  * <script type="text/javascript">
14797  */
14798
14799  
14800 /**
14801  * @class Roo.util.TextMetrics
14802  * Provides precise pixel measurements for blocks of text so that you can determine exactly how high and
14803  * wide, in pixels, a given block of text will be.
14804  * @singleton
14805  */
14806 Roo.util.TextMetrics = function(){
14807     var shared;
14808     return {
14809         /**
14810          * Measures the size of the specified text
14811          * @param {String/HTMLElement} el The element, dom node or id from which to copy existing CSS styles
14812          * that can affect the size of the rendered text
14813          * @param {String} text The text to measure
14814          * @param {Number} fixedWidth (optional) If the text will be multiline, you have to set a fixed width
14815          * in order to accurately measure the text height
14816          * @return {Object} An object containing the text's size {width: (width), height: (height)}
14817          */
14818         measure : function(el, text, fixedWidth){
14819             if(!shared){
14820                 shared = Roo.util.TextMetrics.Instance(el, fixedWidth);
14821             }
14822             shared.bind(el);
14823             shared.setFixedWidth(fixedWidth || 'auto');
14824             return shared.getSize(text);
14825         },
14826
14827         /**
14828          * Return a unique TextMetrics instance that can be bound directly to an element and reused.  This reduces
14829          * the overhead of multiple calls to initialize the style properties on each measurement.
14830          * @param {String/HTMLElement} el The element, dom node or id that the instance will be bound to
14831          * @param {Number} fixedWidth (optional) If the text will be multiline, you have to set a fixed width
14832          * in order to accurately measure the text height
14833          * @return {Roo.util.TextMetrics.Instance} instance The new instance
14834          */
14835         createInstance : function(el, fixedWidth){
14836             return Roo.util.TextMetrics.Instance(el, fixedWidth);
14837         }
14838     };
14839 }();
14840
14841  
14842
14843 Roo.util.TextMetrics.Instance = function(bindTo, fixedWidth){
14844     var ml = new Roo.Element(document.createElement('div'));
14845     document.body.appendChild(ml.dom);
14846     ml.position('absolute');
14847     ml.setLeftTop(-1000, -1000);
14848     ml.hide();
14849
14850     if(fixedWidth){
14851         ml.setWidth(fixedWidth);
14852     }
14853      
14854     var instance = {
14855         /**
14856          * Returns the size of the specified text based on the internal element's style and width properties
14857          * @memberOf Roo.util.TextMetrics.Instance#
14858          * @param {String} text The text to measure
14859          * @return {Object} An object containing the text's size {width: (width), height: (height)}
14860          */
14861         getSize : function(text){
14862             ml.update(text);
14863             var s = ml.getSize();
14864             ml.update('');
14865             return s;
14866         },
14867
14868         /**
14869          * Binds this TextMetrics instance to an element from which to copy existing CSS styles
14870          * that can affect the size of the rendered text
14871          * @memberOf Roo.util.TextMetrics.Instance#
14872          * @param {String/HTMLElement} el The element, dom node or id
14873          */
14874         bind : function(el){
14875             ml.setStyle(
14876                 Roo.fly(el).getStyles('font-size','font-style', 'font-weight', 'font-family','line-height')
14877             );
14878         },
14879
14880         /**
14881          * Sets a fixed width on the internal measurement element.  If the text will be multiline, you have
14882          * to set a fixed width in order to accurately measure the text height.
14883          * @memberOf Roo.util.TextMetrics.Instance#
14884          * @param {Number} width The width to set on the element
14885          */
14886         setFixedWidth : function(width){
14887             ml.setWidth(width);
14888         },
14889
14890         /**
14891          * Returns the measured width of the specified text
14892          * @memberOf Roo.util.TextMetrics.Instance#
14893          * @param {String} text The text to measure
14894          * @return {Number} width The width in pixels
14895          */
14896         getWidth : function(text){
14897             ml.dom.style.width = 'auto';
14898             return this.getSize(text).width;
14899         },
14900
14901         /**
14902          * Returns the measured height of the specified text.  For multiline text, be sure to call
14903          * {@link #setFixedWidth} if necessary.
14904          * @memberOf Roo.util.TextMetrics.Instance#
14905          * @param {String} text The text to measure
14906          * @return {Number} height The height in pixels
14907          */
14908         getHeight : function(text){
14909             return this.getSize(text).height;
14910         }
14911     };
14912
14913     instance.bind(bindTo);
14914
14915     return instance;
14916 };
14917
14918 // backwards compat
14919 Roo.Element.measureText = Roo.util.TextMetrics.measure;/*
14920  * Based on:
14921  * Ext JS Library 1.1.1
14922  * Copyright(c) 2006-2007, Ext JS, LLC.
14923  *
14924  * Originally Released Under LGPL - original licence link has changed is not relivant.
14925  *
14926  * Fork - LGPL
14927  * <script type="text/javascript">
14928  */
14929
14930 /**
14931  * @class Roo.state.Provider
14932  * Abstract base class for state provider implementations. This class provides methods
14933  * for encoding and decoding <b>typed</b> variables including dates and defines the 
14934  * Provider interface.
14935  */
14936 Roo.state.Provider = function(){
14937     /**
14938      * @event statechange
14939      * Fires when a state change occurs.
14940      * @param {Provider} this This state provider
14941      * @param {String} key The state key which was changed
14942      * @param {String} value The encoded value for the state
14943      */
14944     this.addEvents({
14945         "statechange": true
14946     });
14947     this.state = {};
14948     Roo.state.Provider.superclass.constructor.call(this);
14949 };
14950 Roo.extend(Roo.state.Provider, Roo.util.Observable, {
14951     /**
14952      * Returns the current value for a key
14953      * @param {String} name The key name
14954      * @param {Mixed} defaultValue A default value to return if the key's value is not found
14955      * @return {Mixed} The state data
14956      */
14957     get : function(name, defaultValue){
14958         return typeof this.state[name] == "undefined" ?
14959             defaultValue : this.state[name];
14960     },
14961     
14962     /**
14963      * Clears a value from the state
14964      * @param {String} name The key name
14965      */
14966     clear : function(name){
14967         delete this.state[name];
14968         this.fireEvent("statechange", this, name, null);
14969     },
14970     
14971     /**
14972      * Sets the value for a key
14973      * @param {String} name The key name
14974      * @param {Mixed} value The value to set
14975      */
14976     set : function(name, value){
14977         this.state[name] = value;
14978         this.fireEvent("statechange", this, name, value);
14979     },
14980     
14981     /**
14982      * Decodes a string previously encoded with {@link #encodeValue}.
14983      * @param {String} value The value to decode
14984      * @return {Mixed} The decoded value
14985      */
14986     decodeValue : function(cookie){
14987         var re = /^(a|n|d|b|s|o)\:(.*)$/;
14988         var matches = re.exec(unescape(cookie));
14989         if(!matches || !matches[1]) {
14990             return; // non state cookie
14991         }
14992         var type = matches[1];
14993         var v = matches[2];
14994         switch(type){
14995             case "n":
14996                 return parseFloat(v);
14997             case "d":
14998                 return new Date(Date.parse(v));
14999             case "b":
15000                 return (v == "1");
15001             case "a":
15002                 var all = [];
15003                 var values = v.split("^");
15004                 for(var i = 0, len = values.length; i < len; i++){
15005                     all.push(this.decodeValue(values[i]));
15006                 }
15007                 return all;
15008            case "o":
15009                 var all = {};
15010                 var values = v.split("^");
15011                 for(var i = 0, len = values.length; i < len; i++){
15012                     var kv = values[i].split("=");
15013                     all[kv[0]] = this.decodeValue(kv[1]);
15014                 }
15015                 return all;
15016            default:
15017                 return v;
15018         }
15019     },
15020     
15021     /**
15022      * Encodes a value including type information.  Decode with {@link #decodeValue}.
15023      * @param {Mixed} value The value to encode
15024      * @return {String} The encoded value
15025      */
15026     encodeValue : function(v){
15027         var enc;
15028         if(typeof v == "number"){
15029             enc = "n:" + v;
15030         }else if(typeof v == "boolean"){
15031             enc = "b:" + (v ? "1" : "0");
15032         }else if(v instanceof Date){
15033             enc = "d:" + v.toGMTString();
15034         }else if(v instanceof Array){
15035             var flat = "";
15036             for(var i = 0, len = v.length; i < len; i++){
15037                 flat += this.encodeValue(v[i]);
15038                 if(i != len-1) {
15039                     flat += "^";
15040                 }
15041             }
15042             enc = "a:" + flat;
15043         }else if(typeof v == "object"){
15044             var flat = "";
15045             for(var key in v){
15046                 if(typeof v[key] != "function"){
15047                     flat += key + "=" + this.encodeValue(v[key]) + "^";
15048                 }
15049             }
15050             enc = "o:" + flat.substring(0, flat.length-1);
15051         }else{
15052             enc = "s:" + v;
15053         }
15054         return escape(enc);        
15055     }
15056 });
15057
15058 /*
15059  * Based on:
15060  * Ext JS Library 1.1.1
15061  * Copyright(c) 2006-2007, Ext JS, LLC.
15062  *
15063  * Originally Released Under LGPL - original licence link has changed is not relivant.
15064  *
15065  * Fork - LGPL
15066  * <script type="text/javascript">
15067  */
15068 /**
15069  * @class Roo.state.Manager
15070  * This is the global state manager. By default all components that are "state aware" check this class
15071  * for state information if you don't pass them a custom state provider. In order for this class
15072  * to be useful, it must be initialized with a provider when your application initializes.
15073  <pre><code>
15074 // in your initialization function
15075 init : function(){
15076    Roo.state.Manager.setProvider(new Roo.state.CookieProvider());
15077    ...
15078    // supposed you have a {@link Roo.BorderLayout}
15079    var layout = new Roo.BorderLayout(...);
15080    layout.restoreState();
15081    // or a {Roo.BasicDialog}
15082    var dialog = new Roo.BasicDialog(...);
15083    dialog.restoreState();
15084  </code></pre>
15085  * @singleton
15086  */
15087 Roo.state.Manager = function(){
15088     var provider = new Roo.state.Provider();
15089     
15090     return {
15091         /**
15092          * Configures the default state provider for your application
15093          * @param {Provider} stateProvider The state provider to set
15094          */
15095         setProvider : function(stateProvider){
15096             provider = stateProvider;
15097         },
15098         
15099         /**
15100          * Returns the current value for a key
15101          * @param {String} name The key name
15102          * @param {Mixed} defaultValue The default value to return if the key lookup does not match
15103          * @return {Mixed} The state data
15104          */
15105         get : function(key, defaultValue){
15106             return provider.get(key, defaultValue);
15107         },
15108         
15109         /**
15110          * Sets the value for a key
15111          * @param {String} name The key name
15112          * @param {Mixed} value The state data
15113          */
15114          set : function(key, value){
15115             provider.set(key, value);
15116         },
15117         
15118         /**
15119          * Clears a value from the state
15120          * @param {String} name The key name
15121          */
15122         clear : function(key){
15123             provider.clear(key);
15124         },
15125         
15126         /**
15127          * Gets the currently configured state provider
15128          * @return {Provider} The state provider
15129          */
15130         getProvider : function(){
15131             return provider;
15132         }
15133     };
15134 }();
15135 /*
15136  * Based on:
15137  * Ext JS Library 1.1.1
15138  * Copyright(c) 2006-2007, Ext JS, LLC.
15139  *
15140  * Originally Released Under LGPL - original licence link has changed is not relivant.
15141  *
15142  * Fork - LGPL
15143  * <script type="text/javascript">
15144  */
15145 /**
15146  * @class Roo.state.CookieProvider
15147  * @extends Roo.state.Provider
15148  * The default Provider implementation which saves state via cookies.
15149  * <br />Usage:
15150  <pre><code>
15151    var cp = new Roo.state.CookieProvider({
15152        path: "/cgi-bin/",
15153        expires: new Date(new Date().getTime()+(1000*60*60*24*30)); //30 days
15154        domain: "roojs.com"
15155    })
15156    Roo.state.Manager.setProvider(cp);
15157  </code></pre>
15158  * @cfg {String} path The path for which the cookie is active (defaults to root '/' which makes it active for all pages in the site)
15159  * @cfg {Date} expires The cookie expiration date (defaults to 7 days from now)
15160  * @cfg {String} domain The domain to save the cookie for.  Note that you cannot specify a different domain than
15161  * your page is on, but you can specify a sub-domain, or simply the domain itself like 'roojs.com' to include
15162  * all sub-domains if you need to access cookies across different sub-domains (defaults to null which uses the same
15163  * domain the page is running on including the 'www' like 'www.roojs.com')
15164  * @cfg {Boolean} secure True if the site is using SSL (defaults to false)
15165  * @constructor
15166  * Create a new CookieProvider
15167  * @param {Object} config The configuration object
15168  */
15169 Roo.state.CookieProvider = function(config){
15170     Roo.state.CookieProvider.superclass.constructor.call(this);
15171     this.path = "/";
15172     this.expires = new Date(new Date().getTime()+(1000*60*60*24*7)); //7 days
15173     this.domain = null;
15174     this.secure = false;
15175     Roo.apply(this, config);
15176     this.state = this.readCookies();
15177 };
15178
15179 Roo.extend(Roo.state.CookieProvider, Roo.state.Provider, {
15180     // private
15181     set : function(name, value){
15182         if(typeof value == "undefined" || value === null){
15183             this.clear(name);
15184             return;
15185         }
15186         this.setCookie(name, value);
15187         Roo.state.CookieProvider.superclass.set.call(this, name, value);
15188     },
15189
15190     // private
15191     clear : function(name){
15192         this.clearCookie(name);
15193         Roo.state.CookieProvider.superclass.clear.call(this, name);
15194     },
15195
15196     // private
15197     readCookies : function(){
15198         var cookies = {};
15199         var c = document.cookie + ";";
15200         var re = /\s?(.*?)=(.*?);/g;
15201         var matches;
15202         while((matches = re.exec(c)) != null){
15203             var name = matches[1];
15204             var value = matches[2];
15205             if(name && name.substring(0,3) == "ys-"){
15206                 cookies[name.substr(3)] = this.decodeValue(value);
15207             }
15208         }
15209         return cookies;
15210     },
15211
15212     // private
15213     setCookie : function(name, value){
15214         document.cookie = "ys-"+ name + "=" + this.encodeValue(value) +
15215            ((this.expires == null) ? "" : ("; expires=" + this.expires.toGMTString())) +
15216            ((this.path == null) ? "" : ("; path=" + this.path)) +
15217            ((this.domain == null) ? "" : ("; domain=" + this.domain)) +
15218            ((this.secure == true) ? "; secure" : "");
15219     },
15220
15221     // private
15222     clearCookie : function(name){
15223         document.cookie = "ys-" + name + "=null; expires=Thu, 01-Jan-70 00:00:01 GMT" +
15224            ((this.path == null) ? "" : ("; path=" + this.path)) +
15225            ((this.domain == null) ? "" : ("; domain=" + this.domain)) +
15226            ((this.secure == true) ? "; secure" : "");
15227     }
15228 });/*
15229  * Based on:
15230  * Ext JS Library 1.1.1
15231  * Copyright(c) 2006-2007, Ext JS, LLC.
15232  *
15233  * Originally Released Under LGPL - original licence link has changed is not relivant.
15234  *
15235  * Fork - LGPL
15236  * <script type="text/javascript">
15237  */
15238  
15239
15240 /**
15241  * @class Roo.ComponentMgr
15242  * Provides a common registry of all components on a page so that they can be easily accessed by component id (see {@link Roo.getCmp}).
15243  * @singleton
15244  */
15245 Roo.ComponentMgr = function(){
15246     var all = new Roo.util.MixedCollection();
15247
15248     return {
15249         /**
15250          * Registers a component.
15251          * @param {Roo.Component} c The component
15252          */
15253         register : function(c){
15254             all.add(c);
15255         },
15256
15257         /**
15258          * Unregisters a component.
15259          * @param {Roo.Component} c The component
15260          */
15261         unregister : function(c){
15262             all.remove(c);
15263         },
15264
15265         /**
15266          * Returns a component by id
15267          * @param {String} id The component id
15268          */
15269         get : function(id){
15270             return all.get(id);
15271         },
15272
15273         /**
15274          * Registers a function that will be called when a specified component is added to ComponentMgr
15275          * @param {String} id The component id
15276          * @param {Funtction} fn The callback function
15277          * @param {Object} scope The scope of the callback
15278          */
15279         onAvailable : function(id, fn, scope){
15280             all.on("add", function(index, o){
15281                 if(o.id == id){
15282                     fn.call(scope || o, o);
15283                     all.un("add", fn, scope);
15284                 }
15285             });
15286         }
15287     };
15288 }();/*
15289  * Based on:
15290  * Ext JS Library 1.1.1
15291  * Copyright(c) 2006-2007, Ext JS, LLC.
15292  *
15293  * Originally Released Under LGPL - original licence link has changed is not relivant.
15294  *
15295  * Fork - LGPL
15296  * <script type="text/javascript">
15297  */
15298  
15299 /**
15300  * @class Roo.Component
15301  * @extends Roo.util.Observable
15302  * Base class for all major Roo components.  All subclasses of Component can automatically participate in the standard
15303  * Roo component lifecycle of creation, rendering and destruction.  They also have automatic support for basic hide/show
15304  * and enable/disable behavior.  Component allows any subclass to be lazy-rendered into any {@link Roo.Container} and
15305  * to be automatically registered with the {@link Roo.ComponentMgr} so that it can be referenced at any time via {@link Roo.getCmp}.
15306  * All visual components (widgets) that require rendering into a layout should subclass Component.
15307  * @constructor
15308  * @param {Roo.Element/String/Object} config The configuration options.  If an element is passed, it is set as the internal
15309  * element and its id used as the component id.  If a string is passed, it is assumed to be the id of an existing element
15310  * and is used as the component id.  Otherwise, it is assumed to be a standard config object and is applied to the component.
15311  */
15312 Roo.Component = function(config){
15313     config = config || {};
15314     if(config.tagName || config.dom || typeof config == "string"){ // element object
15315         config = {el: config, id: config.id || config};
15316     }
15317     this.initialConfig = config;
15318
15319     Roo.apply(this, config);
15320     this.addEvents({
15321         /**
15322          * @event disable
15323          * Fires after the component is disabled.
15324              * @param {Roo.Component} this
15325              */
15326         disable : true,
15327         /**
15328          * @event enable
15329          * Fires after the component is enabled.
15330              * @param {Roo.Component} this
15331              */
15332         enable : true,
15333         /**
15334          * @event beforeshow
15335          * Fires before the component is shown.  Return false to stop the show.
15336              * @param {Roo.Component} this
15337              */
15338         beforeshow : true,
15339         /**
15340          * @event show
15341          * Fires after the component is shown.
15342              * @param {Roo.Component} this
15343              */
15344         show : true,
15345         /**
15346          * @event beforehide
15347          * Fires before the component is hidden. Return false to stop the hide.
15348              * @param {Roo.Component} this
15349              */
15350         beforehide : true,
15351         /**
15352          * @event hide
15353          * Fires after the component is hidden.
15354              * @param {Roo.Component} this
15355              */
15356         hide : true,
15357         /**
15358          * @event beforerender
15359          * Fires before the component is rendered. Return false to stop the render.
15360              * @param {Roo.Component} this
15361              */
15362         beforerender : true,
15363         /**
15364          * @event render
15365          * Fires after the component is rendered.
15366              * @param {Roo.Component} this
15367              */
15368         render : true,
15369         /**
15370          * @event beforedestroy
15371          * Fires before the component is destroyed. Return false to stop the destroy.
15372              * @param {Roo.Component} this
15373              */
15374         beforedestroy : true,
15375         /**
15376          * @event destroy
15377          * Fires after the component is destroyed.
15378              * @param {Roo.Component} this
15379              */
15380         destroy : true
15381     });
15382     if(!this.id){
15383         this.id = "roo-comp-" + (++Roo.Component.AUTO_ID);
15384     }
15385     Roo.ComponentMgr.register(this);
15386     Roo.Component.superclass.constructor.call(this);
15387     this.initComponent();
15388     if(this.renderTo){ // not supported by all components yet. use at your own risk!
15389         this.render(this.renderTo);
15390         delete this.renderTo;
15391     }
15392 };
15393
15394 /** @private */
15395 Roo.Component.AUTO_ID = 1000;
15396
15397 Roo.extend(Roo.Component, Roo.util.Observable, {
15398     /**
15399      * @scope Roo.Component.prototype
15400      * @type {Boolean}
15401      * true if this component is hidden. Read-only.
15402      */
15403     hidden : false,
15404     /**
15405      * @type {Boolean}
15406      * true if this component is disabled. Read-only.
15407      */
15408     disabled : false,
15409     /**
15410      * @type {Boolean}
15411      * true if this component has been rendered. Read-only.
15412      */
15413     rendered : false,
15414     
15415     /** @cfg {String} disableClass
15416      * CSS class added to the component when it is disabled (defaults to "x-item-disabled").
15417      */
15418     disabledClass : "x-item-disabled",
15419         /** @cfg {Boolean} allowDomMove
15420          * Whether the component can move the Dom node when rendering (defaults to true).
15421          */
15422     allowDomMove : true,
15423     /** @cfg {String} hideMode (display|visibility)
15424      * How this component should hidden. Supported values are
15425      * "visibility" (css visibility), "offsets" (negative offset position) and
15426      * "display" (css display) - defaults to "display".
15427      */
15428     hideMode: 'display',
15429
15430     /** @private */
15431     ctype : "Roo.Component",
15432
15433     /**
15434      * @cfg {String} actionMode 
15435      * which property holds the element that used for  hide() / show() / disable() / enable()
15436      * default is 'el' 
15437      */
15438     actionMode : "el",
15439
15440     /** @private */
15441     getActionEl : function(){
15442         return this[this.actionMode];
15443     },
15444
15445     initComponent : Roo.emptyFn,
15446     /**
15447      * If this is a lazy rendering component, render it to its container element.
15448      * @param {String/HTMLElement/Element} container (optional) The element this component should be rendered into. If it is being applied to existing markup, this should be left off.
15449      */
15450     render : function(container, position){
15451         
15452         if(this.rendered){
15453             return this;
15454         }
15455         
15456         if(this.fireEvent("beforerender", this) === false){
15457             return false;
15458         }
15459         
15460         if(!container && this.el){
15461             this.el = Roo.get(this.el);
15462             container = this.el.dom.parentNode;
15463             this.allowDomMove = false;
15464         }
15465         this.container = Roo.get(container);
15466         this.rendered = true;
15467         if(position !== undefined){
15468             if(typeof position == 'number'){
15469                 position = this.container.dom.childNodes[position];
15470             }else{
15471                 position = Roo.getDom(position);
15472             }
15473         }
15474         this.onRender(this.container, position || null);
15475         if(this.cls){
15476             this.el.addClass(this.cls);
15477             delete this.cls;
15478         }
15479         if(this.style){
15480             this.el.applyStyles(this.style);
15481             delete this.style;
15482         }
15483         this.fireEvent("render", this);
15484         this.afterRender(this.container);
15485         if(this.hidden){
15486             this.hide();
15487         }
15488         if(this.disabled){
15489             this.disable();
15490         }
15491
15492         return this;
15493         
15494     },
15495
15496     /** @private */
15497     // default function is not really useful
15498     onRender : function(ct, position){
15499         if(this.el){
15500             this.el = Roo.get(this.el);
15501             if(this.allowDomMove !== false){
15502                 ct.dom.insertBefore(this.el.dom, position);
15503             }
15504         }
15505     },
15506
15507     /** @private */
15508     getAutoCreate : function(){
15509         var cfg = typeof this.autoCreate == "object" ?
15510                       this.autoCreate : Roo.apply({}, this.defaultAutoCreate);
15511         if(this.id && !cfg.id){
15512             cfg.id = this.id;
15513         }
15514         return cfg;
15515     },
15516
15517     /** @private */
15518     afterRender : Roo.emptyFn,
15519
15520     /**
15521      * Destroys this component by purging any event listeners, removing the component's element from the DOM,
15522      * removing the component from its {@link Roo.Container} (if applicable) and unregistering it from {@link Roo.ComponentMgr}.
15523      */
15524     destroy : function(){
15525         if(this.fireEvent("beforedestroy", this) !== false){
15526             this.purgeListeners();
15527             this.beforeDestroy();
15528             if(this.rendered){
15529                 this.el.removeAllListeners();
15530                 this.el.remove();
15531                 if(this.actionMode == "container"){
15532                     this.container.remove();
15533                 }
15534             }
15535             this.onDestroy();
15536             Roo.ComponentMgr.unregister(this);
15537             this.fireEvent("destroy", this);
15538         }
15539     },
15540
15541         /** @private */
15542     beforeDestroy : function(){
15543
15544     },
15545
15546         /** @private */
15547         onDestroy : function(){
15548
15549     },
15550
15551     /**
15552      * Returns the underlying {@link Roo.Element}.
15553      * @return {Roo.Element} The element
15554      */
15555     getEl : function(){
15556         return this.el;
15557     },
15558
15559     /**
15560      * Returns the id of this component.
15561      * @return {String}
15562      */
15563     getId : function(){
15564         return this.id;
15565     },
15566
15567     /**
15568      * Try to focus this component.
15569      * @param {Boolean} selectText True to also select the text in this component (if applicable)
15570      * @return {Roo.Component} this
15571      */
15572     focus : function(selectText){
15573         if(this.rendered){
15574             this.el.focus();
15575             if(selectText === true){
15576                 this.el.dom.select();
15577             }
15578         }
15579         return this;
15580     },
15581
15582     /** @private */
15583     blur : function(){
15584         if(this.rendered){
15585             this.el.blur();
15586         }
15587         return this;
15588     },
15589
15590     /**
15591      * Disable this component.
15592      * @return {Roo.Component} this
15593      */
15594     disable : function(){
15595         if(this.rendered){
15596             this.onDisable();
15597         }
15598         this.disabled = true;
15599         this.fireEvent("disable", this);
15600         return this;
15601     },
15602
15603         // private
15604     onDisable : function(){
15605         this.getActionEl().addClass(this.disabledClass);
15606         this.el.dom.disabled = true;
15607     },
15608
15609     /**
15610      * Enable this component.
15611      * @return {Roo.Component} this
15612      */
15613     enable : function(){
15614         if(this.rendered){
15615             this.onEnable();
15616         }
15617         this.disabled = false;
15618         this.fireEvent("enable", this);
15619         return this;
15620     },
15621
15622         // private
15623     onEnable : function(){
15624         this.getActionEl().removeClass(this.disabledClass);
15625         this.el.dom.disabled = false;
15626     },
15627
15628     /**
15629      * Convenience function for setting disabled/enabled by boolean.
15630      * @param {Boolean} disabled
15631      */
15632     setDisabled : function(disabled){
15633         this[disabled ? "disable" : "enable"]();
15634     },
15635
15636     /**
15637      * Show this component.
15638      * @return {Roo.Component} this
15639      */
15640     show: function(){
15641         if(this.fireEvent("beforeshow", this) !== false){
15642             this.hidden = false;
15643             if(this.rendered){
15644                 this.onShow();
15645             }
15646             this.fireEvent("show", this);
15647         }
15648         return this;
15649     },
15650
15651     // private
15652     onShow : function(){
15653         var ae = this.getActionEl();
15654         if(this.hideMode == 'visibility'){
15655             ae.dom.style.visibility = "visible";
15656         }else if(this.hideMode == 'offsets'){
15657             ae.removeClass('x-hidden');
15658         }else{
15659             ae.dom.style.display = "";
15660         }
15661     },
15662
15663     /**
15664      * Hide this component.
15665      * @return {Roo.Component} this
15666      */
15667     hide: function(){
15668         if(this.fireEvent("beforehide", this) !== false){
15669             this.hidden = true;
15670             if(this.rendered){
15671                 this.onHide();
15672             }
15673             this.fireEvent("hide", this);
15674         }
15675         return this;
15676     },
15677
15678     // private
15679     onHide : function(){
15680         var ae = this.getActionEl();
15681         if(this.hideMode == 'visibility'){
15682             ae.dom.style.visibility = "hidden";
15683         }else if(this.hideMode == 'offsets'){
15684             ae.addClass('x-hidden');
15685         }else{
15686             ae.dom.style.display = "none";
15687         }
15688     },
15689
15690     /**
15691      * Convenience function to hide or show this component by boolean.
15692      * @param {Boolean} visible True to show, false to hide
15693      * @return {Roo.Component} this
15694      */
15695     setVisible: function(visible){
15696         if(visible) {
15697             this.show();
15698         }else{
15699             this.hide();
15700         }
15701         return this;
15702     },
15703
15704     /**
15705      * Returns true if this component is visible.
15706      */
15707     isVisible : function(){
15708         return this.getActionEl().isVisible();
15709     },
15710
15711     cloneConfig : function(overrides){
15712         overrides = overrides || {};
15713         var id = overrides.id || Roo.id();
15714         var cfg = Roo.applyIf(overrides, this.initialConfig);
15715         cfg.id = id; // prevent dup id
15716         return new this.constructor(cfg);
15717     }
15718 });/*
15719  * Based on:
15720  * Ext JS Library 1.1.1
15721  * Copyright(c) 2006-2007, Ext JS, LLC.
15722  *
15723  * Originally Released Under LGPL - original licence link has changed is not relivant.
15724  *
15725  * Fork - LGPL
15726  * <script type="text/javascript">
15727  */
15728
15729 /**
15730  * @class Roo.BoxComponent
15731  * @extends Roo.Component
15732  * Base class for any visual {@link Roo.Component} that uses a box container.  BoxComponent provides automatic box
15733  * model adjustments for sizing and positioning and will work correctly withnin the Component rendering model.  All
15734  * container classes should subclass BoxComponent so that they will work consistently when nested within other Ext
15735  * layout containers.
15736  * @constructor
15737  * @param {Roo.Element/String/Object} config The configuration options.
15738  */
15739 Roo.BoxComponent = function(config){
15740     Roo.Component.call(this, config);
15741     this.addEvents({
15742         /**
15743          * @event resize
15744          * Fires after the component is resized.
15745              * @param {Roo.Component} this
15746              * @param {Number} adjWidth The box-adjusted width that was set
15747              * @param {Number} adjHeight The box-adjusted height that was set
15748              * @param {Number} rawWidth The width that was originally specified
15749              * @param {Number} rawHeight The height that was originally specified
15750              */
15751         resize : true,
15752         /**
15753          * @event move
15754          * Fires after the component is moved.
15755              * @param {Roo.Component} this
15756              * @param {Number} x The new x position
15757              * @param {Number} y The new y position
15758              */
15759         move : true
15760     });
15761 };
15762
15763 Roo.extend(Roo.BoxComponent, Roo.Component, {
15764     // private, set in afterRender to signify that the component has been rendered
15765     boxReady : false,
15766     // private, used to defer height settings to subclasses
15767     deferHeight: false,
15768     /** @cfg {Number} width
15769      * width (optional) size of component
15770      */
15771      /** @cfg {Number} height
15772      * height (optional) size of component
15773      */
15774      
15775     /**
15776      * Sets the width and height of the component.  This method fires the resize event.  This method can accept
15777      * either width and height as separate numeric arguments, or you can pass a size object like {width:10, height:20}.
15778      * @param {Number/Object} width The new width to set, or a size object in the format {width, height}
15779      * @param {Number} height The new height to set (not required if a size object is passed as the first arg)
15780      * @return {Roo.BoxComponent} this
15781      */
15782     setSize : function(w, h){
15783         // support for standard size objects
15784         if(typeof w == 'object'){
15785             h = w.height;
15786             w = w.width;
15787         }
15788         // not rendered
15789         if(!this.boxReady){
15790             this.width = w;
15791             this.height = h;
15792             return this;
15793         }
15794
15795         // prevent recalcs when not needed
15796         if(this.lastSize && this.lastSize.width == w && this.lastSize.height == h){
15797             return this;
15798         }
15799         this.lastSize = {width: w, height: h};
15800
15801         var adj = this.adjustSize(w, h);
15802         var aw = adj.width, ah = adj.height;
15803         if(aw !== undefined || ah !== undefined){ // this code is nasty but performs better with floaters
15804             var rz = this.getResizeEl();
15805             if(!this.deferHeight && aw !== undefined && ah !== undefined){
15806                 rz.setSize(aw, ah);
15807             }else if(!this.deferHeight && ah !== undefined){
15808                 rz.setHeight(ah);
15809             }else if(aw !== undefined){
15810                 rz.setWidth(aw);
15811             }
15812             this.onResize(aw, ah, w, h);
15813             this.fireEvent('resize', this, aw, ah, w, h);
15814         }
15815         return this;
15816     },
15817
15818     /**
15819      * Gets the current size of the component's underlying element.
15820      * @return {Object} An object containing the element's size {width: (element width), height: (element height)}
15821      */
15822     getSize : function(){
15823         return this.el.getSize();
15824     },
15825
15826     /**
15827      * Gets the current XY position of the component's underlying element.
15828      * @param {Boolean} local (optional) If true the element's left and top are returned instead of page XY (defaults to false)
15829      * @return {Array} The XY position of the element (e.g., [100, 200])
15830      */
15831     getPosition : function(local){
15832         if(local === true){
15833             return [this.el.getLeft(true), this.el.getTop(true)];
15834         }
15835         return this.xy || this.el.getXY();
15836     },
15837
15838     /**
15839      * Gets the current box measurements of the component's underlying element.
15840      * @param {Boolean} local (optional) If true the element's left and top are returned instead of page XY (defaults to false)
15841      * @returns {Object} box An object in the format {x, y, width, height}
15842      */
15843     getBox : function(local){
15844         var s = this.el.getSize();
15845         if(local){
15846             s.x = this.el.getLeft(true);
15847             s.y = this.el.getTop(true);
15848         }else{
15849             var xy = this.xy || this.el.getXY();
15850             s.x = xy[0];
15851             s.y = xy[1];
15852         }
15853         return s;
15854     },
15855
15856     /**
15857      * Sets the current box measurements of the component's underlying element.
15858      * @param {Object} box An object in the format {x, y, width, height}
15859      * @returns {Roo.BoxComponent} this
15860      */
15861     updateBox : function(box){
15862         this.setSize(box.width, box.height);
15863         this.setPagePosition(box.x, box.y);
15864         return this;
15865     },
15866
15867     // protected
15868     getResizeEl : function(){
15869         return this.resizeEl || this.el;
15870     },
15871
15872     // protected
15873     getPositionEl : function(){
15874         return this.positionEl || this.el;
15875     },
15876
15877     /**
15878      * Sets the left and top of the component.  To set the page XY position instead, use {@link #setPagePosition}.
15879      * This method fires the move event.
15880      * @param {Number} left The new left
15881      * @param {Number} top The new top
15882      * @returns {Roo.BoxComponent} this
15883      */
15884     setPosition : function(x, y){
15885         this.x = x;
15886         this.y = y;
15887         if(!this.boxReady){
15888             return this;
15889         }
15890         var adj = this.adjustPosition(x, y);
15891         var ax = adj.x, ay = adj.y;
15892
15893         var el = this.getPositionEl();
15894         if(ax !== undefined || ay !== undefined){
15895             if(ax !== undefined && ay !== undefined){
15896                 el.setLeftTop(ax, ay);
15897             }else if(ax !== undefined){
15898                 el.setLeft(ax);
15899             }else if(ay !== undefined){
15900                 el.setTop(ay);
15901             }
15902             this.onPosition(ax, ay);
15903             this.fireEvent('move', this, ax, ay);
15904         }
15905         return this;
15906     },
15907
15908     /**
15909      * Sets the page XY position of the component.  To set the left and top instead, use {@link #setPosition}.
15910      * This method fires the move event.
15911      * @param {Number} x The new x position
15912      * @param {Number} y The new y position
15913      * @returns {Roo.BoxComponent} this
15914      */
15915     setPagePosition : function(x, y){
15916         this.pageX = x;
15917         this.pageY = y;
15918         if(!this.boxReady){
15919             return;
15920         }
15921         if(x === undefined || y === undefined){ // cannot translate undefined points
15922             return;
15923         }
15924         var p = this.el.translatePoints(x, y);
15925         this.setPosition(p.left, p.top);
15926         return this;
15927     },
15928
15929     // private
15930     onRender : function(ct, position){
15931         Roo.BoxComponent.superclass.onRender.call(this, ct, position);
15932         if(this.resizeEl){
15933             this.resizeEl = Roo.get(this.resizeEl);
15934         }
15935         if(this.positionEl){
15936             this.positionEl = Roo.get(this.positionEl);
15937         }
15938     },
15939
15940     // private
15941     afterRender : function(){
15942         Roo.BoxComponent.superclass.afterRender.call(this);
15943         this.boxReady = true;
15944         this.setSize(this.width, this.height);
15945         if(this.x || this.y){
15946             this.setPosition(this.x, this.y);
15947         }
15948         if(this.pageX || this.pageY){
15949             this.setPagePosition(this.pageX, this.pageY);
15950         }
15951     },
15952
15953     /**
15954      * Force the component's size to recalculate based on the underlying element's current height and width.
15955      * @returns {Roo.BoxComponent} this
15956      */
15957     syncSize : function(){
15958         delete this.lastSize;
15959         this.setSize(this.el.getWidth(), this.el.getHeight());
15960         return this;
15961     },
15962
15963     /**
15964      * Called after the component is resized, this method is empty by default but can be implemented by any
15965      * subclass that needs to perform custom logic after a resize occurs.
15966      * @param {Number} adjWidth The box-adjusted width that was set
15967      * @param {Number} adjHeight The box-adjusted height that was set
15968      * @param {Number} rawWidth The width that was originally specified
15969      * @param {Number} rawHeight The height that was originally specified
15970      */
15971     onResize : function(adjWidth, adjHeight, rawWidth, rawHeight){
15972
15973     },
15974
15975     /**
15976      * Called after the component is moved, this method is empty by default but can be implemented by any
15977      * subclass that needs to perform custom logic after a move occurs.
15978      * @param {Number} x The new x position
15979      * @param {Number} y The new y position
15980      */
15981     onPosition : function(x, y){
15982
15983     },
15984
15985     // private
15986     adjustSize : function(w, h){
15987         if(this.autoWidth){
15988             w = 'auto';
15989         }
15990         if(this.autoHeight){
15991             h = 'auto';
15992         }
15993         return {width : w, height: h};
15994     },
15995
15996     // private
15997     adjustPosition : function(x, y){
15998         return {x : x, y: y};
15999     }
16000 });/*
16001  * Original code for Roojs - LGPL
16002  * <script type="text/javascript">
16003  */
16004  
16005 /**
16006  * @class Roo.XComponent
16007  * A delayed Element creator...
16008  * Or a way to group chunks of interface together.
16009  * technically this is a wrapper around a tree of Roo elements (which defines a 'module'),
16010  *  used in conjunction with XComponent.build() it will create an instance of each element,
16011  *  then call addxtype() to build the User interface.
16012  * 
16013  * Mypart.xyx = new Roo.XComponent({
16014
16015     parent : 'Mypart.xyz', // empty == document.element.!!
16016     order : '001',
16017     name : 'xxxx'
16018     region : 'xxxx'
16019     disabled : function() {} 
16020      
16021     tree : function() { // return an tree of xtype declared components
16022         var MODULE = this;
16023         return 
16024         {
16025             xtype : 'NestedLayoutPanel',
16026             // technicall
16027         }
16028      ]
16029  *})
16030  *
16031  *
16032  * It can be used to build a big heiracy, with parent etc.
16033  * or you can just use this to render a single compoent to a dom element
16034  * MYPART.render(Roo.Element | String(id) | dom_element )
16035  *
16036  *
16037  * Usage patterns.
16038  *
16039  * Classic Roo
16040  *
16041  * Roo is designed primarily as a single page application, so the UI build for a standard interface will
16042  * expect a single 'TOP' level module normally indicated by the 'parent' of the XComponent definition being defined as false.
16043  *
16044  * Each sub module is expected to have a parent pointing to the class name of it's parent module.
16045  *
16046  * When the top level is false, a 'Roo.BorderLayout' is created and the element is flagged as 'topModule'
16047  * - if mulitple topModules exist, the last one is defined as the top module.
16048  *
16049  * Embeded Roo
16050  * 
16051  * When the top level or multiple modules are to embedded into a existing HTML page,
16052  * the parent element can container '#id' of the element where the module will be drawn.
16053  *
16054  * Bootstrap Roo
16055  *
16056  * Unlike classic Roo, the bootstrap tends not to be used as a single page.
16057  * it relies more on a include mechanism, where sub modules are included into an outer page.
16058  * This is normally managed by the builder tools using Roo.apply( options, Included.Sub.Module )
16059  * 
16060  * Bootstrap Roo Included elements
16061  *
16062  * Our builder application needs the ability to preview these sub compoennts. They will normally have parent=false set,
16063  * hence confusing the component builder as it thinks there are multiple top level elements. 
16064  *
16065  * String Over-ride & Translations
16066  *
16067  * Our builder application writes all the strings as _strings and _named_strings. This is to enable the translation of elements,
16068  * and also the 'overlaying of string values - needed when different versions of the same application with different text content
16069  * are needed. @see Roo.XComponent.overlayString  
16070  * 
16071  * 
16072  * 
16073  * @extends Roo.util.Observable
16074  * @constructor
16075  * @param cfg {Object} configuration of component
16076  * 
16077  */
16078 Roo.XComponent = function(cfg) {
16079     Roo.apply(this, cfg);
16080     this.addEvents({ 
16081         /**
16082              * @event built
16083              * Fires when this the componnt is built
16084              * @param {Roo.XComponent} c the component
16085              */
16086         'built' : true
16087         
16088     });
16089     this.region = this.region || 'center'; // default..
16090     Roo.XComponent.register(this);
16091     this.modules = false;
16092     this.el = false; // where the layout goes..
16093     
16094     
16095 }
16096 Roo.extend(Roo.XComponent, Roo.util.Observable, {
16097     /**
16098      * @property el
16099      * The created element (with Roo.factory())
16100      * @type {Roo.Layout}
16101      */
16102     el  : false,
16103     
16104     /**
16105      * @property el
16106      * for BC  - use el in new code
16107      * @type {Roo.Layout}
16108      */
16109     panel : false,
16110     
16111     /**
16112      * @property layout
16113      * for BC  - use el in new code
16114      * @type {Roo.Layout}
16115      */
16116     layout : false,
16117     
16118      /**
16119      * @cfg {Function|boolean} disabled
16120      * If this module is disabled by some rule, return true from the funtion
16121      */
16122     disabled : false,
16123     
16124     /**
16125      * @cfg {String} parent 
16126      * Name of parent element which it get xtype added to..
16127      */
16128     parent: false,
16129     
16130     /**
16131      * @cfg {String} order
16132      * Used to set the order in which elements are created (usefull for multiple tabs)
16133      */
16134     
16135     order : false,
16136     /**
16137      * @cfg {String} name
16138      * String to display while loading.
16139      */
16140     name : false,
16141     /**
16142      * @cfg {String} region
16143      * Region to render component to (defaults to center)
16144      */
16145     region : 'center',
16146     
16147     /**
16148      * @cfg {Array} items
16149      * A single item array - the first element is the root of the tree..
16150      * It's done this way to stay compatible with the Xtype system...
16151      */
16152     items : false,
16153     
16154     /**
16155      * @property _tree
16156      * The method that retuns the tree of parts that make up this compoennt 
16157      * @type {function}
16158      */
16159     _tree  : false,
16160     
16161      /**
16162      * render
16163      * render element to dom or tree
16164      * @param {Roo.Element|String|DomElement} optional render to if parent is not set.
16165      */
16166     
16167     render : function(el)
16168     {
16169         
16170         el = el || false;
16171         var hp = this.parent ? 1 : 0;
16172         Roo.debug &&  Roo.log(this);
16173         
16174         var tree = this._tree ? this._tree() : this.tree();
16175
16176         
16177         if (!el && typeof(this.parent) == 'string' && this.parent.substring(0,1) == '#') {
16178             // if parent is a '#.....' string, then let's use that..
16179             var ename = this.parent.substr(1);
16180             this.parent = false;
16181             Roo.debug && Roo.log(ename);
16182             switch (ename) {
16183                 case 'bootstrap-body':
16184                     if (typeof(tree.el) != 'undefined' && tree.el == document.body)  {
16185                         // this is the BorderLayout standard?
16186                        this.parent = { el : true };
16187                        break;
16188                     }
16189                     if (["Nest", "Content", "Grid", "Tree"].indexOf(tree.xtype)  > -1)  {
16190                         // need to insert stuff...
16191                         this.parent =  {
16192                              el : new Roo.bootstrap.layout.Border({
16193                                  el : document.body, 
16194                      
16195                                  center: {
16196                                     titlebar: false,
16197                                     autoScroll:false,
16198                                     closeOnTab: true,
16199                                     tabPosition: 'top',
16200                                       //resizeTabs: true,
16201                                     alwaysShowTabs: true,
16202                                     hideTabs: false
16203                                      //minTabWidth: 140
16204                                  }
16205                              })
16206                         
16207                          };
16208                          break;
16209                     }
16210                          
16211                     if (typeof(Roo.bootstrap.Body) != 'undefined' ) {
16212                         this.parent = { el :  new  Roo.bootstrap.Body() };
16213                         Roo.debug && Roo.log("setting el to doc body");
16214                          
16215                     } else {
16216                         throw "Container is bootstrap body, but Roo.bootstrap.Body is not defined";
16217                     }
16218                     break;
16219                 case 'bootstrap':
16220                     this.parent = { el : true};
16221                     // fall through
16222                 default:
16223                     el = Roo.get(ename);
16224                     if (typeof(Roo.bootstrap) != 'undefined' && tree['|xns'] == 'Roo.bootstrap') {
16225                         this.parent = { el : true};
16226                     }
16227                     
16228                     break;
16229             }
16230                 
16231             
16232             if (!el && !this.parent) {
16233                 Roo.debug && Roo.log("Warning - element can not be found :#" + ename );
16234                 return;
16235             }
16236         }
16237         
16238         Roo.debug && Roo.log("EL:");
16239         Roo.debug && Roo.log(el);
16240         Roo.debug && Roo.log("this.parent.el:");
16241         Roo.debug && Roo.log(this.parent.el);
16242         
16243
16244         // altertive root elements ??? - we need a better way to indicate these.
16245         var is_alt = Roo.XComponent.is_alt ||
16246                     (typeof(tree.el) != 'undefined' && tree.el == document.body) ||
16247                     (typeof(Roo.bootstrap) != 'undefined' && tree.xns == Roo.bootstrap) ||
16248                     (typeof(Roo.mailer) != 'undefined' && tree.xns == Roo.mailer) ;
16249         
16250         
16251         
16252         if (!this.parent && is_alt) {
16253             //el = Roo.get(document.body);
16254             this.parent = { el : true };
16255         }
16256             
16257             
16258         
16259         if (!this.parent) {
16260             
16261             Roo.debug && Roo.log("no parent - creating one");
16262             
16263             el = el ? Roo.get(el) : false;      
16264             
16265             if (typeof(Roo.BorderLayout) == 'undefined' ) {
16266                 
16267                 this.parent =  {
16268                     el : new Roo.bootstrap.layout.Border({
16269                         el: el || document.body,
16270                     
16271                         center: {
16272                             titlebar: false,
16273                             autoScroll:false,
16274                             closeOnTab: true,
16275                             tabPosition: 'top',
16276                              //resizeTabs: true,
16277                             alwaysShowTabs: false,
16278                             hideTabs: true,
16279                             minTabWidth: 140,
16280                             overflow: 'visible'
16281                          }
16282                      })
16283                 };
16284             } else {
16285             
16286                 // it's a top level one..
16287                 this.parent =  {
16288                     el : new Roo.BorderLayout(el || document.body, {
16289                         center: {
16290                             titlebar: false,
16291                             autoScroll:false,
16292                             closeOnTab: true,
16293                             tabPosition: 'top',
16294                              //resizeTabs: true,
16295                             alwaysShowTabs: el && hp? false :  true,
16296                             hideTabs: el || !hp ? true :  false,
16297                             minTabWidth: 140
16298                          }
16299                     })
16300                 };
16301             }
16302         }
16303         
16304         if (!this.parent.el) {
16305                 // probably an old style ctor, which has been disabled.
16306                 return;
16307
16308         }
16309                 // The 'tree' method is  '_tree now' 
16310             
16311         tree.region = tree.region || this.region;
16312         var is_body = false;
16313         if (this.parent.el === true) {
16314             // bootstrap... - body..
16315             if (el) {
16316                 tree.el = el;
16317             }
16318             this.parent.el = Roo.factory(tree);
16319             is_body = true;
16320         }
16321         
16322         this.el = this.parent.el.addxtype(tree, undefined, is_body);
16323         this.fireEvent('built', this);
16324         
16325         this.panel = this.el;
16326         this.layout = this.panel.layout;
16327         this.parentLayout = this.parent.layout  || false;  
16328          
16329     }
16330     
16331 });
16332
16333 Roo.apply(Roo.XComponent, {
16334     /**
16335      * @property  hideProgress
16336      * true to disable the building progress bar.. usefull on single page renders.
16337      * @type Boolean
16338      */
16339     hideProgress : false,
16340     /**
16341      * @property  buildCompleted
16342      * True when the builder has completed building the interface.
16343      * @type Boolean
16344      */
16345     buildCompleted : false,
16346      
16347     /**
16348      * @property  topModule
16349      * the upper most module - uses document.element as it's constructor.
16350      * @type Object
16351      */
16352      
16353     topModule  : false,
16354       
16355     /**
16356      * @property  modules
16357      * array of modules to be created by registration system.
16358      * @type {Array} of Roo.XComponent
16359      */
16360     
16361     modules : [],
16362     /**
16363      * @property  elmodules
16364      * array of modules to be created by which use #ID 
16365      * @type {Array} of Roo.XComponent
16366      */
16367      
16368     elmodules : [],
16369
16370      /**
16371      * @property  is_alt
16372      * Is an alternative Root - normally used by bootstrap or other systems,
16373      *    where the top element in the tree can wrap 'body' 
16374      * @type {boolean}  (default false)
16375      */
16376      
16377     is_alt : false,
16378     /**
16379      * @property  build_from_html
16380      * Build elements from html - used by bootstrap HTML stuff 
16381      *    - this is cleared after build is completed
16382      * @type {boolean}    (default false)
16383      */
16384      
16385     build_from_html : false,
16386     /**
16387      * Register components to be built later.
16388      *
16389      * This solves the following issues
16390      * - Building is not done on page load, but after an authentication process has occured.
16391      * - Interface elements are registered on page load
16392      * - Parent Interface elements may not be loaded before child, so this handles that..
16393      * 
16394      *
16395      * example:
16396      * 
16397      * MyApp.register({
16398           order : '000001',
16399           module : 'Pman.Tab.projectMgr',
16400           region : 'center',
16401           parent : 'Pman.layout',
16402           disabled : false,  // or use a function..
16403         })
16404      
16405      * * @param {Object} details about module
16406      */
16407     register : function(obj) {
16408                 
16409         Roo.XComponent.event.fireEvent('register', obj);
16410         switch(typeof(obj.disabled) ) {
16411                 
16412             case 'undefined':
16413                 break;
16414             
16415             case 'function':
16416                 if ( obj.disabled() ) {
16417                         return;
16418                 }
16419                 break;
16420             
16421             default:
16422                 if (obj.disabled || obj.region == '#disabled') {
16423                         return;
16424                 }
16425                 break;
16426         }
16427                 
16428         this.modules.push(obj);
16429          
16430     },
16431     /**
16432      * convert a string to an object..
16433      * eg. 'AAA.BBB' -> finds AAA.BBB
16434
16435      */
16436     
16437     toObject : function(str)
16438     {
16439         if (!str || typeof(str) == 'object') {
16440             return str;
16441         }
16442         if (str.substring(0,1) == '#') {
16443             return str;
16444         }
16445
16446         var ar = str.split('.');
16447         var rt, o;
16448         rt = ar.shift();
16449             /** eval:var:o */
16450         try {
16451             eval('if (typeof ' + rt + ' == "undefined"){ o = false;} o = ' + rt + ';');
16452         } catch (e) {
16453             throw "Module not found : " + str;
16454         }
16455         
16456         if (o === false) {
16457             throw "Module not found : " + str;
16458         }
16459         Roo.each(ar, function(e) {
16460             if (typeof(o[e]) == 'undefined') {
16461                 throw "Module not found : " + str;
16462             }
16463             o = o[e];
16464         });
16465         
16466         return o;
16467         
16468     },
16469     
16470     
16471     /**
16472      * move modules into their correct place in the tree..
16473      * 
16474      */
16475     preBuild : function ()
16476     {
16477         var _t = this;
16478         Roo.each(this.modules , function (obj)
16479         {
16480             Roo.XComponent.event.fireEvent('beforebuild', obj);
16481             
16482             var opar = obj.parent;
16483             try { 
16484                 obj.parent = this.toObject(opar);
16485             } catch(e) {
16486                 Roo.debug && Roo.log("parent:toObject failed: " + e.toString());
16487                 return;
16488             }
16489             
16490             if (!obj.parent) {
16491                 Roo.debug && Roo.log("GOT top level module");
16492                 Roo.debug && Roo.log(obj);
16493                 obj.modules = new Roo.util.MixedCollection(false, 
16494                     function(o) { return o.order + '' }
16495                 );
16496                 this.topModule = obj;
16497                 return;
16498             }
16499                         // parent is a string (usually a dom element name..)
16500             if (typeof(obj.parent) == 'string') {
16501                 this.elmodules.push(obj);
16502                 return;
16503             }
16504             if (obj.parent.constructor != Roo.XComponent) {
16505                 Roo.debug && Roo.log("Warning : Object Parent is not instance of XComponent:" + obj.name)
16506             }
16507             if (!obj.parent.modules) {
16508                 obj.parent.modules = new Roo.util.MixedCollection(false, 
16509                     function(o) { return o.order + '' }
16510                 );
16511             }
16512             if (obj.parent.disabled) {
16513                 obj.disabled = true;
16514             }
16515             obj.parent.modules.add(obj);
16516         }, this);
16517     },
16518     
16519      /**
16520      * make a list of modules to build.
16521      * @return {Array} list of modules. 
16522      */ 
16523     
16524     buildOrder : function()
16525     {
16526         var _this = this;
16527         var cmp = function(a,b) {   
16528             return String(a).toUpperCase() > String(b).toUpperCase() ? 1 : -1;
16529         };
16530         if ((!this.topModule || !this.topModule.modules) && !this.elmodules.length) {
16531             throw "No top level modules to build";
16532         }
16533         
16534         // make a flat list in order of modules to build.
16535         var mods = this.topModule ? [ this.topModule ] : [];
16536                 
16537         
16538         // elmodules (is a list of DOM based modules )
16539         Roo.each(this.elmodules, function(e) {
16540             mods.push(e);
16541             if (!this.topModule &&
16542                 typeof(e.parent) == 'string' &&
16543                 e.parent.substring(0,1) == '#' &&
16544                 Roo.get(e.parent.substr(1))
16545                ) {
16546                 
16547                 _this.topModule = e;
16548             }
16549             
16550         });
16551
16552         
16553         // add modules to their parents..
16554         var addMod = function(m) {
16555             Roo.debug && Roo.log("build Order: add: " + m.name);
16556                 
16557             mods.push(m);
16558             if (m.modules && !m.disabled) {
16559                 Roo.debug && Roo.log("build Order: " + m.modules.length + " child modules");
16560                 m.modules.keySort('ASC',  cmp );
16561                 Roo.debug && Roo.log("build Order: " + m.modules.length + " child modules (after sort)");
16562     
16563                 m.modules.each(addMod);
16564             } else {
16565                 Roo.debug && Roo.log("build Order: no child modules");
16566             }
16567             // not sure if this is used any more..
16568             if (m.finalize) {
16569                 m.finalize.name = m.name + " (clean up) ";
16570                 mods.push(m.finalize);
16571             }
16572             
16573         }
16574         if (this.topModule && this.topModule.modules) { 
16575             this.topModule.modules.keySort('ASC',  cmp );
16576             this.topModule.modules.each(addMod);
16577         } 
16578         return mods;
16579     },
16580     
16581      /**
16582      * Build the registered modules.
16583      * @param {Object} parent element.
16584      * @param {Function} optional method to call after module has been added.
16585      * 
16586      */ 
16587    
16588     build : function(opts) 
16589     {
16590         
16591         if (typeof(opts) != 'undefined') {
16592             Roo.apply(this,opts);
16593         }
16594         
16595         this.preBuild();
16596         var mods = this.buildOrder();
16597       
16598         //this.allmods = mods;
16599         //Roo.debug && Roo.log(mods);
16600         //return;
16601         if (!mods.length) { // should not happen
16602             throw "NO modules!!!";
16603         }
16604         
16605         
16606         var msg = "Building Interface...";
16607         // flash it up as modal - so we store the mask!?
16608         if (!this.hideProgress && Roo.MessageBox) {
16609             Roo.MessageBox.show({ title: 'loading' });
16610             Roo.MessageBox.show({
16611                title: "Please wait...",
16612                msg: msg,
16613                width:450,
16614                progress:true,
16615                buttons : false,
16616                closable:false,
16617                modal: false
16618               
16619             });
16620         }
16621         var total = mods.length;
16622         
16623         var _this = this;
16624         var progressRun = function() {
16625             if (!mods.length) {
16626                 Roo.debug && Roo.log('hide?');
16627                 if (!this.hideProgress && Roo.MessageBox) {
16628                     Roo.MessageBox.hide();
16629                 }
16630                 Roo.XComponent.build_from_html = false; // reset, so dialogs will be build from javascript
16631                 
16632                 Roo.XComponent.event.fireEvent('buildcomplete', _this.topModule);
16633                 
16634                 // THE END...
16635                 return false;   
16636             }
16637             
16638             var m = mods.shift();
16639             
16640             
16641             Roo.debug && Roo.log(m);
16642             // not sure if this is supported any more.. - modules that are are just function
16643             if (typeof(m) == 'function') { 
16644                 m.call(this);
16645                 return progressRun.defer(10, _this);
16646             } 
16647             
16648             
16649             msg = "Building Interface " + (total  - mods.length) + 
16650                     " of " + total + 
16651                     (m.name ? (' - ' + m.name) : '');
16652                         Roo.debug && Roo.log(msg);
16653             if (!_this.hideProgress &&  Roo.MessageBox) { 
16654                 Roo.MessageBox.updateProgress(  (total  - mods.length)/total, msg  );
16655             }
16656             
16657          
16658             // is the module disabled?
16659             var disabled = (typeof(m.disabled) == 'function') ?
16660                 m.disabled.call(m.module.disabled) : m.disabled;    
16661             
16662             
16663             if (disabled) {
16664                 return progressRun(); // we do not update the display!
16665             }
16666             
16667             // now build 
16668             
16669                         
16670                         
16671             m.render();
16672             // it's 10 on top level, and 1 on others??? why...
16673             return progressRun.defer(10, _this);
16674              
16675         }
16676         progressRun.defer(1, _this);
16677      
16678         
16679         
16680     },
16681     /**
16682      * Overlay a set of modified strings onto a component
16683      * This is dependant on our builder exporting the strings and 'named strings' elements.
16684      * 
16685      * @param {Object} element to overlay on - eg. Pman.Dialog.Login
16686      * @param {Object} associative array of 'named' string and it's new value.
16687      * 
16688      */
16689         overlayStrings : function( component, strings )
16690     {
16691         if (typeof(component['_named_strings']) == 'undefined') {
16692             throw "ERROR: component does not have _named_strings";
16693         }
16694         for ( var k in strings ) {
16695             var md = typeof(component['_named_strings'][k]) == 'undefined' ? false : component['_named_strings'][k];
16696             if (md !== false) {
16697                 component['_strings'][md] = strings[k];
16698             } else {
16699                 Roo.log('could not find named string: ' + k + ' in');
16700                 Roo.log(component);
16701             }
16702             
16703         }
16704         
16705     },
16706     
16707         
16708         /**
16709          * Event Object.
16710          *
16711          *
16712          */
16713         event: false, 
16714     /**
16715          * wrapper for event.on - aliased later..  
16716          * Typically use to register a event handler for register:
16717          *
16718          * eg. Roo.XComponent.on('register', function(comp) { comp.disable = true } );
16719          *
16720          */
16721     on : false
16722    
16723     
16724     
16725 });
16726
16727 Roo.XComponent.event = new Roo.util.Observable({
16728                 events : { 
16729                         /**
16730                          * @event register
16731                          * Fires when an Component is registered,
16732                          * set the disable property on the Component to stop registration.
16733                          * @param {Roo.XComponent} c the component being registerd.
16734                          * 
16735                          */
16736                         'register' : true,
16737             /**
16738                          * @event beforebuild
16739                          * Fires before each Component is built
16740                          * can be used to apply permissions.
16741                          * @param {Roo.XComponent} c the component being registerd.
16742                          * 
16743                          */
16744                         'beforebuild' : true,
16745                         /**
16746                          * @event buildcomplete
16747                          * Fires on the top level element when all elements have been built
16748                          * @param {Roo.XComponent} the top level component.
16749                          */
16750                         'buildcomplete' : true
16751                         
16752                 }
16753 });
16754
16755 Roo.XComponent.on = Roo.XComponent.event.on.createDelegate(Roo.XComponent.event); 
16756  //
16757  /**
16758  * marked - a markdown parser
16759  * Copyright (c) 2011-2014, Christopher Jeffrey. (MIT Licensed)
16760  * https://github.com/chjj/marked
16761  */
16762
16763
16764 /**
16765  *
16766  * Roo.Markdown - is a very crude wrapper around marked..
16767  *
16768  * usage:
16769  * 
16770  * alert( Roo.Markdown.toHtml("Markdown *rocks*.") );
16771  * 
16772  * Note: move the sample code to the bottom of this
16773  * file before uncommenting it.
16774  *
16775  */
16776
16777 Roo.Markdown = {};
16778 Roo.Markdown.toHtml = function(text) {
16779     
16780     var c = new Roo.Markdown.marked.setOptions({
16781             renderer: new Roo.Markdown.marked.Renderer(),
16782             gfm: true,
16783             tables: true,
16784             breaks: false,
16785             pedantic: false,
16786             sanitize: false,
16787             smartLists: true,
16788             smartypants: false
16789           });
16790     // A FEW HACKS!!?
16791     
16792     text = text.replace(/\\\n/g,' ');
16793     return Roo.Markdown.marked(text);
16794 };
16795 //
16796 // converter
16797 //
16798 // Wraps all "globals" so that the only thing
16799 // exposed is makeHtml().
16800 //
16801 (function() {
16802     
16803     /**
16804      * Block-Level Grammar
16805      */
16806     
16807     var block = {
16808       newline: /^\n+/,
16809       code: /^( {4}[^\n]+\n*)+/,
16810       fences: noop,
16811       hr: /^( *[-*_]){3,} *(?:\n+|$)/,
16812       heading: /^ *(#{1,6}) *([^\n]+?) *#* *(?:\n+|$)/,
16813       nptable: noop,
16814       lheading: /^([^\n]+)\n *(=|-){2,} *(?:\n+|$)/,
16815       blockquote: /^( *>[^\n]+(\n(?!def)[^\n]+)*\n*)+/,
16816       list: /^( *)(bull) [\s\S]+?(?:hr|def|\n{2,}(?! )(?!\1bull )\n*|\s*$)/,
16817       html: /^ *(?:comment *(?:\n|\s*$)|closed *(?:\n{2,}|\s*$)|closing *(?:\n{2,}|\s*$))/,
16818       def: /^ *\[([^\]]+)\]: *<?([^\s>]+)>?(?: +["(]([^\n]+)[")])? *(?:\n+|$)/,
16819       table: noop,
16820       paragraph: /^((?:[^\n]+\n?(?!hr|heading|lheading|blockquote|tag|def))+)\n*/,
16821       text: /^[^\n]+/
16822     };
16823     
16824     block.bullet = /(?:[*+-]|\d+\.)/;
16825     block.item = /^( *)(bull) [^\n]*(?:\n(?!\1bull )[^\n]*)*/;
16826     block.item = replace(block.item, 'gm')
16827       (/bull/g, block.bullet)
16828       ();
16829     
16830     block.list = replace(block.list)
16831       (/bull/g, block.bullet)
16832       ('hr', '\\n+(?=\\1?(?:[-*_] *){3,}(?:\\n+|$))')
16833       ('def', '\\n+(?=' + block.def.source + ')')
16834       ();
16835     
16836     block.blockquote = replace(block.blockquote)
16837       ('def', block.def)
16838       ();
16839     
16840     block._tag = '(?!(?:'
16841       + 'a|em|strong|small|s|cite|q|dfn|abbr|data|time|code'
16842       + '|var|samp|kbd|sub|sup|i|b|u|mark|ruby|rt|rp|bdi|bdo'
16843       + '|span|br|wbr|ins|del|img)\\b)\\w+(?!:/|[^\\w\\s@]*@)\\b';
16844     
16845     block.html = replace(block.html)
16846       ('comment', /<!--[\s\S]*?-->/)
16847       ('closed', /<(tag)[\s\S]+?<\/\1>/)
16848       ('closing', /<tag(?:"[^"]*"|'[^']*'|[^'">])*?>/)
16849       (/tag/g, block._tag)
16850       ();
16851     
16852     block.paragraph = replace(block.paragraph)
16853       ('hr', block.hr)
16854       ('heading', block.heading)
16855       ('lheading', block.lheading)
16856       ('blockquote', block.blockquote)
16857       ('tag', '<' + block._tag)
16858       ('def', block.def)
16859       ();
16860     
16861     /**
16862      * Normal Block Grammar
16863      */
16864     
16865     block.normal = merge({}, block);
16866     
16867     /**
16868      * GFM Block Grammar
16869      */
16870     
16871     block.gfm = merge({}, block.normal, {
16872       fences: /^ *(`{3,}|~{3,})[ \.]*(\S+)? *\n([\s\S]*?)\s*\1 *(?:\n+|$)/,
16873       paragraph: /^/,
16874       heading: /^ *(#{1,6}) +([^\n]+?) *#* *(?:\n+|$)/
16875     });
16876     
16877     block.gfm.paragraph = replace(block.paragraph)
16878       ('(?!', '(?!'
16879         + block.gfm.fences.source.replace('\\1', '\\2') + '|'
16880         + block.list.source.replace('\\1', '\\3') + '|')
16881       ();
16882     
16883     /**
16884      * GFM + Tables Block Grammar
16885      */
16886     
16887     block.tables = merge({}, block.gfm, {
16888       nptable: /^ *(\S.*\|.*)\n *([-:]+ *\|[-| :]*)\n((?:.*\|.*(?:\n|$))*)\n*/,
16889       table: /^ *\|(.+)\n *\|( *[-:]+[-| :]*)\n((?: *\|.*(?:\n|$))*)\n*/
16890     });
16891     
16892     /**
16893      * Block Lexer
16894      */
16895     
16896     function Lexer(options) {
16897       this.tokens = [];
16898       this.tokens.links = {};
16899       this.options = options || marked.defaults;
16900       this.rules = block.normal;
16901     
16902       if (this.options.gfm) {
16903         if (this.options.tables) {
16904           this.rules = block.tables;
16905         } else {
16906           this.rules = block.gfm;
16907         }
16908       }
16909     }
16910     
16911     /**
16912      * Expose Block Rules
16913      */
16914     
16915     Lexer.rules = block;
16916     
16917     /**
16918      * Static Lex Method
16919      */
16920     
16921     Lexer.lex = function(src, options) {
16922       var lexer = new Lexer(options);
16923       return lexer.lex(src);
16924     };
16925     
16926     /**
16927      * Preprocessing
16928      */
16929     
16930     Lexer.prototype.lex = function(src) {
16931       src = src
16932         .replace(/\r\n|\r/g, '\n')
16933         .replace(/\t/g, '    ')
16934         .replace(/\u00a0/g, ' ')
16935         .replace(/\u2424/g, '\n');
16936     
16937       return this.token(src, true);
16938     };
16939     
16940     /**
16941      * Lexing
16942      */
16943     
16944     Lexer.prototype.token = function(src, top, bq) {
16945       var src = src.replace(/^ +$/gm, '')
16946         , next
16947         , loose
16948         , cap
16949         , bull
16950         , b
16951         , item
16952         , space
16953         , i
16954         , l;
16955     
16956       while (src) {
16957         // newline
16958         if (cap = this.rules.newline.exec(src)) {
16959           src = src.substring(cap[0].length);
16960           if (cap[0].length > 1) {
16961             this.tokens.push({
16962               type: 'space'
16963             });
16964           }
16965         }
16966     
16967         // code
16968         if (cap = this.rules.code.exec(src)) {
16969           src = src.substring(cap[0].length);
16970           cap = cap[0].replace(/^ {4}/gm, '');
16971           this.tokens.push({
16972             type: 'code',
16973             text: !this.options.pedantic
16974               ? cap.replace(/\n+$/, '')
16975               : cap
16976           });
16977           continue;
16978         }
16979     
16980         // fences (gfm)
16981         if (cap = this.rules.fences.exec(src)) {
16982           src = src.substring(cap[0].length);
16983           this.tokens.push({
16984             type: 'code',
16985             lang: cap[2],
16986             text: cap[3] || ''
16987           });
16988           continue;
16989         }
16990     
16991         // heading
16992         if (cap = this.rules.heading.exec(src)) {
16993           src = src.substring(cap[0].length);
16994           this.tokens.push({
16995             type: 'heading',
16996             depth: cap[1].length,
16997             text: cap[2]
16998           });
16999           continue;
17000         }
17001     
17002         // table no leading pipe (gfm)
17003         if (top && (cap = this.rules.nptable.exec(src))) {
17004           src = src.substring(cap[0].length);
17005     
17006           item = {
17007             type: 'table',
17008             header: cap[1].replace(/^ *| *\| *$/g, '').split(/ *\| */),
17009             align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
17010             cells: cap[3].replace(/\n$/, '').split('\n')
17011           };
17012     
17013           for (i = 0; i < item.align.length; i++) {
17014             if (/^ *-+: *$/.test(item.align[i])) {
17015               item.align[i] = 'right';
17016             } else if (/^ *:-+: *$/.test(item.align[i])) {
17017               item.align[i] = 'center';
17018             } else if (/^ *:-+ *$/.test(item.align[i])) {
17019               item.align[i] = 'left';
17020             } else {
17021               item.align[i] = null;
17022             }
17023           }
17024     
17025           for (i = 0; i < item.cells.length; i++) {
17026             item.cells[i] = item.cells[i].split(/ *\| */);
17027           }
17028     
17029           this.tokens.push(item);
17030     
17031           continue;
17032         }
17033     
17034         // lheading
17035         if (cap = this.rules.lheading.exec(src)) {
17036           src = src.substring(cap[0].length);
17037           this.tokens.push({
17038             type: 'heading',
17039             depth: cap[2] === '=' ? 1 : 2,
17040             text: cap[1]
17041           });
17042           continue;
17043         }
17044     
17045         // hr
17046         if (cap = this.rules.hr.exec(src)) {
17047           src = src.substring(cap[0].length);
17048           this.tokens.push({
17049             type: 'hr'
17050           });
17051           continue;
17052         }
17053     
17054         // blockquote
17055         if (cap = this.rules.blockquote.exec(src)) {
17056           src = src.substring(cap[0].length);
17057     
17058           this.tokens.push({
17059             type: 'blockquote_start'
17060           });
17061     
17062           cap = cap[0].replace(/^ *> ?/gm, '');
17063     
17064           // Pass `top` to keep the current
17065           // "toplevel" state. This is exactly
17066           // how markdown.pl works.
17067           this.token(cap, top, true);
17068     
17069           this.tokens.push({
17070             type: 'blockquote_end'
17071           });
17072     
17073           continue;
17074         }
17075     
17076         // list
17077         if (cap = this.rules.list.exec(src)) {
17078           src = src.substring(cap[0].length);
17079           bull = cap[2];
17080     
17081           this.tokens.push({
17082             type: 'list_start',
17083             ordered: bull.length > 1
17084           });
17085     
17086           // Get each top-level item.
17087           cap = cap[0].match(this.rules.item);
17088     
17089           next = false;
17090           l = cap.length;
17091           i = 0;
17092     
17093           for (; i < l; i++) {
17094             item = cap[i];
17095     
17096             // Remove the list item's bullet
17097             // so it is seen as the next token.
17098             space = item.length;
17099             item = item.replace(/^ *([*+-]|\d+\.) +/, '');
17100     
17101             // Outdent whatever the
17102             // list item contains. Hacky.
17103             if (~item.indexOf('\n ')) {
17104               space -= item.length;
17105               item = !this.options.pedantic
17106                 ? item.replace(new RegExp('^ {1,' + space + '}', 'gm'), '')
17107                 : item.replace(/^ {1,4}/gm, '');
17108             }
17109     
17110             // Determine whether the next list item belongs here.
17111             // Backpedal if it does not belong in this list.
17112             if (this.options.smartLists && i !== l - 1) {
17113               b = block.bullet.exec(cap[i + 1])[0];
17114               if (bull !== b && !(bull.length > 1 && b.length > 1)) {
17115                 src = cap.slice(i + 1).join('\n') + src;
17116                 i = l - 1;
17117               }
17118             }
17119     
17120             // Determine whether item is loose or not.
17121             // Use: /(^|\n)(?! )[^\n]+\n\n(?!\s*$)/
17122             // for discount behavior.
17123             loose = next || /\n\n(?!\s*$)/.test(item);
17124             if (i !== l - 1) {
17125               next = item.charAt(item.length - 1) === '\n';
17126               if (!loose) { loose = next; }
17127             }
17128     
17129             this.tokens.push({
17130               type: loose
17131                 ? 'loose_item_start'
17132                 : 'list_item_start'
17133             });
17134     
17135             // Recurse.
17136             this.token(item, false, bq);
17137     
17138             this.tokens.push({
17139               type: 'list_item_end'
17140             });
17141           }
17142     
17143           this.tokens.push({
17144             type: 'list_end'
17145           });
17146     
17147           continue;
17148         }
17149     
17150         // html
17151         if (cap = this.rules.html.exec(src)) {
17152           src = src.substring(cap[0].length);
17153           this.tokens.push({
17154             type: this.options.sanitize
17155               ? 'paragraph'
17156               : 'html',
17157             pre: !this.options.sanitizer
17158               && (cap[1] === 'pre' || cap[1] === 'script' || cap[1] === 'style'),
17159             text: cap[0]
17160           });
17161           continue;
17162         }
17163     
17164         // def
17165         if ((!bq && top) && (cap = this.rules.def.exec(src))) {
17166           src = src.substring(cap[0].length);
17167           this.tokens.links[cap[1].toLowerCase()] = {
17168             href: cap[2],
17169             title: cap[3]
17170           };
17171           continue;
17172         }
17173     
17174         // table (gfm)
17175         if (top && (cap = this.rules.table.exec(src))) {
17176           src = src.substring(cap[0].length);
17177     
17178           item = {
17179             type: 'table',
17180             header: cap[1].replace(/^ *| *\| *$/g, '').split(/ *\| */),
17181             align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
17182             cells: cap[3].replace(/(?: *\| *)?\n$/, '').split('\n')
17183           };
17184     
17185           for (i = 0; i < item.align.length; i++) {
17186             if (/^ *-+: *$/.test(item.align[i])) {
17187               item.align[i] = 'right';
17188             } else if (/^ *:-+: *$/.test(item.align[i])) {
17189               item.align[i] = 'center';
17190             } else if (/^ *:-+ *$/.test(item.align[i])) {
17191               item.align[i] = 'left';
17192             } else {
17193               item.align[i] = null;
17194             }
17195           }
17196     
17197           for (i = 0; i < item.cells.length; i++) {
17198             item.cells[i] = item.cells[i]
17199               .replace(/^ *\| *| *\| *$/g, '')
17200               .split(/ *\| */);
17201           }
17202     
17203           this.tokens.push(item);
17204     
17205           continue;
17206         }
17207     
17208         // top-level paragraph
17209         if (top && (cap = this.rules.paragraph.exec(src))) {
17210           src = src.substring(cap[0].length);
17211           this.tokens.push({
17212             type: 'paragraph',
17213             text: cap[1].charAt(cap[1].length - 1) === '\n'
17214               ? cap[1].slice(0, -1)
17215               : cap[1]
17216           });
17217           continue;
17218         }
17219     
17220         // text
17221         if (cap = this.rules.text.exec(src)) {
17222           // Top-level should never reach here.
17223           src = src.substring(cap[0].length);
17224           this.tokens.push({
17225             type: 'text',
17226             text: cap[0]
17227           });
17228           continue;
17229         }
17230     
17231         if (src) {
17232           throw new
17233             Error('Infinite loop on byte: ' + src.charCodeAt(0));
17234         }
17235       }
17236     
17237       return this.tokens;
17238     };
17239     
17240     /**
17241      * Inline-Level Grammar
17242      */
17243     
17244     var inline = {
17245       escape: /^\\([\\`*{}\[\]()#+\-.!_>])/,
17246       autolink: /^<([^ >]+(@|:\/)[^ >]+)>/,
17247       url: noop,
17248       tag: /^<!--[\s\S]*?-->|^<\/?\w+(?:"[^"]*"|'[^']*'|[^'">])*?>/,
17249       link: /^!?\[(inside)\]\(href\)/,
17250       reflink: /^!?\[(inside)\]\s*\[([^\]]*)\]/,
17251       nolink: /^!?\[((?:\[[^\]]*\]|[^\[\]])*)\]/,
17252       strong: /^__([\s\S]+?)__(?!_)|^\*\*([\s\S]+?)\*\*(?!\*)/,
17253       em: /^\b_((?:[^_]|__)+?)_\b|^\*((?:\*\*|[\s\S])+?)\*(?!\*)/,
17254       code: /^(`+)\s*([\s\S]*?[^`])\s*\1(?!`)/,
17255       br: /^ {2,}\n(?!\s*$)/,
17256       del: noop,
17257       text: /^[\s\S]+?(?=[\\<!\[_*`]| {2,}\n|$)/
17258     };
17259     
17260     inline._inside = /(?:\[[^\]]*\]|[^\[\]]|\](?=[^\[]*\]))*/;
17261     inline._href = /\s*<?([\s\S]*?)>?(?:\s+['"]([\s\S]*?)['"])?\s*/;
17262     
17263     inline.link = replace(inline.link)
17264       ('inside', inline._inside)
17265       ('href', inline._href)
17266       ();
17267     
17268     inline.reflink = replace(inline.reflink)
17269       ('inside', inline._inside)
17270       ();
17271     
17272     /**
17273      * Normal Inline Grammar
17274      */
17275     
17276     inline.normal = merge({}, inline);
17277     
17278     /**
17279      * Pedantic Inline Grammar
17280      */
17281     
17282     inline.pedantic = merge({}, inline.normal, {
17283       strong: /^__(?=\S)([\s\S]*?\S)__(?!_)|^\*\*(?=\S)([\s\S]*?\S)\*\*(?!\*)/,
17284       em: /^_(?=\S)([\s\S]*?\S)_(?!_)|^\*(?=\S)([\s\S]*?\S)\*(?!\*)/
17285     });
17286     
17287     /**
17288      * GFM Inline Grammar
17289      */
17290     
17291     inline.gfm = merge({}, inline.normal, {
17292       escape: replace(inline.escape)('])', '~|])')(),
17293       url: /^(https?:\/\/[^\s<]+[^<.,:;"')\]\s])/,
17294       del: /^~~(?=\S)([\s\S]*?\S)~~/,
17295       text: replace(inline.text)
17296         (']|', '~]|')
17297         ('|', '|https?://|')
17298         ()
17299     });
17300     
17301     /**
17302      * GFM + Line Breaks Inline Grammar
17303      */
17304     
17305     inline.breaks = merge({}, inline.gfm, {
17306       br: replace(inline.br)('{2,}', '*')(),
17307       text: replace(inline.gfm.text)('{2,}', '*')()
17308     });
17309     
17310     /**
17311      * Inline Lexer & Compiler
17312      */
17313     
17314     function InlineLexer(links, options) {
17315       this.options = options || marked.defaults;
17316       this.links = links;
17317       this.rules = inline.normal;
17318       this.renderer = this.options.renderer || new Renderer;
17319       this.renderer.options = this.options;
17320     
17321       if (!this.links) {
17322         throw new
17323           Error('Tokens array requires a `links` property.');
17324       }
17325     
17326       if (this.options.gfm) {
17327         if (this.options.breaks) {
17328           this.rules = inline.breaks;
17329         } else {
17330           this.rules = inline.gfm;
17331         }
17332       } else if (this.options.pedantic) {
17333         this.rules = inline.pedantic;
17334       }
17335     }
17336     
17337     /**
17338      * Expose Inline Rules
17339      */
17340     
17341     InlineLexer.rules = inline;
17342     
17343     /**
17344      * Static Lexing/Compiling Method
17345      */
17346     
17347     InlineLexer.output = function(src, links, options) {
17348       var inline = new InlineLexer(links, options);
17349       return inline.output(src);
17350     };
17351     
17352     /**
17353      * Lexing/Compiling
17354      */
17355     
17356     InlineLexer.prototype.output = function(src) {
17357       var out = ''
17358         , link
17359         , text
17360         , href
17361         , cap;
17362     
17363       while (src) {
17364         // escape
17365         if (cap = this.rules.escape.exec(src)) {
17366           src = src.substring(cap[0].length);
17367           out += cap[1];
17368           continue;
17369         }
17370     
17371         // autolink
17372         if (cap = this.rules.autolink.exec(src)) {
17373           src = src.substring(cap[0].length);
17374           if (cap[2] === '@') {
17375             text = cap[1].charAt(6) === ':'
17376               ? this.mangle(cap[1].substring(7))
17377               : this.mangle(cap[1]);
17378             href = this.mangle('mailto:') + text;
17379           } else {
17380             text = escape(cap[1]);
17381             href = text;
17382           }
17383           out += this.renderer.link(href, null, text);
17384           continue;
17385         }
17386     
17387         // url (gfm)
17388         if (!this.inLink && (cap = this.rules.url.exec(src))) {
17389           src = src.substring(cap[0].length);
17390           text = escape(cap[1]);
17391           href = text;
17392           out += this.renderer.link(href, null, text);
17393           continue;
17394         }
17395     
17396         // tag
17397         if (cap = this.rules.tag.exec(src)) {
17398           if (!this.inLink && /^<a /i.test(cap[0])) {
17399             this.inLink = true;
17400           } else if (this.inLink && /^<\/a>/i.test(cap[0])) {
17401             this.inLink = false;
17402           }
17403           src = src.substring(cap[0].length);
17404           out += this.options.sanitize
17405             ? this.options.sanitizer
17406               ? this.options.sanitizer(cap[0])
17407               : escape(cap[0])
17408             : cap[0];
17409           continue;
17410         }
17411     
17412         // link
17413         if (cap = this.rules.link.exec(src)) {
17414           src = src.substring(cap[0].length);
17415           this.inLink = true;
17416           out += this.outputLink(cap, {
17417             href: cap[2],
17418             title: cap[3]
17419           });
17420           this.inLink = false;
17421           continue;
17422         }
17423     
17424         // reflink, nolink
17425         if ((cap = this.rules.reflink.exec(src))
17426             || (cap = this.rules.nolink.exec(src))) {
17427           src = src.substring(cap[0].length);
17428           link = (cap[2] || cap[1]).replace(/\s+/g, ' ');
17429           link = this.links[link.toLowerCase()];
17430           if (!link || !link.href) {
17431             out += cap[0].charAt(0);
17432             src = cap[0].substring(1) + src;
17433             continue;
17434           }
17435           this.inLink = true;
17436           out += this.outputLink(cap, link);
17437           this.inLink = false;
17438           continue;
17439         }
17440     
17441         // strong
17442         if (cap = this.rules.strong.exec(src)) {
17443           src = src.substring(cap[0].length);
17444           out += this.renderer.strong(this.output(cap[2] || cap[1]));
17445           continue;
17446         }
17447     
17448         // em
17449         if (cap = this.rules.em.exec(src)) {
17450           src = src.substring(cap[0].length);
17451           out += this.renderer.em(this.output(cap[2] || cap[1]));
17452           continue;
17453         }
17454     
17455         // code
17456         if (cap = this.rules.code.exec(src)) {
17457           src = src.substring(cap[0].length);
17458           out += this.renderer.codespan(escape(cap[2], true));
17459           continue;
17460         }
17461     
17462         // br
17463         if (cap = this.rules.br.exec(src)) {
17464           src = src.substring(cap[0].length);
17465           out += this.renderer.br();
17466           continue;
17467         }
17468     
17469         // del (gfm)
17470         if (cap = this.rules.del.exec(src)) {
17471           src = src.substring(cap[0].length);
17472           out += this.renderer.del(this.output(cap[1]));
17473           continue;
17474         }
17475     
17476         // text
17477         if (cap = this.rules.text.exec(src)) {
17478           src = src.substring(cap[0].length);
17479           out += this.renderer.text(escape(this.smartypants(cap[0])));
17480           continue;
17481         }
17482     
17483         if (src) {
17484           throw new
17485             Error('Infinite loop on byte: ' + src.charCodeAt(0));
17486         }
17487       }
17488     
17489       return out;
17490     };
17491     
17492     /**
17493      * Compile Link
17494      */
17495     
17496     InlineLexer.prototype.outputLink = function(cap, link) {
17497       var href = escape(link.href)
17498         , title = link.title ? escape(link.title) : null;
17499     
17500       return cap[0].charAt(0) !== '!'
17501         ? this.renderer.link(href, title, this.output(cap[1]))
17502         : this.renderer.image(href, title, escape(cap[1]));
17503     };
17504     
17505     /**
17506      * Smartypants Transformations
17507      */
17508     
17509     InlineLexer.prototype.smartypants = function(text) {
17510       if (!this.options.smartypants)  { return text; }
17511       return text
17512         // em-dashes
17513         .replace(/---/g, '\u2014')
17514         // en-dashes
17515         .replace(/--/g, '\u2013')
17516         // opening singles
17517         .replace(/(^|[-\u2014/(\[{"\s])'/g, '$1\u2018')
17518         // closing singles & apostrophes
17519         .replace(/'/g, '\u2019')
17520         // opening doubles
17521         .replace(/(^|[-\u2014/(\[{\u2018\s])"/g, '$1\u201c')
17522         // closing doubles
17523         .replace(/"/g, '\u201d')
17524         // ellipses
17525         .replace(/\.{3}/g, '\u2026');
17526     };
17527     
17528     /**
17529      * Mangle Links
17530      */
17531     
17532     InlineLexer.prototype.mangle = function(text) {
17533       if (!this.options.mangle) { return text; }
17534       var out = ''
17535         , l = text.length
17536         , i = 0
17537         , ch;
17538     
17539       for (; i < l; i++) {
17540         ch = text.charCodeAt(i);
17541         if (Math.random() > 0.5) {
17542           ch = 'x' + ch.toString(16);
17543         }
17544         out += '&#' + ch + ';';
17545       }
17546     
17547       return out;
17548     };
17549     
17550     /**
17551      * Renderer
17552      */
17553     
17554     function Renderer(options) {
17555       this.options = options || {};
17556     }
17557     
17558     Renderer.prototype.code = function(code, lang, escaped) {
17559       if (this.options.highlight) {
17560         var out = this.options.highlight(code, lang);
17561         if (out != null && out !== code) {
17562           escaped = true;
17563           code = out;
17564         }
17565       } else {
17566             // hack!!! - it's already escapeD?
17567             escaped = true;
17568       }
17569     
17570       if (!lang) {
17571         return '<pre><code>'
17572           + (escaped ? code : escape(code, true))
17573           + '\n</code></pre>';
17574       }
17575     
17576       return '<pre><code class="'
17577         + this.options.langPrefix
17578         + escape(lang, true)
17579         + '">'
17580         + (escaped ? code : escape(code, true))
17581         + '\n</code></pre>\n';
17582     };
17583     
17584     Renderer.prototype.blockquote = function(quote) {
17585       return '<blockquote>\n' + quote + '</blockquote>\n';
17586     };
17587     
17588     Renderer.prototype.html = function(html) {
17589       return html;
17590     };
17591     
17592     Renderer.prototype.heading = function(text, level, raw) {
17593       return '<h'
17594         + level
17595         + ' id="'
17596         + this.options.headerPrefix
17597         + raw.toLowerCase().replace(/[^\w]+/g, '-')
17598         + '">'
17599         + text
17600         + '</h'
17601         + level
17602         + '>\n';
17603     };
17604     
17605     Renderer.prototype.hr = function() {
17606       return this.options.xhtml ? '<hr/>\n' : '<hr>\n';
17607     };
17608     
17609     Renderer.prototype.list = function(body, ordered) {
17610       var type = ordered ? 'ol' : 'ul';
17611       return '<' + type + '>\n' + body + '</' + type + '>\n';
17612     };
17613     
17614     Renderer.prototype.listitem = function(text) {
17615       return '<li>' + text + '</li>\n';
17616     };
17617     
17618     Renderer.prototype.paragraph = function(text) {
17619       return '<p>' + text + '</p>\n';
17620     };
17621     
17622     Renderer.prototype.table = function(header, body) {
17623       return '<table class="table table-striped">\n'
17624         + '<thead>\n'
17625         + header
17626         + '</thead>\n'
17627         + '<tbody>\n'
17628         + body
17629         + '</tbody>\n'
17630         + '</table>\n';
17631     };
17632     
17633     Renderer.prototype.tablerow = function(content) {
17634       return '<tr>\n' + content + '</tr>\n';
17635     };
17636     
17637     Renderer.prototype.tablecell = function(content, flags) {
17638       var type = flags.header ? 'th' : 'td';
17639       var tag = flags.align
17640         ? '<' + type + ' style="text-align:' + flags.align + '">'
17641         : '<' + type + '>';
17642       return tag + content + '</' + type + '>\n';
17643     };
17644     
17645     // span level renderer
17646     Renderer.prototype.strong = function(text) {
17647       return '<strong>' + text + '</strong>';
17648     };
17649     
17650     Renderer.prototype.em = function(text) {
17651       return '<em>' + text + '</em>';
17652     };
17653     
17654     Renderer.prototype.codespan = function(text) {
17655       return '<code>' + text + '</code>';
17656     };
17657     
17658     Renderer.prototype.br = function() {
17659       return this.options.xhtml ? '<br/>' : '<br>';
17660     };
17661     
17662     Renderer.prototype.del = function(text) {
17663       return '<del>' + text + '</del>';
17664     };
17665     
17666     Renderer.prototype.link = function(href, title, text) {
17667       if (this.options.sanitize) {
17668         try {
17669           var prot = decodeURIComponent(unescape(href))
17670             .replace(/[^\w:]/g, '')
17671             .toLowerCase();
17672         } catch (e) {
17673           return '';
17674         }
17675         if (prot.indexOf('javascript:') === 0 || prot.indexOf('vbscript:') === 0) {
17676           return '';
17677         }
17678       }
17679       var out = '<a href="' + href + '"';
17680       if (title) {
17681         out += ' title="' + title + '"';
17682       }
17683       out += '>' + text + '</a>';
17684       return out;
17685     };
17686     
17687     Renderer.prototype.image = function(href, title, text) {
17688       var out = '<img src="' + href + '" alt="' + text + '"';
17689       if (title) {
17690         out += ' title="' + title + '"';
17691       }
17692       out += this.options.xhtml ? '/>' : '>';
17693       return out;
17694     };
17695     
17696     Renderer.prototype.text = function(text) {
17697       return text;
17698     };
17699     
17700     /**
17701      * Parsing & Compiling
17702      */
17703     
17704     function Parser(options) {
17705       this.tokens = [];
17706       this.token = null;
17707       this.options = options || marked.defaults;
17708       this.options.renderer = this.options.renderer || new Renderer;
17709       this.renderer = this.options.renderer;
17710       this.renderer.options = this.options;
17711     }
17712     
17713     /**
17714      * Static Parse Method
17715      */
17716     
17717     Parser.parse = function(src, options, renderer) {
17718       var parser = new Parser(options, renderer);
17719       return parser.parse(src);
17720     };
17721     
17722     /**
17723      * Parse Loop
17724      */
17725     
17726     Parser.prototype.parse = function(src) {
17727       this.inline = new InlineLexer(src.links, this.options, this.renderer);
17728       this.tokens = src.reverse();
17729     
17730       var out = '';
17731       while (this.next()) {
17732         out += this.tok();
17733       }
17734     
17735       return out;
17736     };
17737     
17738     /**
17739      * Next Token
17740      */
17741     
17742     Parser.prototype.next = function() {
17743       return this.token = this.tokens.pop();
17744     };
17745     
17746     /**
17747      * Preview Next Token
17748      */
17749     
17750     Parser.prototype.peek = function() {
17751       return this.tokens[this.tokens.length - 1] || 0;
17752     };
17753     
17754     /**
17755      * Parse Text Tokens
17756      */
17757     
17758     Parser.prototype.parseText = function() {
17759       var body = this.token.text;
17760     
17761       while (this.peek().type === 'text') {
17762         body += '\n' + this.next().text;
17763       }
17764     
17765       return this.inline.output(body);
17766     };
17767     
17768     /**
17769      * Parse Current Token
17770      */
17771     
17772     Parser.prototype.tok = function() {
17773       switch (this.token.type) {
17774         case 'space': {
17775           return '';
17776         }
17777         case 'hr': {
17778           return this.renderer.hr();
17779         }
17780         case 'heading': {
17781           return this.renderer.heading(
17782             this.inline.output(this.token.text),
17783             this.token.depth,
17784             this.token.text);
17785         }
17786         case 'code': {
17787           return this.renderer.code(this.token.text,
17788             this.token.lang,
17789             this.token.escaped);
17790         }
17791         case 'table': {
17792           var header = ''
17793             , body = ''
17794             , i
17795             , row
17796             , cell
17797             , flags
17798             , j;
17799     
17800           // header
17801           cell = '';
17802           for (i = 0; i < this.token.header.length; i++) {
17803             flags = { header: true, align: this.token.align[i] };
17804             cell += this.renderer.tablecell(
17805               this.inline.output(this.token.header[i]),
17806               { header: true, align: this.token.align[i] }
17807             );
17808           }
17809           header += this.renderer.tablerow(cell);
17810     
17811           for (i = 0; i < this.token.cells.length; i++) {
17812             row = this.token.cells[i];
17813     
17814             cell = '';
17815             for (j = 0; j < row.length; j++) {
17816               cell += this.renderer.tablecell(
17817                 this.inline.output(row[j]),
17818                 { header: false, align: this.token.align[j] }
17819               );
17820             }
17821     
17822             body += this.renderer.tablerow(cell);
17823           }
17824           return this.renderer.table(header, body);
17825         }
17826         case 'blockquote_start': {
17827           var body = '';
17828     
17829           while (this.next().type !== 'blockquote_end') {
17830             body += this.tok();
17831           }
17832     
17833           return this.renderer.blockquote(body);
17834         }
17835         case 'list_start': {
17836           var body = ''
17837             , ordered = this.token.ordered;
17838     
17839           while (this.next().type !== 'list_end') {
17840             body += this.tok();
17841           }
17842     
17843           return this.renderer.list(body, ordered);
17844         }
17845         case 'list_item_start': {
17846           var body = '';
17847     
17848           while (this.next().type !== 'list_item_end') {
17849             body += this.token.type === 'text'
17850               ? this.parseText()
17851               : this.tok();
17852           }
17853     
17854           return this.renderer.listitem(body);
17855         }
17856         case 'loose_item_start': {
17857           var body = '';
17858     
17859           while (this.next().type !== 'list_item_end') {
17860             body += this.tok();
17861           }
17862     
17863           return this.renderer.listitem(body);
17864         }
17865         case 'html': {
17866           var html = !this.token.pre && !this.options.pedantic
17867             ? this.inline.output(this.token.text)
17868             : this.token.text;
17869           return this.renderer.html(html);
17870         }
17871         case 'paragraph': {
17872           return this.renderer.paragraph(this.inline.output(this.token.text));
17873         }
17874         case 'text': {
17875           return this.renderer.paragraph(this.parseText());
17876         }
17877       }
17878     };
17879     
17880     /**
17881      * Helpers
17882      */
17883     
17884     function escape(html, encode) {
17885       return html
17886         .replace(!encode ? /&(?!#?\w+;)/g : /&/g, '&amp;')
17887         .replace(/</g, '&lt;')
17888         .replace(/>/g, '&gt;')
17889         .replace(/"/g, '&quot;')
17890         .replace(/'/g, '&#39;');
17891     }
17892     
17893     function unescape(html) {
17894         // explicitly match decimal, hex, and named HTML entities 
17895       return html.replace(/&(#(?:\d+)|(?:#x[0-9A-Fa-f]+)|(?:\w+));?/g, function(_, n) {
17896         n = n.toLowerCase();
17897         if (n === 'colon') { return ':'; }
17898         if (n.charAt(0) === '#') {
17899           return n.charAt(1) === 'x'
17900             ? String.fromCharCode(parseInt(n.substring(2), 16))
17901             : String.fromCharCode(+n.substring(1));
17902         }
17903         return '';
17904       });
17905     }
17906     
17907     function replace(regex, opt) {
17908       regex = regex.source;
17909       opt = opt || '';
17910       return function self(name, val) {
17911         if (!name) { return new RegExp(regex, opt); }
17912         val = val.source || val;
17913         val = val.replace(/(^|[^\[])\^/g, '$1');
17914         regex = regex.replace(name, val);
17915         return self;
17916       };
17917     }
17918     
17919     function noop() {}
17920     noop.exec = noop;
17921     
17922     function merge(obj) {
17923       var i = 1
17924         , target
17925         , key;
17926     
17927       for (; i < arguments.length; i++) {
17928         target = arguments[i];
17929         for (key in target) {
17930           if (Object.prototype.hasOwnProperty.call(target, key)) {
17931             obj[key] = target[key];
17932           }
17933         }
17934       }
17935     
17936       return obj;
17937     }
17938     
17939     
17940     /**
17941      * Marked
17942      */
17943     
17944     function marked(src, opt, callback) {
17945       if (callback || typeof opt === 'function') {
17946         if (!callback) {
17947           callback = opt;
17948           opt = null;
17949         }
17950     
17951         opt = merge({}, marked.defaults, opt || {});
17952     
17953         var highlight = opt.highlight
17954           , tokens
17955           , pending
17956           , i = 0;
17957     
17958         try {
17959           tokens = Lexer.lex(src, opt)
17960         } catch (e) {
17961           return callback(e);
17962         }
17963     
17964         pending = tokens.length;
17965     
17966         var done = function(err) {
17967           if (err) {
17968             opt.highlight = highlight;
17969             return callback(err);
17970           }
17971     
17972           var out;
17973     
17974           try {
17975             out = Parser.parse(tokens, opt);
17976           } catch (e) {
17977             err = e;
17978           }
17979     
17980           opt.highlight = highlight;
17981     
17982           return err
17983             ? callback(err)
17984             : callback(null, out);
17985         };
17986     
17987         if (!highlight || highlight.length < 3) {
17988           return done();
17989         }
17990     
17991         delete opt.highlight;
17992     
17993         if (!pending) { return done(); }
17994     
17995         for (; i < tokens.length; i++) {
17996           (function(token) {
17997             if (token.type !== 'code') {
17998               return --pending || done();
17999             }
18000             return highlight(token.text, token.lang, function(err, code) {
18001               if (err) { return done(err); }
18002               if (code == null || code === token.text) {
18003                 return --pending || done();
18004               }
18005               token.text = code;
18006               token.escaped = true;
18007               --pending || done();
18008             });
18009           })(tokens[i]);
18010         }
18011     
18012         return;
18013       }
18014       try {
18015         if (opt) { opt = merge({}, marked.defaults, opt); }
18016         return Parser.parse(Lexer.lex(src, opt), opt);
18017       } catch (e) {
18018         e.message += '\nPlease report this to https://github.com/chjj/marked.';
18019         if ((opt || marked.defaults).silent) {
18020           return '<p>An error occured:</p><pre>'
18021             + escape(e.message + '', true)
18022             + '</pre>';
18023         }
18024         throw e;
18025       }
18026     }
18027     
18028     /**
18029      * Options
18030      */
18031     
18032     marked.options =
18033     marked.setOptions = function(opt) {
18034       merge(marked.defaults, opt);
18035       return marked;
18036     };
18037     
18038     marked.defaults = {
18039       gfm: true,
18040       tables: true,
18041       breaks: false,
18042       pedantic: false,
18043       sanitize: false,
18044       sanitizer: null,
18045       mangle: true,
18046       smartLists: false,
18047       silent: false,
18048       highlight: null,
18049       langPrefix: 'lang-',
18050       smartypants: false,
18051       headerPrefix: '',
18052       renderer: new Renderer,
18053       xhtml: false
18054     };
18055     
18056     /**
18057      * Expose
18058      */
18059     
18060     marked.Parser = Parser;
18061     marked.parser = Parser.parse;
18062     
18063     marked.Renderer = Renderer;
18064     
18065     marked.Lexer = Lexer;
18066     marked.lexer = Lexer.lex;
18067     
18068     marked.InlineLexer = InlineLexer;
18069     marked.inlineLexer = InlineLexer.output;
18070     
18071     marked.parse = marked;
18072     
18073     Roo.Markdown.marked = marked;
18074
18075 })();/*
18076  * Based on:
18077  * Ext JS Library 1.1.1
18078  * Copyright(c) 2006-2007, Ext JS, LLC.
18079  *
18080  * Originally Released Under LGPL - original licence link has changed is not relivant.
18081  *
18082  * Fork - LGPL
18083  * <script type="text/javascript">
18084  */
18085
18086
18087
18088 /*
18089  * These classes are derivatives of the similarly named classes in the YUI Library.
18090  * The original license:
18091  * Copyright (c) 2006, Yahoo! Inc. All rights reserved.
18092  * Code licensed under the BSD License:
18093  * http://developer.yahoo.net/yui/license.txt
18094  */
18095
18096 (function() {
18097
18098 var Event=Roo.EventManager;
18099 var Dom=Roo.lib.Dom;
18100
18101 /**
18102  * @class Roo.dd.DragDrop
18103  * @extends Roo.util.Observable
18104  * Defines the interface and base operation of items that that can be
18105  * dragged or can be drop targets.  It was designed to be extended, overriding
18106  * the event handlers for startDrag, onDrag, onDragOver and onDragOut.
18107  * Up to three html elements can be associated with a DragDrop instance:
18108  * <ul>
18109  * <li>linked element: the element that is passed into the constructor.
18110  * This is the element which defines the boundaries for interaction with
18111  * other DragDrop objects.</li>
18112  * <li>handle element(s): The drag operation only occurs if the element that
18113  * was clicked matches a handle element.  By default this is the linked
18114  * element, but there are times that you will want only a portion of the
18115  * linked element to initiate the drag operation, and the setHandleElId()
18116  * method provides a way to define this.</li>
18117  * <li>drag element: this represents the element that would be moved along
18118  * with the cursor during a drag operation.  By default, this is the linked
18119  * element itself as in {@link Roo.dd.DD}.  setDragElId() lets you define
18120  * a separate element that would be moved, as in {@link Roo.dd.DDProxy}.
18121  * </li>
18122  * </ul>
18123  * This class should not be instantiated until the onload event to ensure that
18124  * the associated elements are available.
18125  * The following would define a DragDrop obj that would interact with any
18126  * other DragDrop obj in the "group1" group:
18127  * <pre>
18128  *  dd = new Roo.dd.DragDrop("div1", "group1");
18129  * </pre>
18130  * Since none of the event handlers have been implemented, nothing would
18131  * actually happen if you were to run the code above.  Normally you would
18132  * override this class or one of the default implementations, but you can
18133  * also override the methods you want on an instance of the class...
18134  * <pre>
18135  *  dd.onDragDrop = function(e, id) {
18136  *  &nbsp;&nbsp;alert("dd was dropped on " + id);
18137  *  }
18138  * </pre>
18139  * @constructor
18140  * @param {String} id of the element that is linked to this instance
18141  * @param {String} sGroup the group of related DragDrop objects
18142  * @param {object} config an object containing configurable attributes
18143  *                Valid properties for DragDrop:
18144  *                    padding, isTarget, maintainOffset, primaryButtonOnly
18145  */
18146 Roo.dd.DragDrop = function(id, sGroup, config) {
18147     if (id) {
18148         this.init(id, sGroup, config);
18149     }
18150     
18151 };
18152
18153 Roo.extend(Roo.dd.DragDrop, Roo.util.Observable , {
18154
18155     /**
18156      * The id of the element associated with this object.  This is what we
18157      * refer to as the "linked element" because the size and position of
18158      * this element is used to determine when the drag and drop objects have
18159      * interacted.
18160      * @property id
18161      * @type String
18162      */
18163     id: null,
18164
18165     /**
18166      * Configuration attributes passed into the constructor
18167      * @property config
18168      * @type object
18169      */
18170     config: null,
18171
18172     /**
18173      * The id of the element that will be dragged.  By default this is same
18174      * as the linked element , but could be changed to another element. Ex:
18175      * Roo.dd.DDProxy
18176      * @property dragElId
18177      * @type String
18178      * @private
18179      */
18180     dragElId: null,
18181
18182     /**
18183      * the id of the element that initiates the drag operation.  By default
18184      * this is the linked element, but could be changed to be a child of this
18185      * element.  This lets us do things like only starting the drag when the
18186      * header element within the linked html element is clicked.
18187      * @property handleElId
18188      * @type String
18189      * @private
18190      */
18191     handleElId: null,
18192
18193     /**
18194      * An associative array of HTML tags that will be ignored if clicked.
18195      * @property invalidHandleTypes
18196      * @type {string: string}
18197      */
18198     invalidHandleTypes: null,
18199
18200     /**
18201      * An associative array of ids for elements that will be ignored if clicked
18202      * @property invalidHandleIds
18203      * @type {string: string}
18204      */
18205     invalidHandleIds: null,
18206
18207     /**
18208      * An indexted array of css class names for elements that will be ignored
18209      * if clicked.
18210      * @property invalidHandleClasses
18211      * @type string[]
18212      */
18213     invalidHandleClasses: null,
18214
18215     /**
18216      * The linked element's absolute X position at the time the drag was
18217      * started
18218      * @property startPageX
18219      * @type int
18220      * @private
18221      */
18222     startPageX: 0,
18223
18224     /**
18225      * The linked element's absolute X position at the time the drag was
18226      * started
18227      * @property startPageY
18228      * @type int
18229      * @private
18230      */
18231     startPageY: 0,
18232
18233     /**
18234      * The group defines a logical collection of DragDrop objects that are
18235      * related.  Instances only get events when interacting with other
18236      * DragDrop object in the same group.  This lets us define multiple
18237      * groups using a single DragDrop subclass if we want.
18238      * @property groups
18239      * @type {string: string}
18240      */
18241     groups: null,
18242
18243     /**
18244      * Individual drag/drop instances can be locked.  This will prevent
18245      * onmousedown start drag.
18246      * @property locked
18247      * @type boolean
18248      * @private
18249      */
18250     locked: false,
18251
18252     /**
18253      * Lock this instance
18254      * @method lock
18255      */
18256     lock: function() { this.locked = true; },
18257
18258     /**
18259      * Unlock this instace
18260      * @method unlock
18261      */
18262     unlock: function() { this.locked = false; },
18263
18264     /**
18265      * By default, all insances can be a drop target.  This can be disabled by
18266      * setting isTarget to false.
18267      * @method isTarget
18268      * @type boolean
18269      */
18270     isTarget: true,
18271
18272     /**
18273      * The padding configured for this drag and drop object for calculating
18274      * the drop zone intersection with this object.
18275      * @method padding
18276      * @type int[]
18277      */
18278     padding: null,
18279
18280     /**
18281      * Cached reference to the linked element
18282      * @property _domRef
18283      * @private
18284      */
18285     _domRef: null,
18286
18287     /**
18288      * Internal typeof flag
18289      * @property __ygDragDrop
18290      * @private
18291      */
18292     __ygDragDrop: true,
18293
18294     /**
18295      * Set to true when horizontal contraints are applied
18296      * @property constrainX
18297      * @type boolean
18298      * @private
18299      */
18300     constrainX: false,
18301
18302     /**
18303      * Set to true when vertical contraints are applied
18304      * @property constrainY
18305      * @type boolean
18306      * @private
18307      */
18308     constrainY: false,
18309
18310     /**
18311      * The left constraint
18312      * @property minX
18313      * @type int
18314      * @private
18315      */
18316     minX: 0,
18317
18318     /**
18319      * The right constraint
18320      * @property maxX
18321      * @type int
18322      * @private
18323      */
18324     maxX: 0,
18325
18326     /**
18327      * The up constraint
18328      * @property minY
18329      * @type int
18330      * @type int
18331      * @private
18332      */
18333     minY: 0,
18334
18335     /**
18336      * The down constraint
18337      * @property maxY
18338      * @type int
18339      * @private
18340      */
18341     maxY: 0,
18342
18343     /**
18344      * Maintain offsets when we resetconstraints.  Set to true when you want
18345      * the position of the element relative to its parent to stay the same
18346      * when the page changes
18347      *
18348      * @property maintainOffset
18349      * @type boolean
18350      */
18351     maintainOffset: false,
18352
18353     /**
18354      * Array of pixel locations the element will snap to if we specified a
18355      * horizontal graduation/interval.  This array is generated automatically
18356      * when you define a tick interval.
18357      * @property xTicks
18358      * @type int[]
18359      */
18360     xTicks: null,
18361
18362     /**
18363      * Array of pixel locations the element will snap to if we specified a
18364      * vertical graduation/interval.  This array is generated automatically
18365      * when you define a tick interval.
18366      * @property yTicks
18367      * @type int[]
18368      */
18369     yTicks: null,
18370
18371     /**
18372      * By default the drag and drop instance will only respond to the primary
18373      * button click (left button for a right-handed mouse).  Set to true to
18374      * allow drag and drop to start with any mouse click that is propogated
18375      * by the browser
18376      * @property primaryButtonOnly
18377      * @type boolean
18378      */
18379     primaryButtonOnly: true,
18380
18381     /**
18382      * The availabe property is false until the linked dom element is accessible.
18383      * @property available
18384      * @type boolean
18385      */
18386     available: false,
18387
18388     /**
18389      * By default, drags can only be initiated if the mousedown occurs in the
18390      * region the linked element is.  This is done in part to work around a
18391      * bug in some browsers that mis-report the mousedown if the previous
18392      * mouseup happened outside of the window.  This property is set to true
18393      * if outer handles are defined.
18394      *
18395      * @property hasOuterHandles
18396      * @type boolean
18397      * @default false
18398      */
18399     hasOuterHandles: false,
18400
18401     /**
18402      * Code that executes immediately before the startDrag event
18403      * @method b4StartDrag
18404      * @private
18405      */
18406     b4StartDrag: function(x, y) { },
18407
18408     /**
18409      * Abstract method called after a drag/drop object is clicked
18410      * and the drag or mousedown time thresholds have beeen met.
18411      * @method startDrag
18412      * @param {int} X click location
18413      * @param {int} Y click location
18414      */
18415     startDrag: function(x, y) { /* override this */ },
18416
18417     /**
18418      * Code that executes immediately before the onDrag event
18419      * @method b4Drag
18420      * @private
18421      */
18422     b4Drag: function(e) { },
18423
18424     /**
18425      * Abstract method called during the onMouseMove event while dragging an
18426      * object.
18427      * @method onDrag
18428      * @param {Event} e the mousemove event
18429      */
18430     onDrag: function(e) { /* override this */ },
18431
18432     /**
18433      * Abstract method called when this element fist begins hovering over
18434      * another DragDrop obj
18435      * @method onDragEnter
18436      * @param {Event} e the mousemove event
18437      * @param {String|DragDrop[]} id In POINT mode, the element
18438      * id this is hovering over.  In INTERSECT mode, an array of one or more
18439      * dragdrop items being hovered over.
18440      */
18441     onDragEnter: function(e, id) { /* override this */ },
18442
18443     /**
18444      * Code that executes immediately before the onDragOver event
18445      * @method b4DragOver
18446      * @private
18447      */
18448     b4DragOver: function(e) { },
18449
18450     /**
18451      * Abstract method called when this element is hovering over another
18452      * DragDrop obj
18453      * @method onDragOver
18454      * @param {Event} e the mousemove event
18455      * @param {String|DragDrop[]} id In POINT mode, the element
18456      * id this is hovering over.  In INTERSECT mode, an array of dd items
18457      * being hovered over.
18458      */
18459     onDragOver: function(e, id) { /* override this */ },
18460
18461     /**
18462      * Code that executes immediately before the onDragOut event
18463      * @method b4DragOut
18464      * @private
18465      */
18466     b4DragOut: function(e) { },
18467
18468     /**
18469      * Abstract method called when we are no longer hovering over an element
18470      * @method onDragOut
18471      * @param {Event} e the mousemove event
18472      * @param {String|DragDrop[]} id In POINT mode, the element
18473      * id this was hovering over.  In INTERSECT mode, an array of dd items
18474      * that the mouse is no longer over.
18475      */
18476     onDragOut: function(e, id) { /* override this */ },
18477
18478     /**
18479      * Code that executes immediately before the onDragDrop event
18480      * @method b4DragDrop
18481      * @private
18482      */
18483     b4DragDrop: function(e) { },
18484
18485     /**
18486      * Abstract method called when this item is dropped on another DragDrop
18487      * obj
18488      * @method onDragDrop
18489      * @param {Event} e the mouseup event
18490      * @param {String|DragDrop[]} id In POINT mode, the element
18491      * id this was dropped on.  In INTERSECT mode, an array of dd items this
18492      * was dropped on.
18493      */
18494     onDragDrop: function(e, id) { /* override this */ },
18495
18496     /**
18497      * Abstract method called when this item is dropped on an area with no
18498      * drop target
18499      * @method onInvalidDrop
18500      * @param {Event} e the mouseup event
18501      */
18502     onInvalidDrop: function(e) { /* override this */ },
18503
18504     /**
18505      * Code that executes immediately before the endDrag event
18506      * @method b4EndDrag
18507      * @private
18508      */
18509     b4EndDrag: function(e) { },
18510
18511     /**
18512      * Fired when we are done dragging the object
18513      * @method endDrag
18514      * @param {Event} e the mouseup event
18515      */
18516     endDrag: function(e) { /* override this */ },
18517
18518     /**
18519      * Code executed immediately before the onMouseDown event
18520      * @method b4MouseDown
18521      * @param {Event} e the mousedown event
18522      * @private
18523      */
18524     b4MouseDown: function(e) {  },
18525
18526     /**
18527      * Event handler that fires when a drag/drop obj gets a mousedown
18528      * @method onMouseDown
18529      * @param {Event} e the mousedown event
18530      */
18531     onMouseDown: function(e) { /* override this */ },
18532
18533     /**
18534      * Event handler that fires when a drag/drop obj gets a mouseup
18535      * @method onMouseUp
18536      * @param {Event} e the mouseup event
18537      */
18538     onMouseUp: function(e) { /* override this */ },
18539
18540     /**
18541      * Override the onAvailable method to do what is needed after the initial
18542      * position was determined.
18543      * @method onAvailable
18544      */
18545     onAvailable: function () {
18546     },
18547
18548     /*
18549      * Provides default constraint padding to "constrainTo" elements (defaults to {left: 0, right:0, top:0, bottom:0}).
18550      * @type Object
18551      */
18552     defaultPadding : {left:0, right:0, top:0, bottom:0},
18553
18554     /*
18555      * Initializes the drag drop object's constraints to restrict movement to a certain element.
18556  *
18557  * Usage:
18558  <pre><code>
18559  var dd = new Roo.dd.DDProxy("dragDiv1", "proxytest",
18560                 { dragElId: "existingProxyDiv" });
18561  dd.startDrag = function(){
18562      this.constrainTo("parent-id");
18563  };
18564  </code></pre>
18565  * Or you can initalize it using the {@link Roo.Element} object:
18566  <pre><code>
18567  Roo.get("dragDiv1").initDDProxy("proxytest", {dragElId: "existingProxyDiv"}, {
18568      startDrag : function(){
18569          this.constrainTo("parent-id");
18570      }
18571  });
18572  </code></pre>
18573      * @param {String/HTMLElement/Element} constrainTo The element to constrain to.
18574      * @param {Object/Number} pad (optional) Pad provides a way to specify "padding" of the constraints,
18575      * and can be either a number for symmetrical padding (4 would be equal to {left:4, right:4, top:4, bottom:4}) or
18576      * an object containing the sides to pad. For example: {right:10, bottom:10}
18577      * @param {Boolean} inContent (optional) Constrain the draggable in the content box of the element (inside padding and borders)
18578      */
18579     constrainTo : function(constrainTo, pad, inContent){
18580         if(typeof pad == "number"){
18581             pad = {left: pad, right:pad, top:pad, bottom:pad};
18582         }
18583         pad = pad || this.defaultPadding;
18584         var b = Roo.get(this.getEl()).getBox();
18585         var ce = Roo.get(constrainTo);
18586         var s = ce.getScroll();
18587         var c, cd = ce.dom;
18588         if(cd == document.body){
18589             c = { x: s.left, y: s.top, width: Roo.lib.Dom.getViewWidth(), height: Roo.lib.Dom.getViewHeight()};
18590         }else{
18591             xy = ce.getXY();
18592             c = {x : xy[0]+s.left, y: xy[1]+s.top, width: cd.clientWidth, height: cd.clientHeight};
18593         }
18594
18595
18596         var topSpace = b.y - c.y;
18597         var leftSpace = b.x - c.x;
18598
18599         this.resetConstraints();
18600         this.setXConstraint(leftSpace - (pad.left||0), // left
18601                 c.width - leftSpace - b.width - (pad.right||0) //right
18602         );
18603         this.setYConstraint(topSpace - (pad.top||0), //top
18604                 c.height - topSpace - b.height - (pad.bottom||0) //bottom
18605         );
18606     },
18607
18608     /**
18609      * Returns a reference to the linked element
18610      * @method getEl
18611      * @return {HTMLElement} the html element
18612      */
18613     getEl: function() {
18614         if (!this._domRef) {
18615             this._domRef = Roo.getDom(this.id);
18616         }
18617
18618         return this._domRef;
18619     },
18620
18621     /**
18622      * Returns a reference to the actual element to drag.  By default this is
18623      * the same as the html element, but it can be assigned to another
18624      * element. An example of this can be found in Roo.dd.DDProxy
18625      * @method getDragEl
18626      * @return {HTMLElement} the html element
18627      */
18628     getDragEl: function() {
18629         return Roo.getDom(this.dragElId);
18630     },
18631
18632     /**
18633      * Sets up the DragDrop object.  Must be called in the constructor of any
18634      * Roo.dd.DragDrop subclass
18635      * @method init
18636      * @param id the id of the linked element
18637      * @param {String} sGroup the group of related items
18638      * @param {object} config configuration attributes
18639      */
18640     init: function(id, sGroup, config) {
18641         this.initTarget(id, sGroup, config);
18642         if (!Roo.isTouch) {
18643             Event.on(this.id, "mousedown", this.handleMouseDown, this);
18644         }
18645         Event.on(this.id, "touchstart", this.handleMouseDown, this);
18646         // Event.on(this.id, "selectstart", Event.preventDefault);
18647     },
18648
18649     /**
18650      * Initializes Targeting functionality only... the object does not
18651      * get a mousedown handler.
18652      * @method initTarget
18653      * @param id the id of the linked element
18654      * @param {String} sGroup the group of related items
18655      * @param {object} config configuration attributes
18656      */
18657     initTarget: function(id, sGroup, config) {
18658
18659         // configuration attributes
18660         this.config = config || {};
18661
18662         // create a local reference to the drag and drop manager
18663         this.DDM = Roo.dd.DDM;
18664         // initialize the groups array
18665         this.groups = {};
18666
18667         // assume that we have an element reference instead of an id if the
18668         // parameter is not a string
18669         if (typeof id !== "string") {
18670             id = Roo.id(id);
18671         }
18672
18673         // set the id
18674         this.id = id;
18675
18676         // add to an interaction group
18677         this.addToGroup((sGroup) ? sGroup : "default");
18678
18679         // We don't want to register this as the handle with the manager
18680         // so we just set the id rather than calling the setter.
18681         this.handleElId = id;
18682
18683         // the linked element is the element that gets dragged by default
18684         this.setDragElId(id);
18685
18686         // by default, clicked anchors will not start drag operations.
18687         this.invalidHandleTypes = { A: "A" };
18688         this.invalidHandleIds = {};
18689         this.invalidHandleClasses = [];
18690
18691         this.applyConfig();
18692
18693         this.handleOnAvailable();
18694     },
18695
18696     /**
18697      * Applies the configuration parameters that were passed into the constructor.
18698      * This is supposed to happen at each level through the inheritance chain.  So
18699      * a DDProxy implentation will execute apply config on DDProxy, DD, and
18700      * DragDrop in order to get all of the parameters that are available in
18701      * each object.
18702      * @method applyConfig
18703      */
18704     applyConfig: function() {
18705
18706         // configurable properties:
18707         //    padding, isTarget, maintainOffset, primaryButtonOnly
18708         this.padding           = this.config.padding || [0, 0, 0, 0];
18709         this.isTarget          = (this.config.isTarget !== false);
18710         this.maintainOffset    = (this.config.maintainOffset);
18711         this.primaryButtonOnly = (this.config.primaryButtonOnly !== false);
18712
18713     },
18714
18715     /**
18716      * Executed when the linked element is available
18717      * @method handleOnAvailable
18718      * @private
18719      */
18720     handleOnAvailable: function() {
18721         this.available = true;
18722         this.resetConstraints();
18723         this.onAvailable();
18724     },
18725
18726      /**
18727      * Configures the padding for the target zone in px.  Effectively expands
18728      * (or reduces) the virtual object size for targeting calculations.
18729      * Supports css-style shorthand; if only one parameter is passed, all sides
18730      * will have that padding, and if only two are passed, the top and bottom
18731      * will have the first param, the left and right the second.
18732      * @method setPadding
18733      * @param {int} iTop    Top pad
18734      * @param {int} iRight  Right pad
18735      * @param {int} iBot    Bot pad
18736      * @param {int} iLeft   Left pad
18737      */
18738     setPadding: function(iTop, iRight, iBot, iLeft) {
18739         // this.padding = [iLeft, iRight, iTop, iBot];
18740         if (!iRight && 0 !== iRight) {
18741             this.padding = [iTop, iTop, iTop, iTop];
18742         } else if (!iBot && 0 !== iBot) {
18743             this.padding = [iTop, iRight, iTop, iRight];
18744         } else {
18745             this.padding = [iTop, iRight, iBot, iLeft];
18746         }
18747     },
18748
18749     /**
18750      * Stores the initial placement of the linked element.
18751      * @method setInitialPosition
18752      * @param {int} diffX   the X offset, default 0
18753      * @param {int} diffY   the Y offset, default 0
18754      */
18755     setInitPosition: function(diffX, diffY) {
18756         var el = this.getEl();
18757
18758         if (!this.DDM.verifyEl(el)) {
18759             return;
18760         }
18761
18762         var dx = diffX || 0;
18763         var dy = diffY || 0;
18764
18765         var p = Dom.getXY( el );
18766
18767         this.initPageX = p[0] - dx;
18768         this.initPageY = p[1] - dy;
18769
18770         this.lastPageX = p[0];
18771         this.lastPageY = p[1];
18772
18773
18774         this.setStartPosition(p);
18775     },
18776
18777     /**
18778      * Sets the start position of the element.  This is set when the obj
18779      * is initialized, the reset when a drag is started.
18780      * @method setStartPosition
18781      * @param pos current position (from previous lookup)
18782      * @private
18783      */
18784     setStartPosition: function(pos) {
18785         var p = pos || Dom.getXY( this.getEl() );
18786         this.deltaSetXY = null;
18787
18788         this.startPageX = p[0];
18789         this.startPageY = p[1];
18790     },
18791
18792     /**
18793      * Add this instance to a group of related drag/drop objects.  All
18794      * instances belong to at least one group, and can belong to as many
18795      * groups as needed.
18796      * @method addToGroup
18797      * @param sGroup {string} the name of the group
18798      */
18799     addToGroup: function(sGroup) {
18800         this.groups[sGroup] = true;
18801         this.DDM.regDragDrop(this, sGroup);
18802     },
18803
18804     /**
18805      * Remove's this instance from the supplied interaction group
18806      * @method removeFromGroup
18807      * @param {string}  sGroup  The group to drop
18808      */
18809     removeFromGroup: function(sGroup) {
18810         if (this.groups[sGroup]) {
18811             delete this.groups[sGroup];
18812         }
18813
18814         this.DDM.removeDDFromGroup(this, sGroup);
18815     },
18816
18817     /**
18818      * Allows you to specify that an element other than the linked element
18819      * will be moved with the cursor during a drag
18820      * @method setDragElId
18821      * @param id {string} the id of the element that will be used to initiate the drag
18822      */
18823     setDragElId: function(id) {
18824         this.dragElId = id;
18825     },
18826
18827     /**
18828      * Allows you to specify a child of the linked element that should be
18829      * used to initiate the drag operation.  An example of this would be if
18830      * you have a content div with text and links.  Clicking anywhere in the
18831      * content area would normally start the drag operation.  Use this method
18832      * to specify that an element inside of the content div is the element
18833      * that starts the drag operation.
18834      * @method setHandleElId
18835      * @param id {string} the id of the element that will be used to
18836      * initiate the drag.
18837      */
18838     setHandleElId: function(id) {
18839         if (typeof id !== "string") {
18840             id = Roo.id(id);
18841         }
18842         this.handleElId = id;
18843         this.DDM.regHandle(this.id, id);
18844     },
18845
18846     /**
18847      * Allows you to set an element outside of the linked element as a drag
18848      * handle
18849      * @method setOuterHandleElId
18850      * @param id the id of the element that will be used to initiate the drag
18851      */
18852     setOuterHandleElId: function(id) {
18853         if (typeof id !== "string") {
18854             id = Roo.id(id);
18855         }
18856         Event.on(id, "mousedown",
18857                 this.handleMouseDown, this);
18858         this.setHandleElId(id);
18859
18860         this.hasOuterHandles = true;
18861     },
18862
18863     /**
18864      * Remove all drag and drop hooks for this element
18865      * @method unreg
18866      */
18867     unreg: function() {
18868         Event.un(this.id, "mousedown",
18869                 this.handleMouseDown);
18870         Event.un(this.id, "touchstart",
18871                 this.handleMouseDown);
18872         this._domRef = null;
18873         this.DDM._remove(this);
18874     },
18875
18876     destroy : function(){
18877         this.unreg();
18878     },
18879
18880     /**
18881      * Returns true if this instance is locked, or the drag drop mgr is locked
18882      * (meaning that all drag/drop is disabled on the page.)
18883      * @method isLocked
18884      * @return {boolean} true if this obj or all drag/drop is locked, else
18885      * false
18886      */
18887     isLocked: function() {
18888         return (this.DDM.isLocked() || this.locked);
18889     },
18890
18891     /**
18892      * Fired when this object is clicked
18893      * @method handleMouseDown
18894      * @param {Event} e
18895      * @param {Roo.dd.DragDrop} oDD the clicked dd object (this dd obj)
18896      * @private
18897      */
18898     handleMouseDown: function(e, oDD){
18899      
18900         if (!Roo.isTouch && this.primaryButtonOnly && e.button != 0) {
18901             //Roo.log('not touch/ button !=0');
18902             return;
18903         }
18904         if (e.browserEvent.touches && e.browserEvent.touches.length != 1) {
18905             return; // double touch..
18906         }
18907         
18908
18909         if (this.isLocked()) {
18910             //Roo.log('locked');
18911             return;
18912         }
18913
18914         this.DDM.refreshCache(this.groups);
18915 //        Roo.log([Roo.lib.Event.getPageX(e), Roo.lib.Event.getPageY(e)]);
18916         var pt = new Roo.lib.Point(Roo.lib.Event.getPageX(e), Roo.lib.Event.getPageY(e));
18917         if (!this.hasOuterHandles && !this.DDM.isOverTarget(pt, this) )  {
18918             //Roo.log('no outer handes or not over target');
18919                 // do nothing.
18920         } else {
18921 //            Roo.log('check validator');
18922             if (this.clickValidator(e)) {
18923 //                Roo.log('validate success');
18924                 // set the initial element position
18925                 this.setStartPosition();
18926
18927
18928                 this.b4MouseDown(e);
18929                 this.onMouseDown(e);
18930
18931                 this.DDM.handleMouseDown(e, this);
18932
18933                 this.DDM.stopEvent(e);
18934             } else {
18935
18936
18937             }
18938         }
18939     },
18940
18941     clickValidator: function(e) {
18942         var target = e.getTarget();
18943         return ( this.isValidHandleChild(target) &&
18944                     (this.id == this.handleElId ||
18945                         this.DDM.handleWasClicked(target, this.id)) );
18946     },
18947
18948     /**
18949      * Allows you to specify a tag name that should not start a drag operation
18950      * when clicked.  This is designed to facilitate embedding links within a
18951      * drag handle that do something other than start the drag.
18952      * @method addInvalidHandleType
18953      * @param {string} tagName the type of element to exclude
18954      */
18955     addInvalidHandleType: function(tagName) {
18956         var type = tagName.toUpperCase();
18957         this.invalidHandleTypes[type] = type;
18958     },
18959
18960     /**
18961      * Lets you to specify an element id for a child of a drag handle
18962      * that should not initiate a drag
18963      * @method addInvalidHandleId
18964      * @param {string} id the element id of the element you wish to ignore
18965      */
18966     addInvalidHandleId: function(id) {
18967         if (typeof id !== "string") {
18968             id = Roo.id(id);
18969         }
18970         this.invalidHandleIds[id] = id;
18971     },
18972
18973     /**
18974      * Lets you specify a css class of elements that will not initiate a drag
18975      * @method addInvalidHandleClass
18976      * @param {string} cssClass the class of the elements you wish to ignore
18977      */
18978     addInvalidHandleClass: function(cssClass) {
18979         this.invalidHandleClasses.push(cssClass);
18980     },
18981
18982     /**
18983      * Unsets an excluded tag name set by addInvalidHandleType
18984      * @method removeInvalidHandleType
18985      * @param {string} tagName the type of element to unexclude
18986      */
18987     removeInvalidHandleType: function(tagName) {
18988         var type = tagName.toUpperCase();
18989         // this.invalidHandleTypes[type] = null;
18990         delete this.invalidHandleTypes[type];
18991     },
18992
18993     /**
18994      * Unsets an invalid handle id
18995      * @method removeInvalidHandleId
18996      * @param {string} id the id of the element to re-enable
18997      */
18998     removeInvalidHandleId: function(id) {
18999         if (typeof id !== "string") {
19000             id = Roo.id(id);
19001         }
19002         delete this.invalidHandleIds[id];
19003     },
19004
19005     /**
19006      * Unsets an invalid css class
19007      * @method removeInvalidHandleClass
19008      * @param {string} cssClass the class of the element(s) you wish to
19009      * re-enable
19010      */
19011     removeInvalidHandleClass: function(cssClass) {
19012         for (var i=0, len=this.invalidHandleClasses.length; i<len; ++i) {
19013             if (this.invalidHandleClasses[i] == cssClass) {
19014                 delete this.invalidHandleClasses[i];
19015             }
19016         }
19017     },
19018
19019     /**
19020      * Checks the tag exclusion list to see if this click should be ignored
19021      * @method isValidHandleChild
19022      * @param {HTMLElement} node the HTMLElement to evaluate
19023      * @return {boolean} true if this is a valid tag type, false if not
19024      */
19025     isValidHandleChild: function(node) {
19026
19027         var valid = true;
19028         // var n = (node.nodeName == "#text") ? node.parentNode : node;
19029         var nodeName;
19030         try {
19031             nodeName = node.nodeName.toUpperCase();
19032         } catch(e) {
19033             nodeName = node.nodeName;
19034         }
19035         valid = valid && !this.invalidHandleTypes[nodeName];
19036         valid = valid && !this.invalidHandleIds[node.id];
19037
19038         for (var i=0, len=this.invalidHandleClasses.length; valid && i<len; ++i) {
19039             valid = !Dom.hasClass(node, this.invalidHandleClasses[i]);
19040         }
19041
19042
19043         return valid;
19044
19045     },
19046
19047     /**
19048      * Create the array of horizontal tick marks if an interval was specified
19049      * in setXConstraint().
19050      * @method setXTicks
19051      * @private
19052      */
19053     setXTicks: function(iStartX, iTickSize) {
19054         this.xTicks = [];
19055         this.xTickSize = iTickSize;
19056
19057         var tickMap = {};
19058
19059         for (var i = this.initPageX; i >= this.minX; i = i - iTickSize) {
19060             if (!tickMap[i]) {
19061                 this.xTicks[this.xTicks.length] = i;
19062                 tickMap[i] = true;
19063             }
19064         }
19065
19066         for (i = this.initPageX; i <= this.maxX; i = i + iTickSize) {
19067             if (!tickMap[i]) {
19068                 this.xTicks[this.xTicks.length] = i;
19069                 tickMap[i] = true;
19070             }
19071         }
19072
19073         this.xTicks.sort(this.DDM.numericSort) ;
19074     },
19075
19076     /**
19077      * Create the array of vertical tick marks if an interval was specified in
19078      * setYConstraint().
19079      * @method setYTicks
19080      * @private
19081      */
19082     setYTicks: function(iStartY, iTickSize) {
19083         this.yTicks = [];
19084         this.yTickSize = iTickSize;
19085
19086         var tickMap = {};
19087
19088         for (var i = this.initPageY; i >= this.minY; i = i - iTickSize) {
19089             if (!tickMap[i]) {
19090                 this.yTicks[this.yTicks.length] = i;
19091                 tickMap[i] = true;
19092             }
19093         }
19094
19095         for (i = this.initPageY; i <= this.maxY; i = i + iTickSize) {
19096             if (!tickMap[i]) {
19097                 this.yTicks[this.yTicks.length] = i;
19098                 tickMap[i] = true;
19099             }
19100         }
19101
19102         this.yTicks.sort(this.DDM.numericSort) ;
19103     },
19104
19105     /**
19106      * By default, the element can be dragged any place on the screen.  Use
19107      * this method to limit the horizontal travel of the element.  Pass in
19108      * 0,0 for the parameters if you want to lock the drag to the y axis.
19109      * @method setXConstraint
19110      * @param {int} iLeft the number of pixels the element can move to the left
19111      * @param {int} iRight the number of pixels the element can move to the
19112      * right
19113      * @param {int} iTickSize optional parameter for specifying that the
19114      * element
19115      * should move iTickSize pixels at a time.
19116      */
19117     setXConstraint: function(iLeft, iRight, iTickSize) {
19118         this.leftConstraint = iLeft;
19119         this.rightConstraint = iRight;
19120
19121         this.minX = this.initPageX - iLeft;
19122         this.maxX = this.initPageX + iRight;
19123         if (iTickSize) { this.setXTicks(this.initPageX, iTickSize); }
19124
19125         this.constrainX = true;
19126     },
19127
19128     /**
19129      * Clears any constraints applied to this instance.  Also clears ticks
19130      * since they can't exist independent of a constraint at this time.
19131      * @method clearConstraints
19132      */
19133     clearConstraints: function() {
19134         this.constrainX = false;
19135         this.constrainY = false;
19136         this.clearTicks();
19137     },
19138
19139     /**
19140      * Clears any tick interval defined for this instance
19141      * @method clearTicks
19142      */
19143     clearTicks: function() {
19144         this.xTicks = null;
19145         this.yTicks = null;
19146         this.xTickSize = 0;
19147         this.yTickSize = 0;
19148     },
19149
19150     /**
19151      * By default, the element can be dragged any place on the screen.  Set
19152      * this to limit the vertical travel of the element.  Pass in 0,0 for the
19153      * parameters if you want to lock the drag to the x axis.
19154      * @method setYConstraint
19155      * @param {int} iUp the number of pixels the element can move up
19156      * @param {int} iDown the number of pixels the element can move down
19157      * @param {int} iTickSize optional parameter for specifying that the
19158      * element should move iTickSize pixels at a time.
19159      */
19160     setYConstraint: function(iUp, iDown, iTickSize) {
19161         this.topConstraint = iUp;
19162         this.bottomConstraint = iDown;
19163
19164         this.minY = this.initPageY - iUp;
19165         this.maxY = this.initPageY + iDown;
19166         if (iTickSize) { this.setYTicks(this.initPageY, iTickSize); }
19167
19168         this.constrainY = true;
19169
19170     },
19171
19172     /**
19173      * resetConstraints must be called if you manually reposition a dd element.
19174      * @method resetConstraints
19175      * @param {boolean} maintainOffset
19176      */
19177     resetConstraints: function() {
19178
19179
19180         // Maintain offsets if necessary
19181         if (this.initPageX || this.initPageX === 0) {
19182             // figure out how much this thing has moved
19183             var dx = (this.maintainOffset) ? this.lastPageX - this.initPageX : 0;
19184             var dy = (this.maintainOffset) ? this.lastPageY - this.initPageY : 0;
19185
19186             this.setInitPosition(dx, dy);
19187
19188         // This is the first time we have detected the element's position
19189         } else {
19190             this.setInitPosition();
19191         }
19192
19193         if (this.constrainX) {
19194             this.setXConstraint( this.leftConstraint,
19195                                  this.rightConstraint,
19196                                  this.xTickSize        );
19197         }
19198
19199         if (this.constrainY) {
19200             this.setYConstraint( this.topConstraint,
19201                                  this.bottomConstraint,
19202                                  this.yTickSize         );
19203         }
19204     },
19205
19206     /**
19207      * Normally the drag element is moved pixel by pixel, but we can specify
19208      * that it move a number of pixels at a time.  This method resolves the
19209      * location when we have it set up like this.
19210      * @method getTick
19211      * @param {int} val where we want to place the object
19212      * @param {int[]} tickArray sorted array of valid points
19213      * @return {int} the closest tick
19214      * @private
19215      */
19216     getTick: function(val, tickArray) {
19217
19218         if (!tickArray) {
19219             // If tick interval is not defined, it is effectively 1 pixel,
19220             // so we return the value passed to us.
19221             return val;
19222         } else if (tickArray[0] >= val) {
19223             // The value is lower than the first tick, so we return the first
19224             // tick.
19225             return tickArray[0];
19226         } else {
19227             for (var i=0, len=tickArray.length; i<len; ++i) {
19228                 var next = i + 1;
19229                 if (tickArray[next] && tickArray[next] >= val) {
19230                     var diff1 = val - tickArray[i];
19231                     var diff2 = tickArray[next] - val;
19232                     return (diff2 > diff1) ? tickArray[i] : tickArray[next];
19233                 }
19234             }
19235
19236             // The value is larger than the last tick, so we return the last
19237             // tick.
19238             return tickArray[tickArray.length - 1];
19239         }
19240     },
19241
19242     /**
19243      * toString method
19244      * @method toString
19245      * @return {string} string representation of the dd obj
19246      */
19247     toString: function() {
19248         return ("DragDrop " + this.id);
19249     }
19250
19251 });
19252
19253 })();
19254 /*
19255  * Based on:
19256  * Ext JS Library 1.1.1
19257  * Copyright(c) 2006-2007, Ext JS, LLC.
19258  *
19259  * Originally Released Under LGPL - original licence link has changed is not relivant.
19260  *
19261  * Fork - LGPL
19262  * <script type="text/javascript">
19263  */
19264
19265
19266 /**
19267  * The drag and drop utility provides a framework for building drag and drop
19268  * applications.  In addition to enabling drag and drop for specific elements,
19269  * the drag and drop elements are tracked by the manager class, and the
19270  * interactions between the various elements are tracked during the drag and
19271  * the implementing code is notified about these important moments.
19272  */
19273
19274 // Only load the library once.  Rewriting the manager class would orphan
19275 // existing drag and drop instances.
19276 if (!Roo.dd.DragDropMgr) {
19277
19278 /**
19279  * @class Roo.dd.DragDropMgr
19280  * DragDropMgr is a singleton that tracks the element interaction for
19281  * all DragDrop items in the window.  Generally, you will not call
19282  * this class directly, but it does have helper methods that could
19283  * be useful in your DragDrop implementations.
19284  * @singleton
19285  */
19286 Roo.dd.DragDropMgr = function() {
19287
19288     var Event = Roo.EventManager;
19289
19290     return {
19291
19292         /**
19293          * Two dimensional Array of registered DragDrop objects.  The first
19294          * dimension is the DragDrop item group, the second the DragDrop
19295          * object.
19296          * @property ids
19297          * @type {string: string}
19298          * @private
19299          * @static
19300          */
19301         ids: {},
19302
19303         /**
19304          * Array of element ids defined as drag handles.  Used to determine
19305          * if the element that generated the mousedown event is actually the
19306          * handle and not the html element itself.
19307          * @property handleIds
19308          * @type {string: string}
19309          * @private
19310          * @static
19311          */
19312         handleIds: {},
19313
19314         /**
19315          * the DragDrop object that is currently being dragged
19316          * @property dragCurrent
19317          * @type DragDrop
19318          * @private
19319          * @static
19320          **/
19321         dragCurrent: null,
19322
19323         /**
19324          * the DragDrop object(s) that are being hovered over
19325          * @property dragOvers
19326          * @type Array
19327          * @private
19328          * @static
19329          */
19330         dragOvers: {},
19331
19332         /**
19333          * the X distance between the cursor and the object being dragged
19334          * @property deltaX
19335          * @type int
19336          * @private
19337          * @static
19338          */
19339         deltaX: 0,
19340
19341         /**
19342          * the Y distance between the cursor and the object being dragged
19343          * @property deltaY
19344          * @type int
19345          * @private
19346          * @static
19347          */
19348         deltaY: 0,
19349
19350         /**
19351          * Flag to determine if we should prevent the default behavior of the
19352          * events we define. By default this is true, but this can be set to
19353          * false if you need the default behavior (not recommended)
19354          * @property preventDefault
19355          * @type boolean
19356          * @static
19357          */
19358         preventDefault: true,
19359
19360         /**
19361          * Flag to determine if we should stop the propagation of the events
19362          * we generate. This is true by default but you may want to set it to
19363          * false if the html element contains other features that require the
19364          * mouse click.
19365          * @property stopPropagation
19366          * @type boolean
19367          * @static
19368          */
19369         stopPropagation: true,
19370
19371         /**
19372          * Internal flag that is set to true when drag and drop has been
19373          * intialized
19374          * @property initialized
19375          * @private
19376          * @static
19377          */
19378         initalized: false,
19379
19380         /**
19381          * All drag and drop can be disabled.
19382          * @property locked
19383          * @private
19384          * @static
19385          */
19386         locked: false,
19387
19388         /**
19389          * Called the first time an element is registered.
19390          * @method init
19391          * @private
19392          * @static
19393          */
19394         init: function() {
19395             this.initialized = true;
19396         },
19397
19398         /**
19399          * In point mode, drag and drop interaction is defined by the
19400          * location of the cursor during the drag/drop
19401          * @property POINT
19402          * @type int
19403          * @static
19404          */
19405         POINT: 0,
19406
19407         /**
19408          * In intersect mode, drag and drop interactio nis defined by the
19409          * overlap of two or more drag and drop objects.
19410          * @property INTERSECT
19411          * @type int
19412          * @static
19413          */
19414         INTERSECT: 1,
19415
19416         /**
19417          * The current drag and drop mode.  Default: POINT
19418          * @property mode
19419          * @type int
19420          * @static
19421          */
19422         mode: 0,
19423
19424         /**
19425          * Runs method on all drag and drop objects
19426          * @method _execOnAll
19427          * @private
19428          * @static
19429          */
19430         _execOnAll: function(sMethod, args) {
19431             for (var i in this.ids) {
19432                 for (var j in this.ids[i]) {
19433                     var oDD = this.ids[i][j];
19434                     if (! this.isTypeOfDD(oDD)) {
19435                         continue;
19436                     }
19437                     oDD[sMethod].apply(oDD, args);
19438                 }
19439             }
19440         },
19441
19442         /**
19443          * Drag and drop initialization.  Sets up the global event handlers
19444          * @method _onLoad
19445          * @private
19446          * @static
19447          */
19448         _onLoad: function() {
19449
19450             this.init();
19451
19452             if (!Roo.isTouch) {
19453                 Event.on(document, "mouseup",   this.handleMouseUp, this, true);
19454                 Event.on(document, "mousemove", this.handleMouseMove, this, true);
19455             }
19456             Event.on(document, "touchend",   this.handleMouseUp, this, true);
19457             Event.on(document, "touchmove", this.handleMouseMove, this, true);
19458             
19459             Event.on(window,   "unload",    this._onUnload, this, true);
19460             Event.on(window,   "resize",    this._onResize, this, true);
19461             // Event.on(window,   "mouseout",    this._test);
19462
19463         },
19464
19465         /**
19466          * Reset constraints on all drag and drop objs
19467          * @method _onResize
19468          * @private
19469          * @static
19470          */
19471         _onResize: function(e) {
19472             this._execOnAll("resetConstraints", []);
19473         },
19474
19475         /**
19476          * Lock all drag and drop functionality
19477          * @method lock
19478          * @static
19479          */
19480         lock: function() { this.locked = true; },
19481
19482         /**
19483          * Unlock all drag and drop functionality
19484          * @method unlock
19485          * @static
19486          */
19487         unlock: function() { this.locked = false; },
19488
19489         /**
19490          * Is drag and drop locked?
19491          * @method isLocked
19492          * @return {boolean} True if drag and drop is locked, false otherwise.
19493          * @static
19494          */
19495         isLocked: function() { return this.locked; },
19496
19497         /**
19498          * Location cache that is set for all drag drop objects when a drag is
19499          * initiated, cleared when the drag is finished.
19500          * @property locationCache
19501          * @private
19502          * @static
19503          */
19504         locationCache: {},
19505
19506         /**
19507          * Set useCache to false if you want to force object the lookup of each
19508          * drag and drop linked element constantly during a drag.
19509          * @property useCache
19510          * @type boolean
19511          * @static
19512          */
19513         useCache: true,
19514
19515         /**
19516          * The number of pixels that the mouse needs to move after the
19517          * mousedown before the drag is initiated.  Default=3;
19518          * @property clickPixelThresh
19519          * @type int
19520          * @static
19521          */
19522         clickPixelThresh: 3,
19523
19524         /**
19525          * The number of milliseconds after the mousedown event to initiate the
19526          * drag if we don't get a mouseup event. Default=1000
19527          * @property clickTimeThresh
19528          * @type int
19529          * @static
19530          */
19531         clickTimeThresh: 350,
19532
19533         /**
19534          * Flag that indicates that either the drag pixel threshold or the
19535          * mousdown time threshold has been met
19536          * @property dragThreshMet
19537          * @type boolean
19538          * @private
19539          * @static
19540          */
19541         dragThreshMet: false,
19542
19543         /**
19544          * Timeout used for the click time threshold
19545          * @property clickTimeout
19546          * @type Object
19547          * @private
19548          * @static
19549          */
19550         clickTimeout: null,
19551
19552         /**
19553          * The X position of the mousedown event stored for later use when a
19554          * drag threshold is met.
19555          * @property startX
19556          * @type int
19557          * @private
19558          * @static
19559          */
19560         startX: 0,
19561
19562         /**
19563          * The Y position of the mousedown event stored for later use when a
19564          * drag threshold is met.
19565          * @property startY
19566          * @type int
19567          * @private
19568          * @static
19569          */
19570         startY: 0,
19571
19572         /**
19573          * Each DragDrop instance must be registered with the DragDropMgr.
19574          * This is executed in DragDrop.init()
19575          * @method regDragDrop
19576          * @param {DragDrop} oDD the DragDrop object to register
19577          * @param {String} sGroup the name of the group this element belongs to
19578          * @static
19579          */
19580         regDragDrop: function(oDD, sGroup) {
19581             if (!this.initialized) { this.init(); }
19582
19583             if (!this.ids[sGroup]) {
19584                 this.ids[sGroup] = {};
19585             }
19586             this.ids[sGroup][oDD.id] = oDD;
19587         },
19588
19589         /**
19590          * Removes the supplied dd instance from the supplied group. Executed
19591          * by DragDrop.removeFromGroup, so don't call this function directly.
19592          * @method removeDDFromGroup
19593          * @private
19594          * @static
19595          */
19596         removeDDFromGroup: function(oDD, sGroup) {
19597             if (!this.ids[sGroup]) {
19598                 this.ids[sGroup] = {};
19599             }
19600
19601             var obj = this.ids[sGroup];
19602             if (obj && obj[oDD.id]) {
19603                 delete obj[oDD.id];
19604             }
19605         },
19606
19607         /**
19608          * Unregisters a drag and drop item.  This is executed in
19609          * DragDrop.unreg, use that method instead of calling this directly.
19610          * @method _remove
19611          * @private
19612          * @static
19613          */
19614         _remove: function(oDD) {
19615             for (var g in oDD.groups) {
19616                 if (g && this.ids[g][oDD.id]) {
19617                     delete this.ids[g][oDD.id];
19618                 }
19619             }
19620             delete this.handleIds[oDD.id];
19621         },
19622
19623         /**
19624          * Each DragDrop handle element must be registered.  This is done
19625          * automatically when executing DragDrop.setHandleElId()
19626          * @method regHandle
19627          * @param {String} sDDId the DragDrop id this element is a handle for
19628          * @param {String} sHandleId the id of the element that is the drag
19629          * handle
19630          * @static
19631          */
19632         regHandle: function(sDDId, sHandleId) {
19633             if (!this.handleIds[sDDId]) {
19634                 this.handleIds[sDDId] = {};
19635             }
19636             this.handleIds[sDDId][sHandleId] = sHandleId;
19637         },
19638
19639         /**
19640          * Utility function to determine if a given element has been
19641          * registered as a drag drop item.
19642          * @method isDragDrop
19643          * @param {String} id the element id to check
19644          * @return {boolean} true if this element is a DragDrop item,
19645          * false otherwise
19646          * @static
19647          */
19648         isDragDrop: function(id) {
19649             return ( this.getDDById(id) ) ? true : false;
19650         },
19651
19652         /**
19653          * Returns the drag and drop instances that are in all groups the
19654          * passed in instance belongs to.
19655          * @method getRelated
19656          * @param {DragDrop} p_oDD the obj to get related data for
19657          * @param {boolean} bTargetsOnly if true, only return targetable objs
19658          * @return {DragDrop[]} the related instances
19659          * @static
19660          */
19661         getRelated: function(p_oDD, bTargetsOnly) {
19662             var oDDs = [];
19663             for (var i in p_oDD.groups) {
19664                 for (j in this.ids[i]) {
19665                     var dd = this.ids[i][j];
19666                     if (! this.isTypeOfDD(dd)) {
19667                         continue;
19668                     }
19669                     if (!bTargetsOnly || dd.isTarget) {
19670                         oDDs[oDDs.length] = dd;
19671                     }
19672                 }
19673             }
19674
19675             return oDDs;
19676         },
19677
19678         /**
19679          * Returns true if the specified dd target is a legal target for
19680          * the specifice drag obj
19681          * @method isLegalTarget
19682          * @param {DragDrop} the drag obj
19683          * @param {DragDrop} the target
19684          * @return {boolean} true if the target is a legal target for the
19685          * dd obj
19686          * @static
19687          */
19688         isLegalTarget: function (oDD, oTargetDD) {
19689             var targets = this.getRelated(oDD, true);
19690             for (var i=0, len=targets.length;i<len;++i) {
19691                 if (targets[i].id == oTargetDD.id) {
19692                     return true;
19693                 }
19694             }
19695
19696             return false;
19697         },
19698
19699         /**
19700          * My goal is to be able to transparently determine if an object is
19701          * typeof DragDrop, and the exact subclass of DragDrop.  typeof
19702          * returns "object", oDD.constructor.toString() always returns
19703          * "DragDrop" and not the name of the subclass.  So for now it just
19704          * evaluates a well-known variable in DragDrop.
19705          * @method isTypeOfDD
19706          * @param {Object} the object to evaluate
19707          * @return {boolean} true if typeof oDD = DragDrop
19708          * @static
19709          */
19710         isTypeOfDD: function (oDD) {
19711             return (oDD && oDD.__ygDragDrop);
19712         },
19713
19714         /**
19715          * Utility function to determine if a given element has been
19716          * registered as a drag drop handle for the given Drag Drop object.
19717          * @method isHandle
19718          * @param {String} id the element id to check
19719          * @return {boolean} true if this element is a DragDrop handle, false
19720          * otherwise
19721          * @static
19722          */
19723         isHandle: function(sDDId, sHandleId) {
19724             return ( this.handleIds[sDDId] &&
19725                             this.handleIds[sDDId][sHandleId] );
19726         },
19727
19728         /**
19729          * Returns the DragDrop instance for a given id
19730          * @method getDDById
19731          * @param {String} id the id of the DragDrop object
19732          * @return {DragDrop} the drag drop object, null if it is not found
19733          * @static
19734          */
19735         getDDById: function(id) {
19736             for (var i in this.ids) {
19737                 if (this.ids[i][id]) {
19738                     return this.ids[i][id];
19739                 }
19740             }
19741             return null;
19742         },
19743
19744         /**
19745          * Fired after a registered DragDrop object gets the mousedown event.
19746          * Sets up the events required to track the object being dragged
19747          * @method handleMouseDown
19748          * @param {Event} e the event
19749          * @param oDD the DragDrop object being dragged
19750          * @private
19751          * @static
19752          */
19753         handleMouseDown: function(e, oDD) {
19754             if(Roo.QuickTips){
19755                 Roo.QuickTips.disable();
19756             }
19757             this.currentTarget = e.getTarget();
19758
19759             this.dragCurrent = oDD;
19760
19761             var el = oDD.getEl();
19762
19763             // track start position
19764             this.startX = e.getPageX();
19765             this.startY = e.getPageY();
19766
19767             this.deltaX = this.startX - el.offsetLeft;
19768             this.deltaY = this.startY - el.offsetTop;
19769
19770             this.dragThreshMet = false;
19771
19772             this.clickTimeout = setTimeout(
19773                     function() {
19774                         var DDM = Roo.dd.DDM;
19775                         DDM.startDrag(DDM.startX, DDM.startY);
19776                     },
19777                     this.clickTimeThresh );
19778         },
19779
19780         /**
19781          * Fired when either the drag pixel threshol or the mousedown hold
19782          * time threshold has been met.
19783          * @method startDrag
19784          * @param x {int} the X position of the original mousedown
19785          * @param y {int} the Y position of the original mousedown
19786          * @static
19787          */
19788         startDrag: function(x, y) {
19789             clearTimeout(this.clickTimeout);
19790             if (this.dragCurrent) {
19791                 this.dragCurrent.b4StartDrag(x, y);
19792                 this.dragCurrent.startDrag(x, y);
19793             }
19794             this.dragThreshMet = true;
19795         },
19796
19797         /**
19798          * Internal function to handle the mouseup event.  Will be invoked
19799          * from the context of the document.
19800          * @method handleMouseUp
19801          * @param {Event} e the event
19802          * @private
19803          * @static
19804          */
19805         handleMouseUp: function(e) {
19806
19807             if(Roo.QuickTips){
19808                 Roo.QuickTips.enable();
19809             }
19810             if (! this.dragCurrent) {
19811                 return;
19812             }
19813
19814             clearTimeout(this.clickTimeout);
19815
19816             if (this.dragThreshMet) {
19817                 this.fireEvents(e, true);
19818             } else {
19819             }
19820
19821             this.stopDrag(e);
19822
19823             this.stopEvent(e);
19824         },
19825
19826         /**
19827          * Utility to stop event propagation and event default, if these
19828          * features are turned on.
19829          * @method stopEvent
19830          * @param {Event} e the event as returned by this.getEvent()
19831          * @static
19832          */
19833         stopEvent: function(e){
19834             if(this.stopPropagation) {
19835                 e.stopPropagation();
19836             }
19837
19838             if (this.preventDefault) {
19839                 e.preventDefault();
19840             }
19841         },
19842
19843         /**
19844          * Internal function to clean up event handlers after the drag
19845          * operation is complete
19846          * @method stopDrag
19847          * @param {Event} e the event
19848          * @private
19849          * @static
19850          */
19851         stopDrag: function(e) {
19852             // Fire the drag end event for the item that was dragged
19853             if (this.dragCurrent) {
19854                 if (this.dragThreshMet) {
19855                     this.dragCurrent.b4EndDrag(e);
19856                     this.dragCurrent.endDrag(e);
19857                 }
19858
19859                 this.dragCurrent.onMouseUp(e);
19860             }
19861
19862             this.dragCurrent = null;
19863             this.dragOvers = {};
19864         },
19865
19866         /**
19867          * Internal function to handle the mousemove event.  Will be invoked
19868          * from the context of the html element.
19869          *
19870          * @TODO figure out what we can do about mouse events lost when the
19871          * user drags objects beyond the window boundary.  Currently we can
19872          * detect this in internet explorer by verifying that the mouse is
19873          * down during the mousemove event.  Firefox doesn't give us the
19874          * button state on the mousemove event.
19875          * @method handleMouseMove
19876          * @param {Event} e the event
19877          * @private
19878          * @static
19879          */
19880         handleMouseMove: function(e) {
19881             if (! this.dragCurrent) {
19882                 return true;
19883             }
19884
19885             // var button = e.which || e.button;
19886
19887             // check for IE mouseup outside of page boundary
19888             if (Roo.isIE && (e.button !== 0 && e.button !== 1 && e.button !== 2)) {
19889                 this.stopEvent(e);
19890                 return this.handleMouseUp(e);
19891             }
19892
19893             if (!this.dragThreshMet) {
19894                 var diffX = Math.abs(this.startX - e.getPageX());
19895                 var diffY = Math.abs(this.startY - e.getPageY());
19896                 if (diffX > this.clickPixelThresh ||
19897                             diffY > this.clickPixelThresh) {
19898                     this.startDrag(this.startX, this.startY);
19899                 }
19900             }
19901
19902             if (this.dragThreshMet) {
19903                 this.dragCurrent.b4Drag(e);
19904                 this.dragCurrent.onDrag(e);
19905                 if(!this.dragCurrent.moveOnly){
19906                     this.fireEvents(e, false);
19907                 }
19908             }
19909
19910             this.stopEvent(e);
19911
19912             return true;
19913         },
19914
19915         /**
19916          * Iterates over all of the DragDrop elements to find ones we are
19917          * hovering over or dropping on
19918          * @method fireEvents
19919          * @param {Event} e the event
19920          * @param {boolean} isDrop is this a drop op or a mouseover op?
19921          * @private
19922          * @static
19923          */
19924         fireEvents: function(e, isDrop) {
19925             var dc = this.dragCurrent;
19926
19927             // If the user did the mouse up outside of the window, we could
19928             // get here even though we have ended the drag.
19929             if (!dc || dc.isLocked()) {
19930                 return;
19931             }
19932
19933             var pt = e.getPoint();
19934
19935             // cache the previous dragOver array
19936             var oldOvers = [];
19937
19938             var outEvts   = [];
19939             var overEvts  = [];
19940             var dropEvts  = [];
19941             var enterEvts = [];
19942
19943             // Check to see if the object(s) we were hovering over is no longer
19944             // being hovered over so we can fire the onDragOut event
19945             for (var i in this.dragOvers) {
19946
19947                 var ddo = this.dragOvers[i];
19948
19949                 if (! this.isTypeOfDD(ddo)) {
19950                     continue;
19951                 }
19952
19953                 if (! this.isOverTarget(pt, ddo, this.mode)) {
19954                     outEvts.push( ddo );
19955                 }
19956
19957                 oldOvers[i] = true;
19958                 delete this.dragOvers[i];
19959             }
19960
19961             for (var sGroup in dc.groups) {
19962
19963                 if ("string" != typeof sGroup) {
19964                     continue;
19965                 }
19966
19967                 for (i in this.ids[sGroup]) {
19968                     var oDD = this.ids[sGroup][i];
19969                     if (! this.isTypeOfDD(oDD)) {
19970                         continue;
19971                     }
19972
19973                     if (oDD.isTarget && !oDD.isLocked() && oDD != dc) {
19974                         if (this.isOverTarget(pt, oDD, this.mode)) {
19975                             // look for drop interactions
19976                             if (isDrop) {
19977                                 dropEvts.push( oDD );
19978                             // look for drag enter and drag over interactions
19979                             } else {
19980
19981                                 // initial drag over: dragEnter fires
19982                                 if (!oldOvers[oDD.id]) {
19983                                     enterEvts.push( oDD );
19984                                 // subsequent drag overs: dragOver fires
19985                                 } else {
19986                                     overEvts.push( oDD );
19987                                 }
19988
19989                                 this.dragOvers[oDD.id] = oDD;
19990                             }
19991                         }
19992                     }
19993                 }
19994             }
19995
19996             if (this.mode) {
19997                 if (outEvts.length) {
19998                     dc.b4DragOut(e, outEvts);
19999                     dc.onDragOut(e, outEvts);
20000                 }
20001
20002                 if (enterEvts.length) {
20003                     dc.onDragEnter(e, enterEvts);
20004                 }
20005
20006                 if (overEvts.length) {
20007                     dc.b4DragOver(e, overEvts);
20008                     dc.onDragOver(e, overEvts);
20009                 }
20010
20011                 if (dropEvts.length) {
20012                     dc.b4DragDrop(e, dropEvts);
20013                     dc.onDragDrop(e, dropEvts);
20014                 }
20015
20016             } else {
20017                 // fire dragout events
20018                 var len = 0;
20019                 for (i=0, len=outEvts.length; i<len; ++i) {
20020                     dc.b4DragOut(e, outEvts[i].id);
20021                     dc.onDragOut(e, outEvts[i].id);
20022                 }
20023
20024                 // fire enter events
20025                 for (i=0,len=enterEvts.length; i<len; ++i) {
20026                     // dc.b4DragEnter(e, oDD.id);
20027                     dc.onDragEnter(e, enterEvts[i].id);
20028                 }
20029
20030                 // fire over events
20031                 for (i=0,len=overEvts.length; i<len; ++i) {
20032                     dc.b4DragOver(e, overEvts[i].id);
20033                     dc.onDragOver(e, overEvts[i].id);
20034                 }
20035
20036                 // fire drop events
20037                 for (i=0, len=dropEvts.length; i<len; ++i) {
20038                     dc.b4DragDrop(e, dropEvts[i].id);
20039                     dc.onDragDrop(e, dropEvts[i].id);
20040                 }
20041
20042             }
20043
20044             // notify about a drop that did not find a target
20045             if (isDrop && !dropEvts.length) {
20046                 dc.onInvalidDrop(e);
20047             }
20048
20049         },
20050
20051         /**
20052          * Helper function for getting the best match from the list of drag
20053          * and drop objects returned by the drag and drop events when we are
20054          * in INTERSECT mode.  It returns either the first object that the
20055          * cursor is over, or the object that has the greatest overlap with
20056          * the dragged element.
20057          * @method getBestMatch
20058          * @param  {DragDrop[]} dds The array of drag and drop objects
20059          * targeted
20060          * @return {DragDrop}       The best single match
20061          * @static
20062          */
20063         getBestMatch: function(dds) {
20064             var winner = null;
20065             // Return null if the input is not what we expect
20066             //if (!dds || !dds.length || dds.length == 0) {
20067                // winner = null;
20068             // If there is only one item, it wins
20069             //} else if (dds.length == 1) {
20070
20071             var len = dds.length;
20072
20073             if (len == 1) {
20074                 winner = dds[0];
20075             } else {
20076                 // Loop through the targeted items
20077                 for (var i=0; i<len; ++i) {
20078                     var dd = dds[i];
20079                     // If the cursor is over the object, it wins.  If the
20080                     // cursor is over multiple matches, the first one we come
20081                     // to wins.
20082                     if (dd.cursorIsOver) {
20083                         winner = dd;
20084                         break;
20085                     // Otherwise the object with the most overlap wins
20086                     } else {
20087                         if (!winner ||
20088                             winner.overlap.getArea() < dd.overlap.getArea()) {
20089                             winner = dd;
20090                         }
20091                     }
20092                 }
20093             }
20094
20095             return winner;
20096         },
20097
20098         /**
20099          * Refreshes the cache of the top-left and bottom-right points of the
20100          * drag and drop objects in the specified group(s).  This is in the
20101          * format that is stored in the drag and drop instance, so typical
20102          * usage is:
20103          * <code>
20104          * Roo.dd.DragDropMgr.refreshCache(ddinstance.groups);
20105          * </code>
20106          * Alternatively:
20107          * <code>
20108          * Roo.dd.DragDropMgr.refreshCache({group1:true, group2:true});
20109          * </code>
20110          * @TODO this really should be an indexed array.  Alternatively this
20111          * method could accept both.
20112          * @method refreshCache
20113          * @param {Object} groups an associative array of groups to refresh
20114          * @static
20115          */
20116         refreshCache: function(groups) {
20117             for (var sGroup in groups) {
20118                 if ("string" != typeof sGroup) {
20119                     continue;
20120                 }
20121                 for (var i in this.ids[sGroup]) {
20122                     var oDD = this.ids[sGroup][i];
20123
20124                     if (this.isTypeOfDD(oDD)) {
20125                     // if (this.isTypeOfDD(oDD) && oDD.isTarget) {
20126                         var loc = this.getLocation(oDD);
20127                         if (loc) {
20128                             this.locationCache[oDD.id] = loc;
20129                         } else {
20130                             delete this.locationCache[oDD.id];
20131                             // this will unregister the drag and drop object if
20132                             // the element is not in a usable state
20133                             // oDD.unreg();
20134                         }
20135                     }
20136                 }
20137             }
20138         },
20139
20140         /**
20141          * This checks to make sure an element exists and is in the DOM.  The
20142          * main purpose is to handle cases where innerHTML is used to remove
20143          * drag and drop objects from the DOM.  IE provides an 'unspecified
20144          * error' when trying to access the offsetParent of such an element
20145          * @method verifyEl
20146          * @param {HTMLElement} el the element to check
20147          * @return {boolean} true if the element looks usable
20148          * @static
20149          */
20150         verifyEl: function(el) {
20151             if (el) {
20152                 var parent;
20153                 if(Roo.isIE){
20154                     try{
20155                         parent = el.offsetParent;
20156                     }catch(e){}
20157                 }else{
20158                     parent = el.offsetParent;
20159                 }
20160                 if (parent) {
20161                     return true;
20162                 }
20163             }
20164
20165             return false;
20166         },
20167
20168         /**
20169          * Returns a Region object containing the drag and drop element's position
20170          * and size, including the padding configured for it
20171          * @method getLocation
20172          * @param {DragDrop} oDD the drag and drop object to get the
20173          *                       location for
20174          * @return {Roo.lib.Region} a Region object representing the total area
20175          *                             the element occupies, including any padding
20176          *                             the instance is configured for.
20177          * @static
20178          */
20179         getLocation: function(oDD) {
20180             if (! this.isTypeOfDD(oDD)) {
20181                 return null;
20182             }
20183
20184             var el = oDD.getEl(), pos, x1, x2, y1, y2, t, r, b, l;
20185
20186             try {
20187                 pos= Roo.lib.Dom.getXY(el);
20188             } catch (e) { }
20189
20190             if (!pos) {
20191                 return null;
20192             }
20193
20194             x1 = pos[0];
20195             x2 = x1 + el.offsetWidth;
20196             y1 = pos[1];
20197             y2 = y1 + el.offsetHeight;
20198
20199             t = y1 - oDD.padding[0];
20200             r = x2 + oDD.padding[1];
20201             b = y2 + oDD.padding[2];
20202             l = x1 - oDD.padding[3];
20203
20204             return new Roo.lib.Region( t, r, b, l );
20205         },
20206
20207         /**
20208          * Checks the cursor location to see if it over the target
20209          * @method isOverTarget
20210          * @param {Roo.lib.Point} pt The point to evaluate
20211          * @param {DragDrop} oTarget the DragDrop object we are inspecting
20212          * @return {boolean} true if the mouse is over the target
20213          * @private
20214          * @static
20215          */
20216         isOverTarget: function(pt, oTarget, intersect) {
20217             // use cache if available
20218             var loc = this.locationCache[oTarget.id];
20219             if (!loc || !this.useCache) {
20220                 loc = this.getLocation(oTarget);
20221                 this.locationCache[oTarget.id] = loc;
20222
20223             }
20224
20225             if (!loc) {
20226                 return false;
20227             }
20228
20229             oTarget.cursorIsOver = loc.contains( pt );
20230
20231             // DragDrop is using this as a sanity check for the initial mousedown
20232             // in this case we are done.  In POINT mode, if the drag obj has no
20233             // contraints, we are also done. Otherwise we need to evaluate the
20234             // location of the target as related to the actual location of the
20235             // dragged element.
20236             var dc = this.dragCurrent;
20237             if (!dc || !dc.getTargetCoord ||
20238                     (!intersect && !dc.constrainX && !dc.constrainY)) {
20239                 return oTarget.cursorIsOver;
20240             }
20241
20242             oTarget.overlap = null;
20243
20244             // Get the current location of the drag element, this is the
20245             // location of the mouse event less the delta that represents
20246             // where the original mousedown happened on the element.  We
20247             // need to consider constraints and ticks as well.
20248             var pos = dc.getTargetCoord(pt.x, pt.y);
20249
20250             var el = dc.getDragEl();
20251             var curRegion = new Roo.lib.Region( pos.y,
20252                                                    pos.x + el.offsetWidth,
20253                                                    pos.y + el.offsetHeight,
20254                                                    pos.x );
20255
20256             var overlap = curRegion.intersect(loc);
20257
20258             if (overlap) {
20259                 oTarget.overlap = overlap;
20260                 return (intersect) ? true : oTarget.cursorIsOver;
20261             } else {
20262                 return false;
20263             }
20264         },
20265
20266         /**
20267          * unload event handler
20268          * @method _onUnload
20269          * @private
20270          * @static
20271          */
20272         _onUnload: function(e, me) {
20273             Roo.dd.DragDropMgr.unregAll();
20274         },
20275
20276         /**
20277          * Cleans up the drag and drop events and objects.
20278          * @method unregAll
20279          * @private
20280          * @static
20281          */
20282         unregAll: function() {
20283
20284             if (this.dragCurrent) {
20285                 this.stopDrag();
20286                 this.dragCurrent = null;
20287             }
20288
20289             this._execOnAll("unreg", []);
20290
20291             for (i in this.elementCache) {
20292                 delete this.elementCache[i];
20293             }
20294
20295             this.elementCache = {};
20296             this.ids = {};
20297         },
20298
20299         /**
20300          * A cache of DOM elements
20301          * @property elementCache
20302          * @private
20303          * @static
20304          */
20305         elementCache: {},
20306
20307         /**
20308          * Get the wrapper for the DOM element specified
20309          * @method getElWrapper
20310          * @param {String} id the id of the element to get
20311          * @return {Roo.dd.DDM.ElementWrapper} the wrapped element
20312          * @private
20313          * @deprecated This wrapper isn't that useful
20314          * @static
20315          */
20316         getElWrapper: function(id) {
20317             var oWrapper = this.elementCache[id];
20318             if (!oWrapper || !oWrapper.el) {
20319                 oWrapper = this.elementCache[id] =
20320                     new this.ElementWrapper(Roo.getDom(id));
20321             }
20322             return oWrapper;
20323         },
20324
20325         /**
20326          * Returns the actual DOM element
20327          * @method getElement
20328          * @param {String} id the id of the elment to get
20329          * @return {Object} The element
20330          * @deprecated use Roo.getDom instead
20331          * @static
20332          */
20333         getElement: function(id) {
20334             return Roo.getDom(id);
20335         },
20336
20337         /**
20338          * Returns the style property for the DOM element (i.e.,
20339          * document.getElById(id).style)
20340          * @method getCss
20341          * @param {String} id the id of the elment to get
20342          * @return {Object} The style property of the element
20343          * @deprecated use Roo.getDom instead
20344          * @static
20345          */
20346         getCss: function(id) {
20347             var el = Roo.getDom(id);
20348             return (el) ? el.style : null;
20349         },
20350
20351         /**
20352          * Inner class for cached elements
20353          * @class DragDropMgr.ElementWrapper
20354          * @for DragDropMgr
20355          * @private
20356          * @deprecated
20357          */
20358         ElementWrapper: function(el) {
20359                 /**
20360                  * The element
20361                  * @property el
20362                  */
20363                 this.el = el || null;
20364                 /**
20365                  * The element id
20366                  * @property id
20367                  */
20368                 this.id = this.el && el.id;
20369                 /**
20370                  * A reference to the style property
20371                  * @property css
20372                  */
20373                 this.css = this.el && el.style;
20374             },
20375
20376         /**
20377          * Returns the X position of an html element
20378          * @method getPosX
20379          * @param el the element for which to get the position
20380          * @return {int} the X coordinate
20381          * @for DragDropMgr
20382          * @deprecated use Roo.lib.Dom.getX instead
20383          * @static
20384          */
20385         getPosX: function(el) {
20386             return Roo.lib.Dom.getX(el);
20387         },
20388
20389         /**
20390          * Returns the Y position of an html element
20391          * @method getPosY
20392          * @param el the element for which to get the position
20393          * @return {int} the Y coordinate
20394          * @deprecated use Roo.lib.Dom.getY instead
20395          * @static
20396          */
20397         getPosY: function(el) {
20398             return Roo.lib.Dom.getY(el);
20399         },
20400
20401         /**
20402          * Swap two nodes.  In IE, we use the native method, for others we
20403          * emulate the IE behavior
20404          * @method swapNode
20405          * @param n1 the first node to swap
20406          * @param n2 the other node to swap
20407          * @static
20408          */
20409         swapNode: function(n1, n2) {
20410             if (n1.swapNode) {
20411                 n1.swapNode(n2);
20412             } else {
20413                 var p = n2.parentNode;
20414                 var s = n2.nextSibling;
20415
20416                 if (s == n1) {
20417                     p.insertBefore(n1, n2);
20418                 } else if (n2 == n1.nextSibling) {
20419                     p.insertBefore(n2, n1);
20420                 } else {
20421                     n1.parentNode.replaceChild(n2, n1);
20422                     p.insertBefore(n1, s);
20423                 }
20424             }
20425         },
20426
20427         /**
20428          * Returns the current scroll position
20429          * @method getScroll
20430          * @private
20431          * @static
20432          */
20433         getScroll: function () {
20434             var t, l, dde=document.documentElement, db=document.body;
20435             if (dde && (dde.scrollTop || dde.scrollLeft)) {
20436                 t = dde.scrollTop;
20437                 l = dde.scrollLeft;
20438             } else if (db) {
20439                 t = db.scrollTop;
20440                 l = db.scrollLeft;
20441             } else {
20442
20443             }
20444             return { top: t, left: l };
20445         },
20446
20447         /**
20448          * Returns the specified element style property
20449          * @method getStyle
20450          * @param {HTMLElement} el          the element
20451          * @param {string}      styleProp   the style property
20452          * @return {string} The value of the style property
20453          * @deprecated use Roo.lib.Dom.getStyle
20454          * @static
20455          */
20456         getStyle: function(el, styleProp) {
20457             return Roo.fly(el).getStyle(styleProp);
20458         },
20459
20460         /**
20461          * Gets the scrollTop
20462          * @method getScrollTop
20463          * @return {int} the document's scrollTop
20464          * @static
20465          */
20466         getScrollTop: function () { return this.getScroll().top; },
20467
20468         /**
20469          * Gets the scrollLeft
20470          * @method getScrollLeft
20471          * @return {int} the document's scrollTop
20472          * @static
20473          */
20474         getScrollLeft: function () { return this.getScroll().left; },
20475
20476         /**
20477          * Sets the x/y position of an element to the location of the
20478          * target element.
20479          * @method moveToEl
20480          * @param {HTMLElement} moveEl      The element to move
20481          * @param {HTMLElement} targetEl    The position reference element
20482          * @static
20483          */
20484         moveToEl: function (moveEl, targetEl) {
20485             var aCoord = Roo.lib.Dom.getXY(targetEl);
20486             Roo.lib.Dom.setXY(moveEl, aCoord);
20487         },
20488
20489         /**
20490          * Numeric array sort function
20491          * @method numericSort
20492          * @static
20493          */
20494         numericSort: function(a, b) { return (a - b); },
20495
20496         /**
20497          * Internal counter
20498          * @property _timeoutCount
20499          * @private
20500          * @static
20501          */
20502         _timeoutCount: 0,
20503
20504         /**
20505          * Trying to make the load order less important.  Without this we get
20506          * an error if this file is loaded before the Event Utility.
20507          * @method _addListeners
20508          * @private
20509          * @static
20510          */
20511         _addListeners: function() {
20512             var DDM = Roo.dd.DDM;
20513             if ( Roo.lib.Event && document ) {
20514                 DDM._onLoad();
20515             } else {
20516                 if (DDM._timeoutCount > 2000) {
20517                 } else {
20518                     setTimeout(DDM._addListeners, 10);
20519                     if (document && document.body) {
20520                         DDM._timeoutCount += 1;
20521                     }
20522                 }
20523             }
20524         },
20525
20526         /**
20527          * Recursively searches the immediate parent and all child nodes for
20528          * the handle element in order to determine wheter or not it was
20529          * clicked.
20530          * @method handleWasClicked
20531          * @param node the html element to inspect
20532          * @static
20533          */
20534         handleWasClicked: function(node, id) {
20535             if (this.isHandle(id, node.id)) {
20536                 return true;
20537             } else {
20538                 // check to see if this is a text node child of the one we want
20539                 var p = node.parentNode;
20540
20541                 while (p) {
20542                     if (this.isHandle(id, p.id)) {
20543                         return true;
20544                     } else {
20545                         p = p.parentNode;
20546                     }
20547                 }
20548             }
20549
20550             return false;
20551         }
20552
20553     };
20554
20555 }();
20556
20557 // shorter alias, save a few bytes
20558 Roo.dd.DDM = Roo.dd.DragDropMgr;
20559 Roo.dd.DDM._addListeners();
20560
20561 }/*
20562  * Based on:
20563  * Ext JS Library 1.1.1
20564  * Copyright(c) 2006-2007, Ext JS, LLC.
20565  *
20566  * Originally Released Under LGPL - original licence link has changed is not relivant.
20567  *
20568  * Fork - LGPL
20569  * <script type="text/javascript">
20570  */
20571
20572 /**
20573  * @class Roo.dd.DD
20574  * A DragDrop implementation where the linked element follows the
20575  * mouse cursor during a drag.
20576  * @extends Roo.dd.DragDrop
20577  * @constructor
20578  * @param {String} id the id of the linked element
20579  * @param {String} sGroup the group of related DragDrop items
20580  * @param {object} config an object containing configurable attributes
20581  *                Valid properties for DD:
20582  *                    scroll
20583  */
20584 Roo.dd.DD = function(id, sGroup, config) {
20585     if (id) {
20586         this.init(id, sGroup, config);
20587     }
20588 };
20589
20590 Roo.extend(Roo.dd.DD, Roo.dd.DragDrop, {
20591
20592     /**
20593      * When set to true, the utility automatically tries to scroll the browser
20594      * window wehn a drag and drop element is dragged near the viewport boundary.
20595      * Defaults to true.
20596      * @property scroll
20597      * @type boolean
20598      */
20599     scroll: true,
20600
20601     /**
20602      * Sets the pointer offset to the distance between the linked element's top
20603      * left corner and the location the element was clicked
20604      * @method autoOffset
20605      * @param {int} iPageX the X coordinate of the click
20606      * @param {int} iPageY the Y coordinate of the click
20607      */
20608     autoOffset: function(iPageX, iPageY) {
20609         var x = iPageX - this.startPageX;
20610         var y = iPageY - this.startPageY;
20611         this.setDelta(x, y);
20612     },
20613
20614     /**
20615      * Sets the pointer offset.  You can call this directly to force the
20616      * offset to be in a particular location (e.g., pass in 0,0 to set it
20617      * to the center of the object)
20618      * @method setDelta
20619      * @param {int} iDeltaX the distance from the left
20620      * @param {int} iDeltaY the distance from the top
20621      */
20622     setDelta: function(iDeltaX, iDeltaY) {
20623         this.deltaX = iDeltaX;
20624         this.deltaY = iDeltaY;
20625     },
20626
20627     /**
20628      * Sets the drag element to the location of the mousedown or click event,
20629      * maintaining the cursor location relative to the location on the element
20630      * that was clicked.  Override this if you want to place the element in a
20631      * location other than where the cursor is.
20632      * @method setDragElPos
20633      * @param {int} iPageX the X coordinate of the mousedown or drag event
20634      * @param {int} iPageY the Y coordinate of the mousedown or drag event
20635      */
20636     setDragElPos: function(iPageX, iPageY) {
20637         // the first time we do this, we are going to check to make sure
20638         // the element has css positioning
20639
20640         var el = this.getDragEl();
20641         this.alignElWithMouse(el, iPageX, iPageY);
20642     },
20643
20644     /**
20645      * Sets the element to the location of the mousedown or click event,
20646      * maintaining the cursor location relative to the location on the element
20647      * that was clicked.  Override this if you want to place the element in a
20648      * location other than where the cursor is.
20649      * @method alignElWithMouse
20650      * @param {HTMLElement} el the element to move
20651      * @param {int} iPageX the X coordinate of the mousedown or drag event
20652      * @param {int} iPageY the Y coordinate of the mousedown or drag event
20653      */
20654     alignElWithMouse: function(el, iPageX, iPageY) {
20655         var oCoord = this.getTargetCoord(iPageX, iPageY);
20656         var fly = el.dom ? el : Roo.fly(el);
20657         if (!this.deltaSetXY) {
20658             var aCoord = [oCoord.x, oCoord.y];
20659             fly.setXY(aCoord);
20660             var newLeft = fly.getLeft(true);
20661             var newTop  = fly.getTop(true);
20662             this.deltaSetXY = [ newLeft - oCoord.x, newTop - oCoord.y ];
20663         } else {
20664             fly.setLeftTop(oCoord.x + this.deltaSetXY[0], oCoord.y + this.deltaSetXY[1]);
20665         }
20666
20667         this.cachePosition(oCoord.x, oCoord.y);
20668         this.autoScroll(oCoord.x, oCoord.y, el.offsetHeight, el.offsetWidth);
20669         return oCoord;
20670     },
20671
20672     /**
20673      * Saves the most recent position so that we can reset the constraints and
20674      * tick marks on-demand.  We need to know this so that we can calculate the
20675      * number of pixels the element is offset from its original position.
20676      * @method cachePosition
20677      * @param iPageX the current x position (optional, this just makes it so we
20678      * don't have to look it up again)
20679      * @param iPageY the current y position (optional, this just makes it so we
20680      * don't have to look it up again)
20681      */
20682     cachePosition: function(iPageX, iPageY) {
20683         if (iPageX) {
20684             this.lastPageX = iPageX;
20685             this.lastPageY = iPageY;
20686         } else {
20687             var aCoord = Roo.lib.Dom.getXY(this.getEl());
20688             this.lastPageX = aCoord[0];
20689             this.lastPageY = aCoord[1];
20690         }
20691     },
20692
20693     /**
20694      * Auto-scroll the window if the dragged object has been moved beyond the
20695      * visible window boundary.
20696      * @method autoScroll
20697      * @param {int} x the drag element's x position
20698      * @param {int} y the drag element's y position
20699      * @param {int} h the height of the drag element
20700      * @param {int} w the width of the drag element
20701      * @private
20702      */
20703     autoScroll: function(x, y, h, w) {
20704
20705         if (this.scroll) {
20706             // The client height
20707             var clientH = Roo.lib.Dom.getViewWidth();
20708
20709             // The client width
20710             var clientW = Roo.lib.Dom.getViewHeight();
20711
20712             // The amt scrolled down
20713             var st = this.DDM.getScrollTop();
20714
20715             // The amt scrolled right
20716             var sl = this.DDM.getScrollLeft();
20717
20718             // Location of the bottom of the element
20719             var bot = h + y;
20720
20721             // Location of the right of the element
20722             var right = w + x;
20723
20724             // The distance from the cursor to the bottom of the visible area,
20725             // adjusted so that we don't scroll if the cursor is beyond the
20726             // element drag constraints
20727             var toBot = (clientH + st - y - this.deltaY);
20728
20729             // The distance from the cursor to the right of the visible area
20730             var toRight = (clientW + sl - x - this.deltaX);
20731
20732
20733             // How close to the edge the cursor must be before we scroll
20734             // var thresh = (document.all) ? 100 : 40;
20735             var thresh = 40;
20736
20737             // How many pixels to scroll per autoscroll op.  This helps to reduce
20738             // clunky scrolling. IE is more sensitive about this ... it needs this
20739             // value to be higher.
20740             var scrAmt = (document.all) ? 80 : 30;
20741
20742             // Scroll down if we are near the bottom of the visible page and the
20743             // obj extends below the crease
20744             if ( bot > clientH && toBot < thresh ) {
20745                 window.scrollTo(sl, st + scrAmt);
20746             }
20747
20748             // Scroll up if the window is scrolled down and the top of the object
20749             // goes above the top border
20750             if ( y < st && st > 0 && y - st < thresh ) {
20751                 window.scrollTo(sl, st - scrAmt);
20752             }
20753
20754             // Scroll right if the obj is beyond the right border and the cursor is
20755             // near the border.
20756             if ( right > clientW && toRight < thresh ) {
20757                 window.scrollTo(sl + scrAmt, st);
20758             }
20759
20760             // Scroll left if the window has been scrolled to the right and the obj
20761             // extends past the left border
20762             if ( x < sl && sl > 0 && x - sl < thresh ) {
20763                 window.scrollTo(sl - scrAmt, st);
20764             }
20765         }
20766     },
20767
20768     /**
20769      * Finds the location the element should be placed if we want to move
20770      * it to where the mouse location less the click offset would place us.
20771      * @method getTargetCoord
20772      * @param {int} iPageX the X coordinate of the click
20773      * @param {int} iPageY the Y coordinate of the click
20774      * @return an object that contains the coordinates (Object.x and Object.y)
20775      * @private
20776      */
20777     getTargetCoord: function(iPageX, iPageY) {
20778
20779
20780         var x = iPageX - this.deltaX;
20781         var y = iPageY - this.deltaY;
20782
20783         if (this.constrainX) {
20784             if (x < this.minX) { x = this.minX; }
20785             if (x > this.maxX) { x = this.maxX; }
20786         }
20787
20788         if (this.constrainY) {
20789             if (y < this.minY) { y = this.minY; }
20790             if (y > this.maxY) { y = this.maxY; }
20791         }
20792
20793         x = this.getTick(x, this.xTicks);
20794         y = this.getTick(y, this.yTicks);
20795
20796
20797         return {x:x, y:y};
20798     },
20799
20800     /*
20801      * Sets up config options specific to this class. Overrides
20802      * Roo.dd.DragDrop, but all versions of this method through the
20803      * inheritance chain are called
20804      */
20805     applyConfig: function() {
20806         Roo.dd.DD.superclass.applyConfig.call(this);
20807         this.scroll = (this.config.scroll !== false);
20808     },
20809
20810     /*
20811      * Event that fires prior to the onMouseDown event.  Overrides
20812      * Roo.dd.DragDrop.
20813      */
20814     b4MouseDown: function(e) {
20815         // this.resetConstraints();
20816         this.autoOffset(e.getPageX(),
20817                             e.getPageY());
20818     },
20819
20820     /*
20821      * Event that fires prior to the onDrag event.  Overrides
20822      * Roo.dd.DragDrop.
20823      */
20824     b4Drag: function(e) {
20825         this.setDragElPos(e.getPageX(),
20826                             e.getPageY());
20827     },
20828
20829     toString: function() {
20830         return ("DD " + this.id);
20831     }
20832
20833     //////////////////////////////////////////////////////////////////////////
20834     // Debugging ygDragDrop events that can be overridden
20835     //////////////////////////////////////////////////////////////////////////
20836     /*
20837     startDrag: function(x, y) {
20838     },
20839
20840     onDrag: function(e) {
20841     },
20842
20843     onDragEnter: function(e, id) {
20844     },
20845
20846     onDragOver: function(e, id) {
20847     },
20848
20849     onDragOut: function(e, id) {
20850     },
20851
20852     onDragDrop: function(e, id) {
20853     },
20854
20855     endDrag: function(e) {
20856     }
20857
20858     */
20859
20860 });/*
20861  * Based on:
20862  * Ext JS Library 1.1.1
20863  * Copyright(c) 2006-2007, Ext JS, LLC.
20864  *
20865  * Originally Released Under LGPL - original licence link has changed is not relivant.
20866  *
20867  * Fork - LGPL
20868  * <script type="text/javascript">
20869  */
20870
20871 /**
20872  * @class Roo.dd.DDProxy
20873  * A DragDrop implementation that inserts an empty, bordered div into
20874  * the document that follows the cursor during drag operations.  At the time of
20875  * the click, the frame div is resized to the dimensions of the linked html
20876  * element, and moved to the exact location of the linked element.
20877  *
20878  * References to the "frame" element refer to the single proxy element that
20879  * was created to be dragged in place of all DDProxy elements on the
20880  * page.
20881  *
20882  * @extends Roo.dd.DD
20883  * @constructor
20884  * @param {String} id the id of the linked html element
20885  * @param {String} sGroup the group of related DragDrop objects
20886  * @param {object} config an object containing configurable attributes
20887  *                Valid properties for DDProxy in addition to those in DragDrop:
20888  *                   resizeFrame, centerFrame, dragElId
20889  */
20890 Roo.dd.DDProxy = function(id, sGroup, config) {
20891     if (id) {
20892         this.init(id, sGroup, config);
20893         this.initFrame();
20894     }
20895 };
20896
20897 /**
20898  * The default drag frame div id
20899  * @property Roo.dd.DDProxy.dragElId
20900  * @type String
20901  * @static
20902  */
20903 Roo.dd.DDProxy.dragElId = "ygddfdiv";
20904
20905 Roo.extend(Roo.dd.DDProxy, Roo.dd.DD, {
20906
20907     /**
20908      * By default we resize the drag frame to be the same size as the element
20909      * we want to drag (this is to get the frame effect).  We can turn it off
20910      * if we want a different behavior.
20911      * @property resizeFrame
20912      * @type boolean
20913      */
20914     resizeFrame: true,
20915
20916     /**
20917      * By default the frame is positioned exactly where the drag element is, so
20918      * we use the cursor offset provided by Roo.dd.DD.  Another option that works only if
20919      * you do not have constraints on the obj is to have the drag frame centered
20920      * around the cursor.  Set centerFrame to true for this effect.
20921      * @property centerFrame
20922      * @type boolean
20923      */
20924     centerFrame: false,
20925
20926     /**
20927      * Creates the proxy element if it does not yet exist
20928      * @method createFrame
20929      */
20930     createFrame: function() {
20931         var self = this;
20932         var body = document.body;
20933
20934         if (!body || !body.firstChild) {
20935             setTimeout( function() { self.createFrame(); }, 50 );
20936             return;
20937         }
20938
20939         var div = this.getDragEl();
20940
20941         if (!div) {
20942             div    = document.createElement("div");
20943             div.id = this.dragElId;
20944             var s  = div.style;
20945
20946             s.position   = "absolute";
20947             s.visibility = "hidden";
20948             s.cursor     = "move";
20949             s.border     = "2px solid #aaa";
20950             s.zIndex     = 999;
20951
20952             // appendChild can blow up IE if invoked prior to the window load event
20953             // while rendering a table.  It is possible there are other scenarios
20954             // that would cause this to happen as well.
20955             body.insertBefore(div, body.firstChild);
20956         }
20957     },
20958
20959     /**
20960      * Initialization for the drag frame element.  Must be called in the
20961      * constructor of all subclasses
20962      * @method initFrame
20963      */
20964     initFrame: function() {
20965         this.createFrame();
20966     },
20967
20968     applyConfig: function() {
20969         Roo.dd.DDProxy.superclass.applyConfig.call(this);
20970
20971         this.resizeFrame = (this.config.resizeFrame !== false);
20972         this.centerFrame = (this.config.centerFrame);
20973         this.setDragElId(this.config.dragElId || Roo.dd.DDProxy.dragElId);
20974     },
20975
20976     /**
20977      * Resizes the drag frame to the dimensions of the clicked object, positions
20978      * it over the object, and finally displays it
20979      * @method showFrame
20980      * @param {int} iPageX X click position
20981      * @param {int} iPageY Y click position
20982      * @private
20983      */
20984     showFrame: function(iPageX, iPageY) {
20985         var el = this.getEl();
20986         var dragEl = this.getDragEl();
20987         var s = dragEl.style;
20988
20989         this._resizeProxy();
20990
20991         if (this.centerFrame) {
20992             this.setDelta( Math.round(parseInt(s.width,  10)/2),
20993                            Math.round(parseInt(s.height, 10)/2) );
20994         }
20995
20996         this.setDragElPos(iPageX, iPageY);
20997
20998         Roo.fly(dragEl).show();
20999     },
21000
21001     /**
21002      * The proxy is automatically resized to the dimensions of the linked
21003      * element when a drag is initiated, unless resizeFrame is set to false
21004      * @method _resizeProxy
21005      * @private
21006      */
21007     _resizeProxy: function() {
21008         if (this.resizeFrame) {
21009             var el = this.getEl();
21010             Roo.fly(this.getDragEl()).setSize(el.offsetWidth, el.offsetHeight);
21011         }
21012     },
21013
21014     // overrides Roo.dd.DragDrop
21015     b4MouseDown: function(e) {
21016         var x = e.getPageX();
21017         var y = e.getPageY();
21018         this.autoOffset(x, y);
21019         this.setDragElPos(x, y);
21020     },
21021
21022     // overrides Roo.dd.DragDrop
21023     b4StartDrag: function(x, y) {
21024         // show the drag frame
21025         this.showFrame(x, y);
21026     },
21027
21028     // overrides Roo.dd.DragDrop
21029     b4EndDrag: function(e) {
21030         Roo.fly(this.getDragEl()).hide();
21031     },
21032
21033     // overrides Roo.dd.DragDrop
21034     // By default we try to move the element to the last location of the frame.
21035     // This is so that the default behavior mirrors that of Roo.dd.DD.
21036     endDrag: function(e) {
21037
21038         var lel = this.getEl();
21039         var del = this.getDragEl();
21040
21041         // Show the drag frame briefly so we can get its position
21042         del.style.visibility = "";
21043
21044         this.beforeMove();
21045         // Hide the linked element before the move to get around a Safari
21046         // rendering bug.
21047         lel.style.visibility = "hidden";
21048         Roo.dd.DDM.moveToEl(lel, del);
21049         del.style.visibility = "hidden";
21050         lel.style.visibility = "";
21051
21052         this.afterDrag();
21053     },
21054
21055     beforeMove : function(){
21056
21057     },
21058
21059     afterDrag : function(){
21060
21061     },
21062
21063     toString: function() {
21064         return ("DDProxy " + this.id);
21065     }
21066
21067 });
21068 /*
21069  * Based on:
21070  * Ext JS Library 1.1.1
21071  * Copyright(c) 2006-2007, Ext JS, LLC.
21072  *
21073  * Originally Released Under LGPL - original licence link has changed is not relivant.
21074  *
21075  * Fork - LGPL
21076  * <script type="text/javascript">
21077  */
21078
21079  /**
21080  * @class Roo.dd.DDTarget
21081  * A DragDrop implementation that does not move, but can be a drop
21082  * target.  You would get the same result by simply omitting implementation
21083  * for the event callbacks, but this way we reduce the processing cost of the
21084  * event listener and the callbacks.
21085  * @extends Roo.dd.DragDrop
21086  * @constructor
21087  * @param {String} id the id of the element that is a drop target
21088  * @param {String} sGroup the group of related DragDrop objects
21089  * @param {object} config an object containing configurable attributes
21090  *                 Valid properties for DDTarget in addition to those in
21091  *                 DragDrop:
21092  *                    none
21093  */
21094 Roo.dd.DDTarget = function(id, sGroup, config) {
21095     if (id) {
21096         this.initTarget(id, sGroup, config);
21097     }
21098     if (config.listeners || config.events) { 
21099        Roo.dd.DragDrop.superclass.constructor.call(this,  { 
21100             listeners : config.listeners || {}, 
21101             events : config.events || {} 
21102         });    
21103     }
21104 };
21105
21106 // Roo.dd.DDTarget.prototype = new Roo.dd.DragDrop();
21107 Roo.extend(Roo.dd.DDTarget, Roo.dd.DragDrop, {
21108     toString: function() {
21109         return ("DDTarget " + this.id);
21110     }
21111 });
21112 /*
21113  * Based on:
21114  * Ext JS Library 1.1.1
21115  * Copyright(c) 2006-2007, Ext JS, LLC.
21116  *
21117  * Originally Released Under LGPL - original licence link has changed is not relivant.
21118  *
21119  * Fork - LGPL
21120  * <script type="text/javascript">
21121  */
21122  
21123
21124 /**
21125  * @class Roo.dd.ScrollManager
21126  * Provides automatic scrolling of overflow regions in the page during drag operations.<br><br>
21127  * <b>Note: This class uses "Point Mode" and is untested in "Intersect Mode".</b>
21128  * @singleton
21129  */
21130 Roo.dd.ScrollManager = function(){
21131     var ddm = Roo.dd.DragDropMgr;
21132     var els = {};
21133     var dragEl = null;
21134     var proc = {};
21135     
21136     
21137     
21138     var onStop = function(e){
21139         dragEl = null;
21140         clearProc();
21141     };
21142     
21143     var triggerRefresh = function(){
21144         if(ddm.dragCurrent){
21145              ddm.refreshCache(ddm.dragCurrent.groups);
21146         }
21147     };
21148     
21149     var doScroll = function(){
21150         if(ddm.dragCurrent){
21151             var dds = Roo.dd.ScrollManager;
21152             if(!dds.animate){
21153                 if(proc.el.scroll(proc.dir, dds.increment)){
21154                     triggerRefresh();
21155                 }
21156             }else{
21157                 proc.el.scroll(proc.dir, dds.increment, true, dds.animDuration, triggerRefresh);
21158             }
21159         }
21160     };
21161     
21162     var clearProc = function(){
21163         if(proc.id){
21164             clearInterval(proc.id);
21165         }
21166         proc.id = 0;
21167         proc.el = null;
21168         proc.dir = "";
21169     };
21170     
21171     var startProc = function(el, dir){
21172          Roo.log('scroll startproc');
21173         clearProc();
21174         proc.el = el;
21175         proc.dir = dir;
21176         proc.id = setInterval(doScroll, Roo.dd.ScrollManager.frequency);
21177     };
21178     
21179     var onFire = function(e, isDrop){
21180        
21181         if(isDrop || !ddm.dragCurrent){ return; }
21182         var dds = Roo.dd.ScrollManager;
21183         if(!dragEl || dragEl != ddm.dragCurrent){
21184             dragEl = ddm.dragCurrent;
21185             // refresh regions on drag start
21186             dds.refreshCache();
21187         }
21188         
21189         var xy = Roo.lib.Event.getXY(e);
21190         var pt = new Roo.lib.Point(xy[0], xy[1]);
21191         for(var id in els){
21192             var el = els[id], r = el._region;
21193             if(r && r.contains(pt) && el.isScrollable()){
21194                 if(r.bottom - pt.y <= dds.thresh){
21195                     if(proc.el != el){
21196                         startProc(el, "down");
21197                     }
21198                     return;
21199                 }else if(r.right - pt.x <= dds.thresh){
21200                     if(proc.el != el){
21201                         startProc(el, "left");
21202                     }
21203                     return;
21204                 }else if(pt.y - r.top <= dds.thresh){
21205                     if(proc.el != el){
21206                         startProc(el, "up");
21207                     }
21208                     return;
21209                 }else if(pt.x - r.left <= dds.thresh){
21210                     if(proc.el != el){
21211                         startProc(el, "right");
21212                     }
21213                     return;
21214                 }
21215             }
21216         }
21217         clearProc();
21218     };
21219     
21220     ddm.fireEvents = ddm.fireEvents.createSequence(onFire, ddm);
21221     ddm.stopDrag = ddm.stopDrag.createSequence(onStop, ddm);
21222     
21223     return {
21224         /**
21225          * Registers new overflow element(s) to auto scroll
21226          * @param {String/HTMLElement/Element/Array} el The id of or the element to be scrolled or an array of either
21227          */
21228         register : function(el){
21229             if(el instanceof Array){
21230                 for(var i = 0, len = el.length; i < len; i++) {
21231                         this.register(el[i]);
21232                 }
21233             }else{
21234                 el = Roo.get(el);
21235                 els[el.id] = el;
21236             }
21237             Roo.dd.ScrollManager.els = els;
21238         },
21239         
21240         /**
21241          * Unregisters overflow element(s) so they are no longer scrolled
21242          * @param {String/HTMLElement/Element/Array} el The id of or the element to be removed or an array of either
21243          */
21244         unregister : function(el){
21245             if(el instanceof Array){
21246                 for(var i = 0, len = el.length; i < len; i++) {
21247                         this.unregister(el[i]);
21248                 }
21249             }else{
21250                 el = Roo.get(el);
21251                 delete els[el.id];
21252             }
21253         },
21254         
21255         /**
21256          * The number of pixels from the edge of a container the pointer needs to be to 
21257          * trigger scrolling (defaults to 25)
21258          * @type Number
21259          */
21260         thresh : 25,
21261         
21262         /**
21263          * The number of pixels to scroll in each scroll increment (defaults to 50)
21264          * @type Number
21265          */
21266         increment : 100,
21267         
21268         /**
21269          * The frequency of scrolls in milliseconds (defaults to 500)
21270          * @type Number
21271          */
21272         frequency : 500,
21273         
21274         /**
21275          * True to animate the scroll (defaults to true)
21276          * @type Boolean
21277          */
21278         animate: true,
21279         
21280         /**
21281          * The animation duration in seconds - 
21282          * MUST BE less than Roo.dd.ScrollManager.frequency! (defaults to .4)
21283          * @type Number
21284          */
21285         animDuration: .4,
21286         
21287         /**
21288          * Manually trigger a cache refresh.
21289          */
21290         refreshCache : function(){
21291             for(var id in els){
21292                 if(typeof els[id] == 'object'){ // for people extending the object prototype
21293                     els[id]._region = els[id].getRegion();
21294                 }
21295             }
21296         }
21297     };
21298 }();/*
21299  * Based on:
21300  * Ext JS Library 1.1.1
21301  * Copyright(c) 2006-2007, Ext JS, LLC.
21302  *
21303  * Originally Released Under LGPL - original licence link has changed is not relivant.
21304  *
21305  * Fork - LGPL
21306  * <script type="text/javascript">
21307  */
21308  
21309
21310 /**
21311  * @class Roo.dd.Registry
21312  * Provides easy access to all drag drop components that are registered on a page.  Items can be retrieved either
21313  * directly by DOM node id, or by passing in the drag drop event that occurred and looking up the event target.
21314  * @singleton
21315  */
21316 Roo.dd.Registry = function(){
21317     var elements = {}; 
21318     var handles = {}; 
21319     var autoIdSeed = 0;
21320
21321     var getId = function(el, autogen){
21322         if(typeof el == "string"){
21323             return el;
21324         }
21325         var id = el.id;
21326         if(!id && autogen !== false){
21327             id = "roodd-" + (++autoIdSeed);
21328             el.id = id;
21329         }
21330         return id;
21331     };
21332     
21333     return {
21334     /**
21335      * Register a drag drop element
21336      * @param {String|HTMLElement} element The id or DOM node to register
21337      * @param {Object} data (optional) A custom data object that will be passed between the elements that are involved
21338      * in drag drop operations.  You can populate this object with any arbitrary properties that your own code
21339      * knows how to interpret, plus there are some specific properties known to the Registry that should be
21340      * populated in the data object (if applicable):
21341      * <pre>
21342 Value      Description<br />
21343 ---------  ------------------------------------------<br />
21344 handles    Array of DOM nodes that trigger dragging<br />
21345            for the element being registered<br />
21346 isHandle   True if the element passed in triggers<br />
21347            dragging itself, else false
21348 </pre>
21349      */
21350         register : function(el, data){
21351             data = data || {};
21352             if(typeof el == "string"){
21353                 el = document.getElementById(el);
21354             }
21355             data.ddel = el;
21356             elements[getId(el)] = data;
21357             if(data.isHandle !== false){
21358                 handles[data.ddel.id] = data;
21359             }
21360             if(data.handles){
21361                 var hs = data.handles;
21362                 for(var i = 0, len = hs.length; i < len; i++){
21363                         handles[getId(hs[i])] = data;
21364                 }
21365             }
21366         },
21367
21368     /**
21369      * Unregister a drag drop element
21370      * @param {String|HTMLElement}  element The id or DOM node to unregister
21371      */
21372         unregister : function(el){
21373             var id = getId(el, false);
21374             var data = elements[id];
21375             if(data){
21376                 delete elements[id];
21377                 if(data.handles){
21378                     var hs = data.handles;
21379                     for(var i = 0, len = hs.length; i < len; i++){
21380                         delete handles[getId(hs[i], false)];
21381                     }
21382                 }
21383             }
21384         },
21385
21386     /**
21387      * Returns the handle registered for a DOM Node by id
21388      * @param {String|HTMLElement} id The DOM node or id to look up
21389      * @return {Object} handle The custom handle data
21390      */
21391         getHandle : function(id){
21392             if(typeof id != "string"){ // must be element?
21393                 id = id.id;
21394             }
21395             return handles[id];
21396         },
21397
21398     /**
21399      * Returns the handle that is registered for the DOM node that is the target of the event
21400      * @param {Event} e The event
21401      * @return {Object} handle The custom handle data
21402      */
21403         getHandleFromEvent : function(e){
21404             var t = Roo.lib.Event.getTarget(e);
21405             return t ? handles[t.id] : null;
21406         },
21407
21408     /**
21409      * Returns a custom data object that is registered for a DOM node by id
21410      * @param {String|HTMLElement} id The DOM node or id to look up
21411      * @return {Object} data The custom data
21412      */
21413         getTarget : function(id){
21414             if(typeof id != "string"){ // must be element?
21415                 id = id.id;
21416             }
21417             return elements[id];
21418         },
21419
21420     /**
21421      * Returns a custom data object that is registered for the DOM node that is the target of the event
21422      * @param {Event} e The event
21423      * @return {Object} data The custom data
21424      */
21425         getTargetFromEvent : function(e){
21426             var t = Roo.lib.Event.getTarget(e);
21427             return t ? elements[t.id] || handles[t.id] : null;
21428         }
21429     };
21430 }();/*
21431  * Based on:
21432  * Ext JS Library 1.1.1
21433  * Copyright(c) 2006-2007, Ext JS, LLC.
21434  *
21435  * Originally Released Under LGPL - original licence link has changed is not relivant.
21436  *
21437  * Fork - LGPL
21438  * <script type="text/javascript">
21439  */
21440  
21441
21442 /**
21443  * @class Roo.dd.StatusProxy
21444  * A specialized drag proxy that supports a drop status icon, {@link Roo.Layer} styles and auto-repair.  This is the
21445  * default drag proxy used by all Roo.dd components.
21446  * @constructor
21447  * @param {Object} config
21448  */
21449 Roo.dd.StatusProxy = function(config){
21450     Roo.apply(this, config);
21451     this.id = this.id || Roo.id();
21452     this.el = new Roo.Layer({
21453         dh: {
21454             id: this.id, tag: "div", cls: "x-dd-drag-proxy "+this.dropNotAllowed, children: [
21455                 {tag: "div", cls: "x-dd-drop-icon"},
21456                 {tag: "div", cls: "x-dd-drag-ghost"}
21457             ]
21458         }, 
21459         shadow: !config || config.shadow !== false
21460     });
21461     this.ghost = Roo.get(this.el.dom.childNodes[1]);
21462     this.dropStatus = this.dropNotAllowed;
21463 };
21464
21465 Roo.dd.StatusProxy.prototype = {
21466     /**
21467      * @cfg {String} dropAllowed
21468      * The CSS class to apply to the status element when drop is allowed (defaults to "x-dd-drop-ok").
21469      */
21470     dropAllowed : "x-dd-drop-ok",
21471     /**
21472      * @cfg {String} dropNotAllowed
21473      * The CSS class to apply to the status element when drop is not allowed (defaults to "x-dd-drop-nodrop").
21474      */
21475     dropNotAllowed : "x-dd-drop-nodrop",
21476
21477     /**
21478      * Updates the proxy's visual element to indicate the status of whether or not drop is allowed
21479      * over the current target element.
21480      * @param {String} cssClass The css class for the new drop status indicator image
21481      */
21482     setStatus : function(cssClass){
21483         cssClass = cssClass || this.dropNotAllowed;
21484         if(this.dropStatus != cssClass){
21485             this.el.replaceClass(this.dropStatus, cssClass);
21486             this.dropStatus = cssClass;
21487         }
21488     },
21489
21490     /**
21491      * Resets the status indicator to the default dropNotAllowed value
21492      * @param {Boolean} clearGhost True to also remove all content from the ghost, false to preserve it
21493      */
21494     reset : function(clearGhost){
21495         this.el.dom.className = "x-dd-drag-proxy " + this.dropNotAllowed;
21496         this.dropStatus = this.dropNotAllowed;
21497         if(clearGhost){
21498             this.ghost.update("");
21499         }
21500     },
21501
21502     /**
21503      * Updates the contents of the ghost element
21504      * @param {String} html The html that will replace the current innerHTML of the ghost element
21505      */
21506     update : function(html){
21507         if(typeof html == "string"){
21508             this.ghost.update(html);
21509         }else{
21510             this.ghost.update("");
21511             html.style.margin = "0";
21512             this.ghost.dom.appendChild(html);
21513         }
21514         // ensure float = none set?? cant remember why though.
21515         var el = this.ghost.dom.firstChild;
21516                 if(el){
21517                         Roo.fly(el).setStyle('float', 'none');
21518                 }
21519     },
21520     
21521     /**
21522      * Returns the underlying proxy {@link Roo.Layer}
21523      * @return {Roo.Layer} el
21524     */
21525     getEl : function(){
21526         return this.el;
21527     },
21528
21529     /**
21530      * Returns the ghost element
21531      * @return {Roo.Element} el
21532      */
21533     getGhost : function(){
21534         return this.ghost;
21535     },
21536
21537     /**
21538      * Hides the proxy
21539      * @param {Boolean} clear True to reset the status and clear the ghost contents, false to preserve them
21540      */
21541     hide : function(clear){
21542         this.el.hide();
21543         if(clear){
21544             this.reset(true);
21545         }
21546     },
21547
21548     /**
21549      * Stops the repair animation if it's currently running
21550      */
21551     stop : function(){
21552         if(this.anim && this.anim.isAnimated && this.anim.isAnimated()){
21553             this.anim.stop();
21554         }
21555     },
21556
21557     /**
21558      * Displays this proxy
21559      */
21560     show : function(){
21561         this.el.show();
21562     },
21563
21564     /**
21565      * Force the Layer to sync its shadow and shim positions to the element
21566      */
21567     sync : function(){
21568         this.el.sync();
21569     },
21570
21571     /**
21572      * Causes the proxy to return to its position of origin via an animation.  Should be called after an
21573      * invalid drop operation by the item being dragged.
21574      * @param {Array} xy The XY position of the element ([x, y])
21575      * @param {Function} callback The function to call after the repair is complete
21576      * @param {Object} scope The scope in which to execute the callback
21577      */
21578     repair : function(xy, callback, scope){
21579         this.callback = callback;
21580         this.scope = scope;
21581         if(xy && this.animRepair !== false){
21582             this.el.addClass("x-dd-drag-repair");
21583             this.el.hideUnders(true);
21584             this.anim = this.el.shift({
21585                 duration: this.repairDuration || .5,
21586                 easing: 'easeOut',
21587                 xy: xy,
21588                 stopFx: true,
21589                 callback: this.afterRepair,
21590                 scope: this
21591             });
21592         }else{
21593             this.afterRepair();
21594         }
21595     },
21596
21597     // private
21598     afterRepair : function(){
21599         this.hide(true);
21600         if(typeof this.callback == "function"){
21601             this.callback.call(this.scope || this);
21602         }
21603         this.callback = null;
21604         this.scope = null;
21605     }
21606 };/*
21607  * Based on:
21608  * Ext JS Library 1.1.1
21609  * Copyright(c) 2006-2007, Ext JS, LLC.
21610  *
21611  * Originally Released Under LGPL - original licence link has changed is not relivant.
21612  *
21613  * Fork - LGPL
21614  * <script type="text/javascript">
21615  */
21616
21617 /**
21618  * @class Roo.dd.DragSource
21619  * @extends Roo.dd.DDProxy
21620  * A simple class that provides the basic implementation needed to make any element draggable.
21621  * @constructor
21622  * @param {String/HTMLElement/Element} el The container element
21623  * @param {Object} config
21624  */
21625 Roo.dd.DragSource = function(el, config){
21626     this.el = Roo.get(el);
21627     this.dragData = {};
21628     
21629     Roo.apply(this, config);
21630     
21631     if(!this.proxy){
21632         this.proxy = new Roo.dd.StatusProxy();
21633     }
21634
21635     Roo.dd.DragSource.superclass.constructor.call(this, this.el.dom, this.ddGroup || this.group,
21636           {dragElId : this.proxy.id, resizeFrame: false, isTarget: false, scroll: this.scroll === true});
21637     
21638     this.dragging = false;
21639 };
21640
21641 Roo.extend(Roo.dd.DragSource, Roo.dd.DDProxy, {
21642     /**
21643      * @cfg {String} dropAllowed
21644      * The CSS class returned to the drag source when drop is allowed (defaults to "x-dd-drop-ok").
21645      */
21646     dropAllowed : "x-dd-drop-ok",
21647     /**
21648      * @cfg {String} dropNotAllowed
21649      * The CSS class returned to the drag source when drop is not allowed (defaults to "x-dd-drop-nodrop").
21650      */
21651     dropNotAllowed : "x-dd-drop-nodrop",
21652
21653     /**
21654      * Returns the data object associated with this drag source
21655      * @return {Object} data An object containing arbitrary data
21656      */
21657     getDragData : function(e){
21658         return this.dragData;
21659     },
21660
21661     // private
21662     onDragEnter : function(e, id){
21663         var target = Roo.dd.DragDropMgr.getDDById(id);
21664         this.cachedTarget = target;
21665         if(this.beforeDragEnter(target, e, id) !== false){
21666             if(target.isNotifyTarget){
21667                 var status = target.notifyEnter(this, e, this.dragData);
21668                 this.proxy.setStatus(status);
21669             }else{
21670                 this.proxy.setStatus(this.dropAllowed);
21671             }
21672             
21673             if(this.afterDragEnter){
21674                 /**
21675                  * An empty function by default, but provided so that you can perform a custom action
21676                  * when the dragged item enters the drop target by providing an implementation.
21677                  * @param {Roo.dd.DragDrop} target The drop target
21678                  * @param {Event} e The event object
21679                  * @param {String} id The id of the dragged element
21680                  * @method afterDragEnter
21681                  */
21682                 this.afterDragEnter(target, e, id);
21683             }
21684         }
21685     },
21686
21687     /**
21688      * An empty function by default, but provided so that you can perform a custom action
21689      * before the dragged item enters the drop target and optionally cancel the onDragEnter.
21690      * @param {Roo.dd.DragDrop} target The drop target
21691      * @param {Event} e The event object
21692      * @param {String} id The id of the dragged element
21693      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21694      */
21695     beforeDragEnter : function(target, e, id){
21696         return true;
21697     },
21698
21699     // private
21700     alignElWithMouse: function() {
21701         Roo.dd.DragSource.superclass.alignElWithMouse.apply(this, arguments);
21702         this.proxy.sync();
21703     },
21704
21705     // private
21706     onDragOver : function(e, id){
21707         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21708         if(this.beforeDragOver(target, e, id) !== false){
21709             if(target.isNotifyTarget){
21710                 var status = target.notifyOver(this, e, this.dragData);
21711                 this.proxy.setStatus(status);
21712             }
21713
21714             if(this.afterDragOver){
21715                 /**
21716                  * An empty function by default, but provided so that you can perform a custom action
21717                  * while the dragged item is over the drop target by providing an implementation.
21718                  * @param {Roo.dd.DragDrop} target The drop target
21719                  * @param {Event} e The event object
21720                  * @param {String} id The id of the dragged element
21721                  * @method afterDragOver
21722                  */
21723                 this.afterDragOver(target, e, id);
21724             }
21725         }
21726     },
21727
21728     /**
21729      * An empty function by default, but provided so that you can perform a custom action
21730      * while the dragged item is over the drop target and optionally cancel the onDragOver.
21731      * @param {Roo.dd.DragDrop} target The drop target
21732      * @param {Event} e The event object
21733      * @param {String} id The id of the dragged element
21734      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21735      */
21736     beforeDragOver : function(target, e, id){
21737         return true;
21738     },
21739
21740     // private
21741     onDragOut : function(e, id){
21742         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21743         if(this.beforeDragOut(target, e, id) !== false){
21744             if(target.isNotifyTarget){
21745                 target.notifyOut(this, e, this.dragData);
21746             }
21747             this.proxy.reset();
21748             if(this.afterDragOut){
21749                 /**
21750                  * An empty function by default, but provided so that you can perform a custom action
21751                  * after the dragged item is dragged out of the target without dropping.
21752                  * @param {Roo.dd.DragDrop} target The drop target
21753                  * @param {Event} e The event object
21754                  * @param {String} id The id of the dragged element
21755                  * @method afterDragOut
21756                  */
21757                 this.afterDragOut(target, e, id);
21758             }
21759         }
21760         this.cachedTarget = null;
21761     },
21762
21763     /**
21764      * An empty function by default, but provided so that you can perform a custom action before the dragged
21765      * item is dragged out of the target without dropping, and optionally cancel the onDragOut.
21766      * @param {Roo.dd.DragDrop} target The drop target
21767      * @param {Event} e The event object
21768      * @param {String} id The id of the dragged element
21769      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21770      */
21771     beforeDragOut : function(target, e, id){
21772         return true;
21773     },
21774     
21775     // private
21776     onDragDrop : function(e, id){
21777         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21778         if(this.beforeDragDrop(target, e, id) !== false){
21779             if(target.isNotifyTarget){
21780                 if(target.notifyDrop(this, e, this.dragData)){ // valid drop?
21781                     this.onValidDrop(target, e, id);
21782                 }else{
21783                     this.onInvalidDrop(target, e, id);
21784                 }
21785             }else{
21786                 this.onValidDrop(target, e, id);
21787             }
21788             
21789             if(this.afterDragDrop){
21790                 /**
21791                  * An empty function by default, but provided so that you can perform a custom action
21792                  * after a valid drag drop has occurred by providing an implementation.
21793                  * @param {Roo.dd.DragDrop} target The drop target
21794                  * @param {Event} e The event object
21795                  * @param {String} id The id of the dropped element
21796                  * @method afterDragDrop
21797                  */
21798                 this.afterDragDrop(target, e, id);
21799             }
21800         }
21801         delete this.cachedTarget;
21802     },
21803
21804     /**
21805      * An empty function by default, but provided so that you can perform a custom action before the dragged
21806      * item is dropped onto the target and optionally cancel the onDragDrop.
21807      * @param {Roo.dd.DragDrop} target The drop target
21808      * @param {Event} e The event object
21809      * @param {String} id The id of the dragged element
21810      * @return {Boolean} isValid True if the drag drop event is valid, else false to cancel
21811      */
21812     beforeDragDrop : function(target, e, id){
21813         return true;
21814     },
21815
21816     // private
21817     onValidDrop : function(target, e, id){
21818         this.hideProxy();
21819         if(this.afterValidDrop){
21820             /**
21821              * An empty function by default, but provided so that you can perform a custom action
21822              * after a valid drop has occurred by providing an implementation.
21823              * @param {Object} target The target DD 
21824              * @param {Event} e The event object
21825              * @param {String} id The id of the dropped element
21826              * @method afterInvalidDrop
21827              */
21828             this.afterValidDrop(target, e, id);
21829         }
21830     },
21831
21832     // private
21833     getRepairXY : function(e, data){
21834         return this.el.getXY();  
21835     },
21836
21837     // private
21838     onInvalidDrop : function(target, e, id){
21839         this.beforeInvalidDrop(target, e, id);
21840         if(this.cachedTarget){
21841             if(this.cachedTarget.isNotifyTarget){
21842                 this.cachedTarget.notifyOut(this, e, this.dragData);
21843             }
21844             this.cacheTarget = null;
21845         }
21846         this.proxy.repair(this.getRepairXY(e, this.dragData), this.afterRepair, this);
21847
21848         if(this.afterInvalidDrop){
21849             /**
21850              * An empty function by default, but provided so that you can perform a custom action
21851              * after an invalid drop has occurred by providing an implementation.
21852              * @param {Event} e The event object
21853              * @param {String} id The id of the dropped element
21854              * @method afterInvalidDrop
21855              */
21856             this.afterInvalidDrop(e, id);
21857         }
21858     },
21859
21860     // private
21861     afterRepair : function(){
21862         if(Roo.enableFx){
21863             this.el.highlight(this.hlColor || "c3daf9");
21864         }
21865         this.dragging = false;
21866     },
21867
21868     /**
21869      * An empty function by default, but provided so that you can perform a custom action after an invalid
21870      * drop has occurred.
21871      * @param {Roo.dd.DragDrop} target The drop target
21872      * @param {Event} e The event object
21873      * @param {String} id The id of the dragged element
21874      * @return {Boolean} isValid True if the invalid drop should proceed, else false to cancel
21875      */
21876     beforeInvalidDrop : function(target, e, id){
21877         return true;
21878     },
21879
21880     // private
21881     handleMouseDown : function(e){
21882         if(this.dragging) {
21883             return;
21884         }
21885         var data = this.getDragData(e);
21886         if(data && this.onBeforeDrag(data, e) !== false){
21887             this.dragData = data;
21888             this.proxy.stop();
21889             Roo.dd.DragSource.superclass.handleMouseDown.apply(this, arguments);
21890         } 
21891     },
21892
21893     /**
21894      * An empty function by default, but provided so that you can perform a custom action before the initial
21895      * drag event begins and optionally cancel it.
21896      * @param {Object} data An object containing arbitrary data to be shared with drop targets
21897      * @param {Event} e The event object
21898      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21899      */
21900     onBeforeDrag : function(data, e){
21901         return true;
21902     },
21903
21904     /**
21905      * An empty function by default, but provided so that you can perform a custom action once the initial
21906      * drag event has begun.  The drag cannot be canceled from this function.
21907      * @param {Number} x The x position of the click on the dragged object
21908      * @param {Number} y The y position of the click on the dragged object
21909      */
21910     onStartDrag : Roo.emptyFn,
21911
21912     // private - YUI override
21913     startDrag : function(x, y){
21914         this.proxy.reset();
21915         this.dragging = true;
21916         this.proxy.update("");
21917         this.onInitDrag(x, y);
21918         this.proxy.show();
21919     },
21920
21921     // private
21922     onInitDrag : function(x, y){
21923         var clone = this.el.dom.cloneNode(true);
21924         clone.id = Roo.id(); // prevent duplicate ids
21925         this.proxy.update(clone);
21926         this.onStartDrag(x, y);
21927         return true;
21928     },
21929
21930     /**
21931      * Returns the drag source's underlying {@link Roo.dd.StatusProxy}
21932      * @return {Roo.dd.StatusProxy} proxy The StatusProxy
21933      */
21934     getProxy : function(){
21935         return this.proxy;  
21936     },
21937
21938     /**
21939      * Hides the drag source's {@link Roo.dd.StatusProxy}
21940      */
21941     hideProxy : function(){
21942         this.proxy.hide();  
21943         this.proxy.reset(true);
21944         this.dragging = false;
21945     },
21946
21947     // private
21948     triggerCacheRefresh : function(){
21949         Roo.dd.DDM.refreshCache(this.groups);
21950     },
21951
21952     // private - override to prevent hiding
21953     b4EndDrag: function(e) {
21954     },
21955
21956     // private - override to prevent moving
21957     endDrag : function(e){
21958         this.onEndDrag(this.dragData, e);
21959     },
21960
21961     // private
21962     onEndDrag : function(data, e){
21963     },
21964     
21965     // private - pin to cursor
21966     autoOffset : function(x, y) {
21967         this.setDelta(-12, -20);
21968     }    
21969 });/*
21970  * Based on:
21971  * Ext JS Library 1.1.1
21972  * Copyright(c) 2006-2007, Ext JS, LLC.
21973  *
21974  * Originally Released Under LGPL - original licence link has changed is not relivant.
21975  *
21976  * Fork - LGPL
21977  * <script type="text/javascript">
21978  */
21979
21980
21981 /**
21982  * @class Roo.dd.DropTarget
21983  * @extends Roo.dd.DDTarget
21984  * A simple class that provides the basic implementation needed to make any element a drop target that can have
21985  * draggable items dropped onto it.  The drop has no effect until an implementation of notifyDrop is provided.
21986  * @constructor
21987  * @param {String/HTMLElement/Element} el The container element
21988  * @param {Object} config
21989  */
21990 Roo.dd.DropTarget = function(el, config){
21991     this.el = Roo.get(el);
21992     
21993     var listeners = false; ;
21994     if (config && config.listeners) {
21995         listeners= config.listeners;
21996         delete config.listeners;
21997     }
21998     Roo.apply(this, config);
21999     
22000     if(this.containerScroll){
22001         Roo.dd.ScrollManager.register(this.el);
22002     }
22003     this.addEvents( {
22004          /**
22005          * @scope Roo.dd.DropTarget
22006          */
22007          
22008          /**
22009          * @event enter
22010          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the source is now over the
22011          * target.  This default implementation adds the CSS class specified by overClass (if any) to the drop element
22012          * and returns the dropAllowed config value.  This method should be overridden if drop validation is required.
22013          * 
22014          * IMPORTANT : it should set this.overClass and this.dropAllowed
22015          * 
22016          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22017          * @param {Event} e The event
22018          * @param {Object} data An object containing arbitrary data supplied by the drag source
22019          */
22020         "enter" : true,
22021         
22022          /**
22023          * @event over
22024          * The function a {@link Roo.dd.DragSource} calls continuously while it is being dragged over the target.
22025          * This method will be called on every mouse movement while the drag source is over the drop target.
22026          * This default implementation simply returns the dropAllowed config value.
22027          * 
22028          * IMPORTANT : it should set this.dropAllowed
22029          * 
22030          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22031          * @param {Event} e The event
22032          * @param {Object} data An object containing arbitrary data supplied by the drag source
22033          
22034          */
22035         "over" : true,
22036         /**
22037          * @event out
22038          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the source has been dragged
22039          * out of the target without dropping.  This default implementation simply removes the CSS class specified by
22040          * overClass (if any) from the drop element.
22041          * 
22042          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22043          * @param {Event} e The event
22044          * @param {Object} data An object containing arbitrary data supplied by the drag source
22045          */
22046          "out" : true,
22047          
22048         /**
22049          * @event drop
22050          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the dragged item has
22051          * been dropped on it.  This method has no default implementation and returns false, so you must provide an
22052          * implementation that does something to process the drop event and returns true so that the drag source's
22053          * repair action does not run.
22054          * 
22055          * IMPORTANT : it should set this.success
22056          * 
22057          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22058          * @param {Event} e The event
22059          * @param {Object} data An object containing arbitrary data supplied by the drag source
22060         */
22061          "drop" : true
22062     });
22063             
22064      
22065     Roo.dd.DropTarget.superclass.constructor.call(  this, 
22066         this.el.dom, 
22067         this.ddGroup || this.group,
22068         {
22069             isTarget: true,
22070             listeners : listeners || {} 
22071            
22072         
22073         }
22074     );
22075
22076 };
22077
22078 Roo.extend(Roo.dd.DropTarget, Roo.dd.DDTarget, {
22079     /**
22080      * @cfg {String} overClass
22081      * The CSS class applied to the drop target element while the drag source is over it (defaults to "").
22082      */
22083      /**
22084      * @cfg {String} ddGroup
22085      * The drag drop group to handle drop events for
22086      */
22087      
22088     /**
22089      * @cfg {String} dropAllowed
22090      * The CSS class returned to the drag source when drop is allowed (defaults to "x-dd-drop-ok").
22091      */
22092     dropAllowed : "x-dd-drop-ok",
22093     /**
22094      * @cfg {String} dropNotAllowed
22095      * The CSS class returned to the drag source when drop is not allowed (defaults to "x-dd-drop-nodrop").
22096      */
22097     dropNotAllowed : "x-dd-drop-nodrop",
22098     /**
22099      * @cfg {boolean} success
22100      * set this after drop listener.. 
22101      */
22102     success : false,
22103     /**
22104      * @cfg {boolean|String} valid true/false or string (ok-add/ok-sub/ok/nodrop)
22105      * if the drop point is valid for over/enter..
22106      */
22107     valid : false,
22108     // private
22109     isTarget : true,
22110
22111     // private
22112     isNotifyTarget : true,
22113     
22114     /**
22115      * @hide
22116      */
22117     notifyEnter : function(dd, e, data)
22118     {
22119         this.valid = true;
22120         this.fireEvent('enter', dd, e, data);
22121         if(this.overClass){
22122             this.el.addClass(this.overClass);
22123         }
22124         return typeof(this.valid) == 'string' ? 'x-dd-drop-' + this.valid : (
22125             this.valid ? this.dropAllowed : this.dropNotAllowed
22126         );
22127     },
22128
22129     /**
22130      * @hide
22131      */
22132     notifyOver : function(dd, e, data)
22133     {
22134         this.valid = true;
22135         this.fireEvent('over', dd, e, data);
22136         return typeof(this.valid) == 'string' ? 'x-dd-drop-' + this.valid : (
22137             this.valid ? this.dropAllowed : this.dropNotAllowed
22138         );
22139     },
22140
22141     /**
22142      * @hide
22143      */
22144     notifyOut : function(dd, e, data)
22145     {
22146         this.fireEvent('out', dd, e, data);
22147         if(this.overClass){
22148             this.el.removeClass(this.overClass);
22149         }
22150     },
22151
22152     /**
22153      * @hide
22154      */
22155     notifyDrop : function(dd, e, data)
22156     {
22157         this.success = false;
22158         this.fireEvent('drop', dd, e, data);
22159         return this.success;
22160     }
22161 });/*
22162  * Based on:
22163  * Ext JS Library 1.1.1
22164  * Copyright(c) 2006-2007, Ext JS, LLC.
22165  *
22166  * Originally Released Under LGPL - original licence link has changed is not relivant.
22167  *
22168  * Fork - LGPL
22169  * <script type="text/javascript">
22170  */
22171
22172
22173 /**
22174  * @class Roo.dd.DragZone
22175  * @extends Roo.dd.DragSource
22176  * This class provides a container DD instance that proxies for multiple child node sources.<br />
22177  * By default, this class requires that draggable child nodes are registered with {@link Roo.dd.Registry}.
22178  * @constructor
22179  * @param {String/HTMLElement/Element} el The container element
22180  * @param {Object} config
22181  */
22182 Roo.dd.DragZone = function(el, config){
22183     Roo.dd.DragZone.superclass.constructor.call(this, el, config);
22184     if(this.containerScroll){
22185         Roo.dd.ScrollManager.register(this.el);
22186     }
22187 };
22188
22189 Roo.extend(Roo.dd.DragZone, Roo.dd.DragSource, {
22190     /**
22191      * @cfg {Boolean} containerScroll True to register this container with the Scrollmanager
22192      * for auto scrolling during drag operations.
22193      */
22194     /**
22195      * @cfg {String} hlColor The color to use when visually highlighting the drag source in the afterRepair
22196      * method after a failed drop (defaults to "c3daf9" - light blue)
22197      */
22198
22199     /**
22200      * Called when a mousedown occurs in this container. Looks in {@link Roo.dd.Registry}
22201      * for a valid target to drag based on the mouse down. Override this method
22202      * to provide your own lookup logic (e.g. finding a child by class name). Make sure your returned
22203      * object has a "ddel" attribute (with an HTML Element) for other functions to work.
22204      * @param {EventObject} e The mouse down event
22205      * @return {Object} The dragData
22206      */
22207     getDragData : function(e){
22208         return Roo.dd.Registry.getHandleFromEvent(e);
22209     },
22210     
22211     /**
22212      * Called once drag threshold has been reached to initialize the proxy element. By default, it clones the
22213      * this.dragData.ddel
22214      * @param {Number} x The x position of the click on the dragged object
22215      * @param {Number} y The y position of the click on the dragged object
22216      * @return {Boolean} true to continue the drag, false to cancel
22217      */
22218     onInitDrag : function(x, y){
22219         this.proxy.update(this.dragData.ddel.cloneNode(true));
22220         this.onStartDrag(x, y);
22221         return true;
22222     },
22223     
22224     /**
22225      * Called after a repair of an invalid drop. By default, highlights this.dragData.ddel 
22226      */
22227     afterRepair : function(){
22228         if(Roo.enableFx){
22229             Roo.Element.fly(this.dragData.ddel).highlight(this.hlColor || "c3daf9");
22230         }
22231         this.dragging = false;
22232     },
22233
22234     /**
22235      * Called before a repair of an invalid drop to get the XY to animate to. By default returns
22236      * the XY of this.dragData.ddel
22237      * @param {EventObject} e The mouse up event
22238      * @return {Array} The xy location (e.g. [100, 200])
22239      */
22240     getRepairXY : function(e){
22241         return Roo.Element.fly(this.dragData.ddel).getXY();  
22242     }
22243 });/*
22244  * Based on:
22245  * Ext JS Library 1.1.1
22246  * Copyright(c) 2006-2007, Ext JS, LLC.
22247  *
22248  * Originally Released Under LGPL - original licence link has changed is not relivant.
22249  *
22250  * Fork - LGPL
22251  * <script type="text/javascript">
22252  */
22253 /**
22254  * @class Roo.dd.DropZone
22255  * @extends Roo.dd.DropTarget
22256  * This class provides a container DD instance that proxies for multiple child node targets.<br />
22257  * By default, this class requires that child nodes accepting drop are registered with {@link Roo.dd.Registry}.
22258  * @constructor
22259  * @param {String/HTMLElement/Element} el The container element
22260  * @param {Object} config
22261  */
22262 Roo.dd.DropZone = function(el, config){
22263     Roo.dd.DropZone.superclass.constructor.call(this, el, config);
22264 };
22265
22266 Roo.extend(Roo.dd.DropZone, Roo.dd.DropTarget, {
22267     /**
22268      * Returns a custom data object associated with the DOM node that is the target of the event.  By default
22269      * this looks up the event target in the {@link Roo.dd.Registry}, although you can override this method to
22270      * provide your own custom lookup.
22271      * @param {Event} e The event
22272      * @return {Object} data The custom data
22273      */
22274     getTargetFromEvent : function(e){
22275         return Roo.dd.Registry.getTargetFromEvent(e);
22276     },
22277
22278     /**
22279      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has entered a drop node
22280      * that it has registered.  This method has no default implementation and should be overridden to provide
22281      * node-specific processing if necessary.
22282      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from 
22283      * {@link #getTargetFromEvent} for this node)
22284      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22285      * @param {Event} e The event
22286      * @param {Object} data An object containing arbitrary data supplied by the drag source
22287      */
22288     onNodeEnter : function(n, dd, e, data){
22289         
22290     },
22291
22292     /**
22293      * Called internally while the DropZone determines that a {@link Roo.dd.DragSource} is over a drop node
22294      * that it has registered.  The default implementation returns this.dropNotAllowed, so it should be
22295      * overridden to provide the proper feedback.
22296      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22297      * {@link #getTargetFromEvent} for this node)
22298      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22299      * @param {Event} e The event
22300      * @param {Object} data An object containing arbitrary data supplied by the drag source
22301      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22302      * underlying {@link Roo.dd.StatusProxy} can be updated
22303      */
22304     onNodeOver : function(n, dd, e, data){
22305         return this.dropAllowed;
22306     },
22307
22308     /**
22309      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dragged out of
22310      * the drop node without dropping.  This method has no default implementation and should be overridden to provide
22311      * node-specific processing if necessary.
22312      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22313      * {@link #getTargetFromEvent} for this node)
22314      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22315      * @param {Event} e The event
22316      * @param {Object} data An object containing arbitrary data supplied by the drag source
22317      */
22318     onNodeOut : function(n, dd, e, data){
22319         
22320     },
22321
22322     /**
22323      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dropped onto
22324      * the drop node.  The default implementation returns false, so it should be overridden to provide the
22325      * appropriate processing of the drop event and return true so that the drag source's repair action does not run.
22326      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22327      * {@link #getTargetFromEvent} for this node)
22328      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22329      * @param {Event} e The event
22330      * @param {Object} data An object containing arbitrary data supplied by the drag source
22331      * @return {Boolean} True if the drop was valid, else false
22332      */
22333     onNodeDrop : function(n, dd, e, data){
22334         return false;
22335     },
22336
22337     /**
22338      * Called internally while the DropZone determines that a {@link Roo.dd.DragSource} is being dragged over it,
22339      * but not over any of its registered drop nodes.  The default implementation returns this.dropNotAllowed, so
22340      * it should be overridden to provide the proper feedback if necessary.
22341      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22342      * @param {Event} e The event
22343      * @param {Object} data An object containing arbitrary data supplied by the drag source
22344      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22345      * underlying {@link Roo.dd.StatusProxy} can be updated
22346      */
22347     onContainerOver : function(dd, e, data){
22348         return this.dropNotAllowed;
22349     },
22350
22351     /**
22352      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dropped on it,
22353      * but not on any of its registered drop nodes.  The default implementation returns false, so it should be
22354      * overridden to provide the appropriate processing of the drop event if you need the drop zone itself to
22355      * be able to accept drops.  It should return true when valid so that the drag source's repair action does not run.
22356      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22357      * @param {Event} e The event
22358      * @param {Object} data An object containing arbitrary data supplied by the drag source
22359      * @return {Boolean} True if the drop was valid, else false
22360      */
22361     onContainerDrop : function(dd, e, data){
22362         return false;
22363     },
22364
22365     /**
22366      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the source is now over
22367      * the zone.  The default implementation returns this.dropNotAllowed and expects that only registered drop
22368      * nodes can process drag drop operations, so if you need the drop zone itself to be able to process drops
22369      * you should override this method and provide a custom implementation.
22370      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22371      * @param {Event} e The event
22372      * @param {Object} data An object containing arbitrary data supplied by the drag source
22373      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22374      * underlying {@link Roo.dd.StatusProxy} can be updated
22375      */
22376     notifyEnter : function(dd, e, data){
22377         return this.dropNotAllowed;
22378     },
22379
22380     /**
22381      * The function a {@link Roo.dd.DragSource} calls continuously while it is being dragged over the drop zone.
22382      * This method will be called on every mouse movement while the drag source is over the drop zone.
22383      * It will call {@link #onNodeOver} while the drag source is over a registered node, and will also automatically
22384      * delegate to the appropriate node-specific methods as necessary when the drag source enters and exits
22385      * registered nodes ({@link #onNodeEnter}, {@link #onNodeOut}). If the drag source is not currently over a
22386      * registered node, it will call {@link #onContainerOver}.
22387      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22388      * @param {Event} e The event
22389      * @param {Object} data An object containing arbitrary data supplied by the drag source
22390      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22391      * underlying {@link Roo.dd.StatusProxy} can be updated
22392      */
22393     notifyOver : function(dd, e, data){
22394         var n = this.getTargetFromEvent(e);
22395         if(!n){ // not over valid drop target
22396             if(this.lastOverNode){
22397                 this.onNodeOut(this.lastOverNode, dd, e, data);
22398                 this.lastOverNode = null;
22399             }
22400             return this.onContainerOver(dd, e, data);
22401         }
22402         if(this.lastOverNode != n){
22403             if(this.lastOverNode){
22404                 this.onNodeOut(this.lastOverNode, dd, e, data);
22405             }
22406             this.onNodeEnter(n, dd, e, data);
22407             this.lastOverNode = n;
22408         }
22409         return this.onNodeOver(n, dd, e, data);
22410     },
22411
22412     /**
22413      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the source has been dragged
22414      * out of the zone without dropping.  If the drag source is currently over a registered node, the notification
22415      * will be delegated to {@link #onNodeOut} for node-specific handling, otherwise it will be ignored.
22416      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22417      * @param {Event} e The event
22418      * @param {Object} data An object containing arbitrary data supplied by the drag zone
22419      */
22420     notifyOut : function(dd, e, data){
22421         if(this.lastOverNode){
22422             this.onNodeOut(this.lastOverNode, dd, e, data);
22423             this.lastOverNode = null;
22424         }
22425     },
22426
22427     /**
22428      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the dragged item has
22429      * been dropped on it.  The drag zone will look up the target node based on the event passed in, and if there
22430      * is a node registered for that event, it will delegate to {@link #onNodeDrop} for node-specific handling,
22431      * otherwise it will call {@link #onContainerDrop}.
22432      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22433      * @param {Event} e The event
22434      * @param {Object} data An object containing arbitrary data supplied by the drag source
22435      * @return {Boolean} True if the drop was valid, else false
22436      */
22437     notifyDrop : function(dd, e, data){
22438         if(this.lastOverNode){
22439             this.onNodeOut(this.lastOverNode, dd, e, data);
22440             this.lastOverNode = null;
22441         }
22442         var n = this.getTargetFromEvent(e);
22443         return n ?
22444             this.onNodeDrop(n, dd, e, data) :
22445             this.onContainerDrop(dd, e, data);
22446     },
22447
22448     // private
22449     triggerCacheRefresh : function(){
22450         Roo.dd.DDM.refreshCache(this.groups);
22451     }  
22452 });/*
22453  * Based on:
22454  * Ext JS Library 1.1.1
22455  * Copyright(c) 2006-2007, Ext JS, LLC.
22456  *
22457  * Originally Released Under LGPL - original licence link has changed is not relivant.
22458  *
22459  * Fork - LGPL
22460  * <script type="text/javascript">
22461  */
22462
22463
22464 /**
22465  * @class Roo.data.SortTypes
22466  * @singleton
22467  * Defines the default sorting (casting?) comparison functions used when sorting data.
22468  */
22469 Roo.data.SortTypes = {
22470     /**
22471      * Default sort that does nothing
22472      * @param {Mixed} s The value being converted
22473      * @return {Mixed} The comparison value
22474      */
22475     none : function(s){
22476         return s;
22477     },
22478     
22479     /**
22480      * The regular expression used to strip tags
22481      * @type {RegExp}
22482      * @property
22483      */
22484     stripTagsRE : /<\/?[^>]+>/gi,
22485     
22486     /**
22487      * Strips all HTML tags to sort on text only
22488      * @param {Mixed} s The value being converted
22489      * @return {String} The comparison value
22490      */
22491     asText : function(s){
22492         return String(s).replace(this.stripTagsRE, "");
22493     },
22494     
22495     /**
22496      * Strips all HTML tags to sort on text only - Case insensitive
22497      * @param {Mixed} s The value being converted
22498      * @return {String} The comparison value
22499      */
22500     asUCText : function(s){
22501         return String(s).toUpperCase().replace(this.stripTagsRE, "");
22502     },
22503     
22504     /**
22505      * Case insensitive string
22506      * @param {Mixed} s The value being converted
22507      * @return {String} The comparison value
22508      */
22509     asUCString : function(s) {
22510         return String(s).toUpperCase();
22511     },
22512     
22513     /**
22514      * Date sorting
22515      * @param {Mixed} s The value being converted
22516      * @return {Number} The comparison value
22517      */
22518     asDate : function(s) {
22519         if(!s){
22520             return 0;
22521         }
22522         if(s instanceof Date){
22523             return s.getTime();
22524         }
22525         return Date.parse(String(s));
22526     },
22527     
22528     /**
22529      * Float sorting
22530      * @param {Mixed} s The value being converted
22531      * @return {Float} The comparison value
22532      */
22533     asFloat : function(s) {
22534         var val = parseFloat(String(s).replace(/,/g, ""));
22535         if(isNaN(val)) {
22536             val = 0;
22537         }
22538         return val;
22539     },
22540     
22541     /**
22542      * Integer sorting
22543      * @param {Mixed} s The value being converted
22544      * @return {Number} The comparison value
22545      */
22546     asInt : function(s) {
22547         var val = parseInt(String(s).replace(/,/g, ""));
22548         if(isNaN(val)) {
22549             val = 0;
22550         }
22551         return val;
22552     }
22553 };/*
22554  * Based on:
22555  * Ext JS Library 1.1.1
22556  * Copyright(c) 2006-2007, Ext JS, LLC.
22557  *
22558  * Originally Released Under LGPL - original licence link has changed is not relivant.
22559  *
22560  * Fork - LGPL
22561  * <script type="text/javascript">
22562  */
22563
22564 /**
22565 * @class Roo.data.Record
22566  * Instances of this class encapsulate both record <em>definition</em> information, and record
22567  * <em>value</em> information for use in {@link Roo.data.Store} objects, or any code which needs
22568  * to access Records cached in an {@link Roo.data.Store} object.<br>
22569  * <p>
22570  * Constructors for this class are generated by passing an Array of field definition objects to {@link #create}.
22571  * Instances are usually only created by {@link Roo.data.Reader} implementations when processing unformatted data
22572  * objects.<br>
22573  * <p>
22574  * Record objects generated by this constructor inherit all the methods of Roo.data.Record listed below.
22575  * @constructor
22576  * This constructor should not be used to create Record objects. Instead, use the constructor generated by
22577  * {@link #create}. The parameters are the same.
22578  * @param {Array} data An associative Array of data values keyed by the field name.
22579  * @param {Object} id (Optional) The id of the record. This id should be unique, and is used by the
22580  * {@link Roo.data.Store} object which owns the Record to index its collection of Records. If
22581  * not specified an integer id is generated.
22582  */
22583 Roo.data.Record = function(data, id){
22584     this.id = (id || id === 0) ? id : ++Roo.data.Record.AUTO_ID;
22585     this.data = data;
22586 };
22587
22588 /**
22589  * Generate a constructor for a specific record layout.
22590  * @param {Array} o An Array of field definition objects which specify field names, and optionally,
22591  * data types, and a mapping for an {@link Roo.data.Reader} to extract the field's value from a data object.
22592  * Each field definition object may contain the following properties: <ul>
22593  * <li><b>name</b> : String<p style="margin-left:1em">The name by which the field is referenced within the Record. This is referenced by,
22594  * for example the <em>dataIndex</em> property in column definition objects passed to {@link Roo.grid.ColumnModel}</p></li>
22595  * <li><b>mapping</b> : String<p style="margin-left:1em">(Optional) A path specification for use by the {@link Roo.data.Reader} implementation
22596  * that is creating the Record to access the data value from the data object. If an {@link Roo.data.JsonReader}
22597  * is being used, then this is a string containing the javascript expression to reference the data relative to 
22598  * the record item's root. If an {@link Roo.data.XmlReader} is being used, this is an {@link Roo.DomQuery} path
22599  * to the data item relative to the record element. If the mapping expression is the same as the field name,
22600  * this may be omitted.</p></li>
22601  * <li><b>type</b> : String<p style="margin-left:1em">(Optional) The data type for conversion to displayable value. Possible values are
22602  * <ul><li>auto (Default, implies no conversion)</li>
22603  * <li>string</li>
22604  * <li>int</li>
22605  * <li>float</li>
22606  * <li>boolean</li>
22607  * <li>date</li></ul></p></li>
22608  * <li><b>sortType</b> : Mixed<p style="margin-left:1em">(Optional) A member of {@link Roo.data.SortTypes}.</p></li>
22609  * <li><b>sortDir</b> : String<p style="margin-left:1em">(Optional) Initial direction to sort. "ASC" or "DESC"</p></li>
22610  * <li><b>convert</b> : Function<p style="margin-left:1em">(Optional) A function which converts the value provided
22611  * by the Reader into an object that will be stored in the Record. It is passed the
22612  * following parameters:<ul>
22613  * <li><b>v</b> : Mixed<p style="margin-left:1em">The data value as read by the Reader.</p></li>
22614  * </ul></p></li>
22615  * <li><b>dateFormat</b> : String<p style="margin-left:1em">(Optional) A format String for the Date.parseDate function.</p></li>
22616  * </ul>
22617  * <br>usage:<br><pre><code>
22618 var TopicRecord = Roo.data.Record.create(
22619     {name: 'title', mapping: 'topic_title'},
22620     {name: 'author', mapping: 'username'},
22621     {name: 'totalPosts', mapping: 'topic_replies', type: 'int'},
22622     {name: 'lastPost', mapping: 'post_time', type: 'date'},
22623     {name: 'lastPoster', mapping: 'user2'},
22624     {name: 'excerpt', mapping: 'post_text'}
22625 );
22626
22627 var myNewRecord = new TopicRecord({
22628     title: 'Do my job please',
22629     author: 'noobie',
22630     totalPosts: 1,
22631     lastPost: new Date(),
22632     lastPoster: 'Animal',
22633     excerpt: 'No way dude!'
22634 });
22635 myStore.add(myNewRecord);
22636 </code></pre>
22637  * @method create
22638  * @static
22639  */
22640 Roo.data.Record.create = function(o){
22641     var f = function(){
22642         f.superclass.constructor.apply(this, arguments);
22643     };
22644     Roo.extend(f, Roo.data.Record);
22645     var p = f.prototype;
22646     p.fields = new Roo.util.MixedCollection(false, function(field){
22647         return field.name;
22648     });
22649     for(var i = 0, len = o.length; i < len; i++){
22650         p.fields.add(new Roo.data.Field(o[i]));
22651     }
22652     f.getField = function(name){
22653         return p.fields.get(name);  
22654     };
22655     return f;
22656 };
22657
22658 Roo.data.Record.AUTO_ID = 1000;
22659 Roo.data.Record.EDIT = 'edit';
22660 Roo.data.Record.REJECT = 'reject';
22661 Roo.data.Record.COMMIT = 'commit';
22662
22663 Roo.data.Record.prototype = {
22664     /**
22665      * Readonly flag - true if this record has been modified.
22666      * @type Boolean
22667      */
22668     dirty : false,
22669     editing : false,
22670     error: null,
22671     modified: null,
22672
22673     // private
22674     join : function(store){
22675         this.store = store;
22676     },
22677
22678     /**
22679      * Set the named field to the specified value.
22680      * @param {String} name The name of the field to set.
22681      * @param {Object} value The value to set the field to.
22682      */
22683     set : function(name, value){
22684         if(this.data[name] == value){
22685             return;
22686         }
22687         this.dirty = true;
22688         if(!this.modified){
22689             this.modified = {};
22690         }
22691         if(typeof this.modified[name] == 'undefined'){
22692             this.modified[name] = this.data[name];
22693         }
22694         this.data[name] = value;
22695         if(!this.editing && this.store){
22696             this.store.afterEdit(this);
22697         }       
22698     },
22699
22700     /**
22701      * Get the value of the named field.
22702      * @param {String} name The name of the field to get the value of.
22703      * @return {Object} The value of the field.
22704      */
22705     get : function(name){
22706         return this.data[name]; 
22707     },
22708
22709     // private
22710     beginEdit : function(){
22711         this.editing = true;
22712         this.modified = {}; 
22713     },
22714
22715     // private
22716     cancelEdit : function(){
22717         this.editing = false;
22718         delete this.modified;
22719     },
22720
22721     // private
22722     endEdit : function(){
22723         this.editing = false;
22724         if(this.dirty && this.store){
22725             this.store.afterEdit(this);
22726         }
22727     },
22728
22729     /**
22730      * Usually called by the {@link Roo.data.Store} which owns the Record.
22731      * Rejects all changes made to the Record since either creation, or the last commit operation.
22732      * Modified fields are reverted to their original values.
22733      * <p>
22734      * Developers should subscribe to the {@link Roo.data.Store#update} event to have their code notified
22735      * of reject operations.
22736      */
22737     reject : function(){
22738         var m = this.modified;
22739         for(var n in m){
22740             if(typeof m[n] != "function"){
22741                 this.data[n] = m[n];
22742             }
22743         }
22744         this.dirty = false;
22745         delete this.modified;
22746         this.editing = false;
22747         if(this.store){
22748             this.store.afterReject(this);
22749         }
22750     },
22751
22752     /**
22753      * Usually called by the {@link Roo.data.Store} which owns the Record.
22754      * Commits all changes made to the Record since either creation, or the last commit operation.
22755      * <p>
22756      * Developers should subscribe to the {@link Roo.data.Store#update} event to have their code notified
22757      * of commit operations.
22758      */
22759     commit : function(){
22760         this.dirty = false;
22761         delete this.modified;
22762         this.editing = false;
22763         if(this.store){
22764             this.store.afterCommit(this);
22765         }
22766     },
22767
22768     // private
22769     hasError : function(){
22770         return this.error != null;
22771     },
22772
22773     // private
22774     clearError : function(){
22775         this.error = null;
22776     },
22777
22778     /**
22779      * Creates a copy of this record.
22780      * @param {String} id (optional) A new record id if you don't want to use this record's id
22781      * @return {Record}
22782      */
22783     copy : function(newId) {
22784         return new this.constructor(Roo.apply({}, this.data), newId || this.id);
22785     }
22786 };/*
22787  * Based on:
22788  * Ext JS Library 1.1.1
22789  * Copyright(c) 2006-2007, Ext JS, LLC.
22790  *
22791  * Originally Released Under LGPL - original licence link has changed is not relivant.
22792  *
22793  * Fork - LGPL
22794  * <script type="text/javascript">
22795  */
22796
22797
22798
22799 /**
22800  * @class Roo.data.Store
22801  * @extends Roo.util.Observable
22802  * The Store class encapsulates a client side cache of {@link Roo.data.Record} objects which provide input data
22803  * for widgets such as the Roo.grid.Grid, or the Roo.form.ComboBox.<br>
22804  * <p>
22805  * A Store object uses an implementation of {@link Roo.data.DataProxy} to access a data object unless you call loadData() directly and pass in your data. The Store object
22806  * has no knowledge of the format of the data returned by the Proxy.<br>
22807  * <p>
22808  * A Store object uses its configured implementation of {@link Roo.data.DataReader} to create {@link Roo.data.Record}
22809  * instances from the data object. These records are cached and made available through accessor functions.
22810  * @constructor
22811  * Creates a new Store.
22812  * @param {Object} config A config object containing the objects needed for the Store to access data,
22813  * and read the data into Records.
22814  */
22815 Roo.data.Store = function(config){
22816     this.data = new Roo.util.MixedCollection(false);
22817     this.data.getKey = function(o){
22818         return o.id;
22819     };
22820     this.baseParams = {};
22821     // private
22822     this.paramNames = {
22823         "start" : "start",
22824         "limit" : "limit",
22825         "sort" : "sort",
22826         "dir" : "dir",
22827         "multisort" : "_multisort"
22828     };
22829
22830     if(config && config.data){
22831         this.inlineData = config.data;
22832         delete config.data;
22833     }
22834
22835     Roo.apply(this, config);
22836     
22837     if(this.reader){ // reader passed
22838         this.reader = Roo.factory(this.reader, Roo.data);
22839         this.reader.xmodule = this.xmodule || false;
22840         if(!this.recordType){
22841             this.recordType = this.reader.recordType;
22842         }
22843         if(this.reader.onMetaChange){
22844             this.reader.onMetaChange = this.onMetaChange.createDelegate(this);
22845         }
22846     }
22847
22848     if(this.recordType){
22849         this.fields = this.recordType.prototype.fields;
22850     }
22851     this.modified = [];
22852
22853     this.addEvents({
22854         /**
22855          * @event datachanged
22856          * Fires when the data cache has changed, and a widget which is using this Store
22857          * as a Record cache should refresh its view.
22858          * @param {Store} this
22859          */
22860         datachanged : true,
22861         /**
22862          * @event metachange
22863          * Fires when this store's reader provides new metadata (fields). This is currently only support for JsonReaders.
22864          * @param {Store} this
22865          * @param {Object} meta The JSON metadata
22866          */
22867         metachange : true,
22868         /**
22869          * @event add
22870          * Fires when Records have been added to the Store
22871          * @param {Store} this
22872          * @param {Roo.data.Record[]} records The array of Records added
22873          * @param {Number} index The index at which the record(s) were added
22874          */
22875         add : true,
22876         /**
22877          * @event remove
22878          * Fires when a Record has been removed from the Store
22879          * @param {Store} this
22880          * @param {Roo.data.Record} record The Record that was removed
22881          * @param {Number} index The index at which the record was removed
22882          */
22883         remove : true,
22884         /**
22885          * @event update
22886          * Fires when a Record has been updated
22887          * @param {Store} this
22888          * @param {Roo.data.Record} record The Record that was updated
22889          * @param {String} operation The update operation being performed.  Value may be one of:
22890          * <pre><code>
22891  Roo.data.Record.EDIT
22892  Roo.data.Record.REJECT
22893  Roo.data.Record.COMMIT
22894          * </code></pre>
22895          */
22896         update : true,
22897         /**
22898          * @event clear
22899          * Fires when the data cache has been cleared.
22900          * @param {Store} this
22901          */
22902         clear : true,
22903         /**
22904          * @event beforeload
22905          * Fires before a request is made for a new data object.  If the beforeload handler returns false
22906          * the load action will be canceled.
22907          * @param {Store} this
22908          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22909          */
22910         beforeload : true,
22911         /**
22912          * @event beforeloadadd
22913          * Fires after a new set of Records has been loaded.
22914          * @param {Store} this
22915          * @param {Roo.data.Record[]} records The Records that were loaded
22916          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22917          */
22918         beforeloadadd : true,
22919         /**
22920          * @event load
22921          * Fires after a new set of Records has been loaded, before they are added to the store.
22922          * @param {Store} this
22923          * @param {Roo.data.Record[]} records The Records that were loaded
22924          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22925          * @params {Object} return from reader
22926          */
22927         load : true,
22928         /**
22929          * @event loadexception
22930          * Fires if an exception occurs in the Proxy during loading.
22931          * Called with the signature of the Proxy's "loadexception" event.
22932          * If you return Json { data: [] , success: false, .... } then this will be thrown with the following args
22933          * 
22934          * @param {Proxy} 
22935          * @param {Object} return from JsonData.reader() - success, totalRecords, records
22936          * @param {Object} load options 
22937          * @param {Object} jsonData from your request (normally this contains the Exception)
22938          */
22939         loadexception : true
22940     });
22941     
22942     if(this.proxy){
22943         this.proxy = Roo.factory(this.proxy, Roo.data);
22944         this.proxy.xmodule = this.xmodule || false;
22945         this.relayEvents(this.proxy,  ["loadexception"]);
22946     }
22947     this.sortToggle = {};
22948     this.sortOrder = []; // array of order of sorting - updated by grid if multisort is enabled.
22949
22950     Roo.data.Store.superclass.constructor.call(this);
22951
22952     if(this.inlineData){
22953         this.loadData(this.inlineData);
22954         delete this.inlineData;
22955     }
22956 };
22957
22958 Roo.extend(Roo.data.Store, Roo.util.Observable, {
22959      /**
22960     * @cfg {boolean} isLocal   flag if data is locally available (and can be always looked up
22961     * without a remote query - used by combo/forms at present.
22962     */
22963     
22964     /**
22965     * @cfg {Roo.data.DataProxy} proxy The Proxy object which provides access to a data object.
22966     */
22967     /**
22968     * @cfg {Array} data Inline data to be loaded when the store is initialized.
22969     */
22970     /**
22971     * @cfg {Roo.data.Reader} reader The Reader object which processes the data object and returns
22972     * an Array of Roo.data.record objects which are cached keyed by their <em>id</em> property.
22973     */
22974     /**
22975     * @cfg {Object} baseParams An object containing properties which are to be sent as parameters
22976     * on any HTTP request
22977     */
22978     /**
22979     * @cfg {Object} sortInfo A config object in the format: {field: "fieldName", direction: "ASC|DESC"}
22980     */
22981     /**
22982     * @cfg {Boolean} multiSort enable multi column sorting (sort is based on the order of columns, remote only at present)
22983     */
22984     multiSort: false,
22985     /**
22986     * @cfg {boolean} remoteSort True if sorting is to be handled by requesting the Proxy to provide a refreshed
22987     * version of the data object in sorted order, as opposed to sorting the Record cache in place (defaults to false).
22988     */
22989     remoteSort : false,
22990
22991     /**
22992     * @cfg {boolean} pruneModifiedRecords True to clear all modified record information each time the store is
22993      * loaded or when a record is removed. (defaults to false).
22994     */
22995     pruneModifiedRecords : false,
22996
22997     // private
22998     lastOptions : null,
22999
23000     /**
23001      * Add Records to the Store and fires the add event.
23002      * @param {Roo.data.Record[]} records An Array of Roo.data.Record objects to add to the cache.
23003      */
23004     add : function(records){
23005         records = [].concat(records);
23006         for(var i = 0, len = records.length; i < len; i++){
23007             records[i].join(this);
23008         }
23009         var index = this.data.length;
23010         this.data.addAll(records);
23011         this.fireEvent("add", this, records, index);
23012     },
23013
23014     /**
23015      * Remove a Record from the Store and fires the remove event.
23016      * @param {Ext.data.Record} record The Roo.data.Record object to remove from the cache.
23017      */
23018     remove : function(record){
23019         var index = this.data.indexOf(record);
23020         this.data.removeAt(index);
23021  
23022         if(this.pruneModifiedRecords){
23023             this.modified.remove(record);
23024         }
23025         this.fireEvent("remove", this, record, index);
23026     },
23027
23028     /**
23029      * Remove all Records from the Store and fires the clear event.
23030      */
23031     removeAll : function(){
23032         this.data.clear();
23033         if(this.pruneModifiedRecords){
23034             this.modified = [];
23035         }
23036         this.fireEvent("clear", this);
23037     },
23038
23039     /**
23040      * Inserts Records to the Store at the given index and fires the add event.
23041      * @param {Number} index The start index at which to insert the passed Records.
23042      * @param {Roo.data.Record[]} records An Array of Roo.data.Record objects to add to the cache.
23043      */
23044     insert : function(index, records){
23045         records = [].concat(records);
23046         for(var i = 0, len = records.length; i < len; i++){
23047             this.data.insert(index, records[i]);
23048             records[i].join(this);
23049         }
23050         this.fireEvent("add", this, records, index);
23051     },
23052
23053     /**
23054      * Get the index within the cache of the passed Record.
23055      * @param {Roo.data.Record} record The Roo.data.Record object to to find.
23056      * @return {Number} The index of the passed Record. Returns -1 if not found.
23057      */
23058     indexOf : function(record){
23059         return this.data.indexOf(record);
23060     },
23061
23062     /**
23063      * Get the index within the cache of the Record with the passed id.
23064      * @param {String} id The id of the Record to find.
23065      * @return {Number} The index of the Record. Returns -1 if not found.
23066      */
23067     indexOfId : function(id){
23068         return this.data.indexOfKey(id);
23069     },
23070
23071     /**
23072      * Get the Record with the specified id.
23073      * @param {String} id The id of the Record to find.
23074      * @return {Roo.data.Record} The Record with the passed id. Returns undefined if not found.
23075      */
23076     getById : function(id){
23077         return this.data.key(id);
23078     },
23079
23080     /**
23081      * Get the Record at the specified index.
23082      * @param {Number} index The index of the Record to find.
23083      * @return {Roo.data.Record} The Record at the passed index. Returns undefined if not found.
23084      */
23085     getAt : function(index){
23086         return this.data.itemAt(index);
23087     },
23088
23089     /**
23090      * Returns a range of Records between specified indices.
23091      * @param {Number} startIndex (optional) The starting index (defaults to 0)
23092      * @param {Number} endIndex (optional) The ending index (defaults to the last Record in the Store)
23093      * @return {Roo.data.Record[]} An array of Records
23094      */
23095     getRange : function(start, end){
23096         return this.data.getRange(start, end);
23097     },
23098
23099     // private
23100     storeOptions : function(o){
23101         o = Roo.apply({}, o);
23102         delete o.callback;
23103         delete o.scope;
23104         this.lastOptions = o;
23105     },
23106
23107     /**
23108      * Loads the Record cache from the configured Proxy using the configured Reader.
23109      * <p>
23110      * If using remote paging, then the first load call must specify the <em>start</em>
23111      * and <em>limit</em> properties in the options.params property to establish the initial
23112      * position within the dataset, and the number of Records to cache on each read from the Proxy.
23113      * <p>
23114      * <strong>It is important to note that for remote data sources, loading is asynchronous,
23115      * and this call will return before the new data has been loaded. Perform any post-processing
23116      * in a callback function, or in a "load" event handler.</strong>
23117      * <p>
23118      * @param {Object} options An object containing properties which control loading options:<ul>
23119      * <li>params {Object} An object containing properties to pass as HTTP parameters to a remote data source.</li>
23120      * <li>callback {Function} A function to be called after the Records have been loaded. The callback is
23121      * passed the following arguments:<ul>
23122      * <li>r : Roo.data.Record[]</li>
23123      * <li>options: Options object from the load call</li>
23124      * <li>success: Boolean success indicator</li></ul></li>
23125      * <li>scope {Object} Scope with which to call the callback (defaults to the Store object)</li>
23126      * <li>add {Boolean} indicator to append loaded records rather than replace the current cache.</li>
23127      * </ul>
23128      */
23129     load : function(options){
23130         options = options || {};
23131         if(this.fireEvent("beforeload", this, options) !== false){
23132             this.storeOptions(options);
23133             var p = Roo.apply(options.params || {}, this.baseParams);
23134             // if meta was not loaded from remote source.. try requesting it.
23135             if (!this.reader.metaFromRemote) {
23136                 p._requestMeta = 1;
23137             }
23138             if(this.sortInfo && this.remoteSort){
23139                 var pn = this.paramNames;
23140                 p[pn["sort"]] = this.sortInfo.field;
23141                 p[pn["dir"]] = this.sortInfo.direction;
23142             }
23143             if (this.multiSort) {
23144                 var pn = this.paramNames;
23145                 p[pn["multisort"]] = Roo.encode( { sort : this.sortToggle, order: this.sortOrder });
23146             }
23147             
23148             this.proxy.load(p, this.reader, this.loadRecords, this, options);
23149         }
23150     },
23151
23152     /**
23153      * Reloads the Record cache from the configured Proxy using the configured Reader and
23154      * the options from the last load operation performed.
23155      * @param {Object} options (optional) An object containing properties which may override the options
23156      * used in the last load operation. See {@link #load} for details (defaults to null, in which case
23157      * the most recently used options are reused).
23158      */
23159     reload : function(options){
23160         this.load(Roo.applyIf(options||{}, this.lastOptions));
23161     },
23162
23163     // private
23164     // Called as a callback by the Reader during a load operation.
23165     loadRecords : function(o, options, success){
23166         if(!o || success === false){
23167             if(success !== false){
23168                 this.fireEvent("load", this, [], options, o);
23169             }
23170             if(options.callback){
23171                 options.callback.call(options.scope || this, [], options, false);
23172             }
23173             return;
23174         }
23175         // if data returned failure - throw an exception.
23176         if (o.success === false) {
23177             // show a message if no listener is registered.
23178             if (!this.hasListener('loadexception') && typeof(o.raw.errorMsg) != 'undefined') {
23179                     Roo.MessageBox.alert("Error loading",o.raw.errorMsg);
23180             }
23181             // loadmask wil be hooked into this..
23182             this.fireEvent("loadexception", this, o, options, o.raw.errorMsg);
23183             return;
23184         }
23185         var r = o.records, t = o.totalRecords || r.length;
23186         
23187         this.fireEvent("beforeloadadd", this, r, options, o);
23188         
23189         if(!options || options.add !== true){
23190             if(this.pruneModifiedRecords){
23191                 this.modified = [];
23192             }
23193             for(var i = 0, len = r.length; i < len; i++){
23194                 r[i].join(this);
23195             }
23196             if(this.snapshot){
23197                 this.data = this.snapshot;
23198                 delete this.snapshot;
23199             }
23200             this.data.clear();
23201             this.data.addAll(r);
23202             this.totalLength = t;
23203             this.applySort();
23204             this.fireEvent("datachanged", this);
23205         }else{
23206             this.totalLength = Math.max(t, this.data.length+r.length);
23207             this.add(r);
23208         }
23209         
23210         if(this.parent && !Roo.isIOS && !this.useNativeIOS && this.parent.emptyTitle.length) {
23211                 
23212             var e = new Roo.data.Record({});
23213
23214             e.set(this.parent.displayField, this.parent.emptyTitle);
23215             e.set(this.parent.valueField, '');
23216
23217             this.insert(0, e);
23218         }
23219             
23220         this.fireEvent("load", this, r, options, o);
23221         if(options.callback){
23222             options.callback.call(options.scope || this, r, options, true);
23223         }
23224     },
23225
23226
23227     /**
23228      * Loads data from a passed data block. A Reader which understands the format of the data
23229      * must have been configured in the constructor.
23230      * @param {Object} data The data block from which to read the Records.  The format of the data expected
23231      * is dependent on the type of Reader that is configured and should correspond to that Reader's readRecords parameter.
23232      * @param {Boolean} append (Optional) True to append the new Records rather than replace the existing cache.
23233      */
23234     loadData : function(o, append){
23235         var r = this.reader.readRecords(o);
23236         this.loadRecords(r, {add: append}, true);
23237     },
23238
23239     /**
23240      * Gets the number of cached records.
23241      * <p>
23242      * <em>If using paging, this may not be the total size of the dataset. If the data object
23243      * used by the Reader contains the dataset size, then the getTotalCount() function returns
23244      * the data set size</em>
23245      */
23246     getCount : function(){
23247         return this.data.length || 0;
23248     },
23249
23250     /**
23251      * Gets the total number of records in the dataset as returned by the server.
23252      * <p>
23253      * <em>If using paging, for this to be accurate, the data object used by the Reader must contain
23254      * the dataset size</em>
23255      */
23256     getTotalCount : function(){
23257         return this.totalLength || 0;
23258     },
23259
23260     /**
23261      * Returns the sort state of the Store as an object with two properties:
23262      * <pre><code>
23263  field {String} The name of the field by which the Records are sorted
23264  direction {String} The sort order, "ASC" or "DESC"
23265      * </code></pre>
23266      */
23267     getSortState : function(){
23268         return this.sortInfo;
23269     },
23270
23271     // private
23272     applySort : function(){
23273         if(this.sortInfo && !this.remoteSort){
23274             var s = this.sortInfo, f = s.field;
23275             var st = this.fields.get(f).sortType;
23276             var fn = function(r1, r2){
23277                 var v1 = st(r1.data[f]), v2 = st(r2.data[f]);
23278                 return v1 > v2 ? 1 : (v1 < v2 ? -1 : 0);
23279             };
23280             this.data.sort(s.direction, fn);
23281             if(this.snapshot && this.snapshot != this.data){
23282                 this.snapshot.sort(s.direction, fn);
23283             }
23284         }
23285     },
23286
23287     /**
23288      * Sets the default sort column and order to be used by the next load operation.
23289      * @param {String} fieldName The name of the field to sort by.
23290      * @param {String} dir (optional) The sort order, "ASC" or "DESC" (defaults to "ASC")
23291      */
23292     setDefaultSort : function(field, dir){
23293         this.sortInfo = {field: field, direction: dir ? dir.toUpperCase() : "ASC"};
23294     },
23295
23296     /**
23297      * Sort the Records.
23298      * If remote sorting is used, the sort is performed on the server, and the cache is
23299      * reloaded. If local sorting is used, the cache is sorted internally.
23300      * @param {String} fieldName The name of the field to sort by.
23301      * @param {String} dir (optional) The sort order, "ASC" or "DESC" (defaults to "ASC")
23302      */
23303     sort : function(fieldName, dir){
23304         var f = this.fields.get(fieldName);
23305         if(!dir){
23306             this.sortToggle[f.name] = this.sortToggle[f.name] || f.sortDir;
23307             
23308             if(this.multiSort || (this.sortInfo && this.sortInfo.field == f.name) ){ // toggle sort dir
23309                 dir = (this.sortToggle[f.name] || "ASC").toggle("ASC", "DESC");
23310             }else{
23311                 dir = f.sortDir;
23312             }
23313         }
23314         this.sortToggle[f.name] = dir;
23315         this.sortInfo = {field: f.name, direction: dir};
23316         if(!this.remoteSort){
23317             this.applySort();
23318             this.fireEvent("datachanged", this);
23319         }else{
23320             this.load(this.lastOptions);
23321         }
23322     },
23323
23324     /**
23325      * Calls the specified function for each of the Records in the cache.
23326      * @param {Function} fn The function to call. The Record is passed as the first parameter.
23327      * Returning <em>false</em> aborts and exits the iteration.
23328      * @param {Object} scope (optional) The scope in which to call the function (defaults to the Record).
23329      */
23330     each : function(fn, scope){
23331         this.data.each(fn, scope);
23332     },
23333
23334     /**
23335      * Gets all records modified since the last commit.  Modified records are persisted across load operations
23336      * (e.g., during paging).
23337      * @return {Roo.data.Record[]} An array of Records containing outstanding modifications.
23338      */
23339     getModifiedRecords : function(){
23340         return this.modified;
23341     },
23342
23343     // private
23344     createFilterFn : function(property, value, anyMatch){
23345         if(!value.exec){ // not a regex
23346             value = String(value);
23347             if(value.length == 0){
23348                 return false;
23349             }
23350             value = new RegExp((anyMatch === true ? '' : '^') + Roo.escapeRe(value), "i");
23351         }
23352         return function(r){
23353             return value.test(r.data[property]);
23354         };
23355     },
23356
23357     /**
23358      * Sums the value of <i>property</i> for each record between start and end and returns the result.
23359      * @param {String} property A field on your records
23360      * @param {Number} start The record index to start at (defaults to 0)
23361      * @param {Number} end The last record index to include (defaults to length - 1)
23362      * @return {Number} The sum
23363      */
23364     sum : function(property, start, end){
23365         var rs = this.data.items, v = 0;
23366         start = start || 0;
23367         end = (end || end === 0) ? end : rs.length-1;
23368
23369         for(var i = start; i <= end; i++){
23370             v += (rs[i].data[property] || 0);
23371         }
23372         return v;
23373     },
23374
23375     /**
23376      * Filter the records by a specified property.
23377      * @param {String} field A field on your records
23378      * @param {String/RegExp} value Either a string that the field
23379      * should start with or a RegExp to test against the field
23380      * @param {Boolean} anyMatch True to match any part not just the beginning
23381      */
23382     filter : function(property, value, anyMatch){
23383         var fn = this.createFilterFn(property, value, anyMatch);
23384         return fn ? this.filterBy(fn) : this.clearFilter();
23385     },
23386
23387     /**
23388      * Filter by a function. The specified function will be called with each
23389      * record in this data source. If the function returns true the record is included,
23390      * otherwise it is filtered.
23391      * @param {Function} fn The function to be called, it will receive 2 args (record, id)
23392      * @param {Object} scope (optional) The scope of the function (defaults to this)
23393      */
23394     filterBy : function(fn, scope){
23395         this.snapshot = this.snapshot || this.data;
23396         this.data = this.queryBy(fn, scope||this);
23397         this.fireEvent("datachanged", this);
23398     },
23399
23400     /**
23401      * Query the records by a specified property.
23402      * @param {String} field A field on your records
23403      * @param {String/RegExp} value Either a string that the field
23404      * should start with or a RegExp to test against the field
23405      * @param {Boolean} anyMatch True to match any part not just the beginning
23406      * @return {MixedCollection} Returns an Roo.util.MixedCollection of the matched records
23407      */
23408     query : function(property, value, anyMatch){
23409         var fn = this.createFilterFn(property, value, anyMatch);
23410         return fn ? this.queryBy(fn) : this.data.clone();
23411     },
23412
23413     /**
23414      * Query by a function. The specified function will be called with each
23415      * record in this data source. If the function returns true the record is included
23416      * in the results.
23417      * @param {Function} fn The function to be called, it will receive 2 args (record, id)
23418      * @param {Object} scope (optional) The scope of the function (defaults to this)
23419       @return {MixedCollection} Returns an Roo.util.MixedCollection of the matched records
23420      **/
23421     queryBy : function(fn, scope){
23422         var data = this.snapshot || this.data;
23423         return data.filterBy(fn, scope||this);
23424     },
23425
23426     /**
23427      * Collects unique values for a particular dataIndex from this store.
23428      * @param {String} dataIndex The property to collect
23429      * @param {Boolean} allowNull (optional) Pass true to allow null, undefined or empty string values
23430      * @param {Boolean} bypassFilter (optional) Pass true to collect from all records, even ones which are filtered
23431      * @return {Array} An array of the unique values
23432      **/
23433     collect : function(dataIndex, allowNull, bypassFilter){
23434         var d = (bypassFilter === true && this.snapshot) ?
23435                 this.snapshot.items : this.data.items;
23436         var v, sv, r = [], l = {};
23437         for(var i = 0, len = d.length; i < len; i++){
23438             v = d[i].data[dataIndex];
23439             sv = String(v);
23440             if((allowNull || !Roo.isEmpty(v)) && !l[sv]){
23441                 l[sv] = true;
23442                 r[r.length] = v;
23443             }
23444         }
23445         return r;
23446     },
23447
23448     /**
23449      * Revert to a view of the Record cache with no filtering applied.
23450      * @param {Boolean} suppressEvent If true the filter is cleared silently without notifying listeners
23451      */
23452     clearFilter : function(suppressEvent){
23453         if(this.snapshot && this.snapshot != this.data){
23454             this.data = this.snapshot;
23455             delete this.snapshot;
23456             if(suppressEvent !== true){
23457                 this.fireEvent("datachanged", this);
23458             }
23459         }
23460     },
23461
23462     // private
23463     afterEdit : function(record){
23464         if(this.modified.indexOf(record) == -1){
23465             this.modified.push(record);
23466         }
23467         this.fireEvent("update", this, record, Roo.data.Record.EDIT);
23468     },
23469     
23470     // private
23471     afterReject : function(record){
23472         this.modified.remove(record);
23473         this.fireEvent("update", this, record, Roo.data.Record.REJECT);
23474     },
23475
23476     // private
23477     afterCommit : function(record){
23478         this.modified.remove(record);
23479         this.fireEvent("update", this, record, Roo.data.Record.COMMIT);
23480     },
23481
23482     /**
23483      * Commit all Records with outstanding changes. To handle updates for changes, subscribe to the
23484      * Store's "update" event, and perform updating when the third parameter is Roo.data.Record.COMMIT.
23485      */
23486     commitChanges : function(){
23487         var m = this.modified.slice(0);
23488         this.modified = [];
23489         for(var i = 0, len = m.length; i < len; i++){
23490             m[i].commit();
23491         }
23492     },
23493
23494     /**
23495      * Cancel outstanding changes on all changed records.
23496      */
23497     rejectChanges : function(){
23498         var m = this.modified.slice(0);
23499         this.modified = [];
23500         for(var i = 0, len = m.length; i < len; i++){
23501             m[i].reject();
23502         }
23503     },
23504
23505     onMetaChange : function(meta, rtype, o){
23506         this.recordType = rtype;
23507         this.fields = rtype.prototype.fields;
23508         delete this.snapshot;
23509         this.sortInfo = meta.sortInfo || this.sortInfo;
23510         this.modified = [];
23511         this.fireEvent('metachange', this, this.reader.meta);
23512     },
23513     
23514     moveIndex : function(data, type)
23515     {
23516         var index = this.indexOf(data);
23517         
23518         var newIndex = index + type;
23519         
23520         this.remove(data);
23521         
23522         this.insert(newIndex, data);
23523         
23524     }
23525 });/*
23526  * Based on:
23527  * Ext JS Library 1.1.1
23528  * Copyright(c) 2006-2007, Ext JS, LLC.
23529  *
23530  * Originally Released Under LGPL - original licence link has changed is not relivant.
23531  *
23532  * Fork - LGPL
23533  * <script type="text/javascript">
23534  */
23535
23536 /**
23537  * @class Roo.data.SimpleStore
23538  * @extends Roo.data.Store
23539  * Small helper class to make creating Stores from Array data easier.
23540  * @cfg {Number} id The array index of the record id. Leave blank to auto generate ids.
23541  * @cfg {Array} fields An array of field definition objects, or field name strings.
23542  * @cfg {Array} data The multi-dimensional array of data
23543  * @constructor
23544  * @param {Object} config
23545  */
23546 Roo.data.SimpleStore = function(config){
23547     Roo.data.SimpleStore.superclass.constructor.call(this, {
23548         isLocal : true,
23549         reader: new Roo.data.ArrayReader({
23550                 id: config.id
23551             },
23552             Roo.data.Record.create(config.fields)
23553         ),
23554         proxy : new Roo.data.MemoryProxy(config.data)
23555     });
23556     this.load();
23557 };
23558 Roo.extend(Roo.data.SimpleStore, Roo.data.Store);/*
23559  * Based on:
23560  * Ext JS Library 1.1.1
23561  * Copyright(c) 2006-2007, Ext JS, LLC.
23562  *
23563  * Originally Released Under LGPL - original licence link has changed is not relivant.
23564  *
23565  * Fork - LGPL
23566  * <script type="text/javascript">
23567  */
23568
23569 /**
23570 /**
23571  * @extends Roo.data.Store
23572  * @class Roo.data.JsonStore
23573  * Small helper class to make creating Stores for JSON data easier. <br/>
23574 <pre><code>
23575 var store = new Roo.data.JsonStore({
23576     url: 'get-images.php',
23577     root: 'images',
23578     fields: ['name', 'url', {name:'size', type: 'float'}, {name:'lastmod', type:'date'}]
23579 });
23580 </code></pre>
23581  * <b>Note: Although they are not listed, this class inherits all of the config options of Store,
23582  * JsonReader and HttpProxy (unless inline data is provided).</b>
23583  * @cfg {Array} fields An array of field definition objects, or field name strings.
23584  * @constructor
23585  * @param {Object} config
23586  */
23587 Roo.data.JsonStore = function(c){
23588     Roo.data.JsonStore.superclass.constructor.call(this, Roo.apply(c, {
23589         proxy: !c.data ? new Roo.data.HttpProxy({url: c.url}) : undefined,
23590         reader: new Roo.data.JsonReader(c, c.fields)
23591     }));
23592 };
23593 Roo.extend(Roo.data.JsonStore, Roo.data.Store);/*
23594  * Based on:
23595  * Ext JS Library 1.1.1
23596  * Copyright(c) 2006-2007, Ext JS, LLC.
23597  *
23598  * Originally Released Under LGPL - original licence link has changed is not relivant.
23599  *
23600  * Fork - LGPL
23601  * <script type="text/javascript">
23602  */
23603
23604  
23605 Roo.data.Field = function(config){
23606     if(typeof config == "string"){
23607         config = {name: config};
23608     }
23609     Roo.apply(this, config);
23610     
23611     if(!this.type){
23612         this.type = "auto";
23613     }
23614     
23615     var st = Roo.data.SortTypes;
23616     // named sortTypes are supported, here we look them up
23617     if(typeof this.sortType == "string"){
23618         this.sortType = st[this.sortType];
23619     }
23620     
23621     // set default sortType for strings and dates
23622     if(!this.sortType){
23623         switch(this.type){
23624             case "string":
23625                 this.sortType = st.asUCString;
23626                 break;
23627             case "date":
23628                 this.sortType = st.asDate;
23629                 break;
23630             default:
23631                 this.sortType = st.none;
23632         }
23633     }
23634
23635     // define once
23636     var stripRe = /[\$,%]/g;
23637
23638     // prebuilt conversion function for this field, instead of
23639     // switching every time we're reading a value
23640     if(!this.convert){
23641         var cv, dateFormat = this.dateFormat;
23642         switch(this.type){
23643             case "":
23644             case "auto":
23645             case undefined:
23646                 cv = function(v){ return v; };
23647                 break;
23648             case "string":
23649                 cv = function(v){ return (v === undefined || v === null) ? '' : String(v); };
23650                 break;
23651             case "int":
23652                 cv = function(v){
23653                     return v !== undefined && v !== null && v !== '' ?
23654                            parseInt(String(v).replace(stripRe, ""), 10) : '';
23655                     };
23656                 break;
23657             case "float":
23658                 cv = function(v){
23659                     return v !== undefined && v !== null && v !== '' ?
23660                            parseFloat(String(v).replace(stripRe, ""), 10) : ''; 
23661                     };
23662                 break;
23663             case "bool":
23664             case "boolean":
23665                 cv = function(v){ return v === true || v === "true" || v == 1; };
23666                 break;
23667             case "date":
23668                 cv = function(v){
23669                     if(!v){
23670                         return '';
23671                     }
23672                     if(v instanceof Date){
23673                         return v;
23674                     }
23675                     if(dateFormat){
23676                         if(dateFormat == "timestamp"){
23677                             return new Date(v*1000);
23678                         }
23679                         return Date.parseDate(v, dateFormat);
23680                     }
23681                     var parsed = Date.parse(v);
23682                     return parsed ? new Date(parsed) : null;
23683                 };
23684              break;
23685             
23686         }
23687         this.convert = cv;
23688     }
23689 };
23690
23691 Roo.data.Field.prototype = {
23692     dateFormat: null,
23693     defaultValue: "",
23694     mapping: null,
23695     sortType : null,
23696     sortDir : "ASC"
23697 };/*
23698  * Based on:
23699  * Ext JS Library 1.1.1
23700  * Copyright(c) 2006-2007, Ext JS, LLC.
23701  *
23702  * Originally Released Under LGPL - original licence link has changed is not relivant.
23703  *
23704  * Fork - LGPL
23705  * <script type="text/javascript">
23706  */
23707  
23708 // Base class for reading structured data from a data source.  This class is intended to be
23709 // extended (see ArrayReader, JsonReader and XmlReader) and should not be created directly.
23710
23711 /**
23712  * @class Roo.data.DataReader
23713  * Base class for reading structured data from a data source.  This class is intended to be
23714  * extended (see {Roo.data.ArrayReader}, {Roo.data.JsonReader} and {Roo.data.XmlReader}) and should not be created directly.
23715  */
23716
23717 Roo.data.DataReader = function(meta, recordType){
23718     
23719     this.meta = meta;
23720     
23721     this.recordType = recordType instanceof Array ? 
23722         Roo.data.Record.create(recordType) : recordType;
23723 };
23724
23725 Roo.data.DataReader.prototype = {
23726      /**
23727      * Create an empty record
23728      * @param {Object} data (optional) - overlay some values
23729      * @return {Roo.data.Record} record created.
23730      */
23731     newRow :  function(d) {
23732         var da =  {};
23733         this.recordType.prototype.fields.each(function(c) {
23734             switch( c.type) {
23735                 case 'int' : da[c.name] = 0; break;
23736                 case 'date' : da[c.name] = new Date(); break;
23737                 case 'float' : da[c.name] = 0.0; break;
23738                 case 'boolean' : da[c.name] = false; break;
23739                 default : da[c.name] = ""; break;
23740             }
23741             
23742         });
23743         return new this.recordType(Roo.apply(da, d));
23744     }
23745     
23746 };/*
23747  * Based on:
23748  * Ext JS Library 1.1.1
23749  * Copyright(c) 2006-2007, Ext JS, LLC.
23750  *
23751  * Originally Released Under LGPL - original licence link has changed is not relivant.
23752  *
23753  * Fork - LGPL
23754  * <script type="text/javascript">
23755  */
23756
23757 /**
23758  * @class Roo.data.DataProxy
23759  * @extends Roo.data.Observable
23760  * This class is an abstract base class for implementations which provide retrieval of
23761  * unformatted data objects.<br>
23762  * <p>
23763  * DataProxy implementations are usually used in conjunction with an implementation of Roo.data.DataReader
23764  * (of the appropriate type which knows how to parse the data object) to provide a block of
23765  * {@link Roo.data.Records} to an {@link Roo.data.Store}.<br>
23766  * <p>
23767  * Custom implementations must implement the load method as described in
23768  * {@link Roo.data.HttpProxy#load}.
23769  */
23770 Roo.data.DataProxy = function(){
23771     this.addEvents({
23772         /**
23773          * @event beforeload
23774          * Fires before a network request is made to retrieve a data object.
23775          * @param {Object} This DataProxy object.
23776          * @param {Object} params The params parameter to the load function.
23777          */
23778         beforeload : true,
23779         /**
23780          * @event load
23781          * Fires before the load method's callback is called.
23782          * @param {Object} This DataProxy object.
23783          * @param {Object} o The data object.
23784          * @param {Object} arg The callback argument object passed to the load function.
23785          */
23786         load : true,
23787         /**
23788          * @event loadexception
23789          * Fires if an Exception occurs during data retrieval.
23790          * @param {Object} This DataProxy object.
23791          * @param {Object} o The data object.
23792          * @param {Object} arg The callback argument object passed to the load function.
23793          * @param {Object} e The Exception.
23794          */
23795         loadexception : true
23796     });
23797     Roo.data.DataProxy.superclass.constructor.call(this);
23798 };
23799
23800 Roo.extend(Roo.data.DataProxy, Roo.util.Observable);
23801
23802     /**
23803      * @cfg {void} listeners (Not available) Constructor blocks listeners from being set
23804      */
23805 /*
23806  * Based on:
23807  * Ext JS Library 1.1.1
23808  * Copyright(c) 2006-2007, Ext JS, LLC.
23809  *
23810  * Originally Released Under LGPL - original licence link has changed is not relivant.
23811  *
23812  * Fork - LGPL
23813  * <script type="text/javascript">
23814  */
23815 /**
23816  * @class Roo.data.MemoryProxy
23817  * An implementation of Roo.data.DataProxy that simply passes the data specified in its constructor
23818  * to the Reader when its load method is called.
23819  * @constructor
23820  * @param {Object} data The data object which the Reader uses to construct a block of Roo.data.Records.
23821  */
23822 Roo.data.MemoryProxy = function(data){
23823     if (data.data) {
23824         data = data.data;
23825     }
23826     Roo.data.MemoryProxy.superclass.constructor.call(this);
23827     this.data = data;
23828 };
23829
23830 Roo.extend(Roo.data.MemoryProxy, Roo.data.DataProxy, {
23831     
23832     /**
23833      * Load data from the requested source (in this case an in-memory
23834      * data object passed to the constructor), read the data object into
23835      * a block of Roo.data.Records using the passed Roo.data.DataReader implementation, and
23836      * process that block using the passed callback.
23837      * @param {Object} params This parameter is not used by the MemoryProxy class.
23838      * @param {Roo.data.DataReader} reader The Reader object which converts the data
23839      * object into a block of Roo.data.Records.
23840      * @param {Function} callback The function into which to pass the block of Roo.data.records.
23841      * The function must be passed <ul>
23842      * <li>The Record block object</li>
23843      * <li>The "arg" argument from the load function</li>
23844      * <li>A boolean success indicator</li>
23845      * </ul>
23846      * @param {Object} scope The scope in which to call the callback
23847      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
23848      */
23849     load : function(params, reader, callback, scope, arg){
23850         params = params || {};
23851         var result;
23852         try {
23853             result = reader.readRecords(params.data ? params.data :this.data);
23854         }catch(e){
23855             this.fireEvent("loadexception", this, arg, null, e);
23856             callback.call(scope, null, arg, false);
23857             return;
23858         }
23859         callback.call(scope, result, arg, true);
23860     },
23861     
23862     // private
23863     update : function(params, records){
23864         
23865     }
23866 });/*
23867  * Based on:
23868  * Ext JS Library 1.1.1
23869  * Copyright(c) 2006-2007, Ext JS, LLC.
23870  *
23871  * Originally Released Under LGPL - original licence link has changed is not relivant.
23872  *
23873  * Fork - LGPL
23874  * <script type="text/javascript">
23875  */
23876 /**
23877  * @class Roo.data.HttpProxy
23878  * @extends Roo.data.DataProxy
23879  * An implementation of {@link Roo.data.DataProxy} that reads a data object from an {@link Roo.data.Connection} object
23880  * configured to reference a certain URL.<br><br>
23881  * <p>
23882  * <em>Note that this class cannot be used to retrieve data from a domain other than the domain
23883  * from which the running page was served.<br><br>
23884  * <p>
23885  * For cross-domain access to remote data, use an {@link Roo.data.ScriptTagProxy}.</em><br><br>
23886  * <p>
23887  * Be aware that to enable the browser to parse an XML document, the server must set
23888  * the Content-Type header in the HTTP response to "text/xml".
23889  * @constructor
23890  * @param {Object} conn Connection config options to add to each request (e.g. {url: 'foo.php'} or
23891  * an {@link Roo.data.Connection} object.  If a Connection config is passed, the singleton {@link Roo.Ajax} object
23892  * will be used to make the request.
23893  */
23894 Roo.data.HttpProxy = function(conn){
23895     Roo.data.HttpProxy.superclass.constructor.call(this);
23896     // is conn a conn config or a real conn?
23897     this.conn = conn;
23898     this.useAjax = !conn || !conn.events;
23899   
23900 };
23901
23902 Roo.extend(Roo.data.HttpProxy, Roo.data.DataProxy, {
23903     // thse are take from connection...
23904     
23905     /**
23906      * @cfg {String} url (Optional) The default URL to be used for requests to the server. (defaults to undefined)
23907      */
23908     /**
23909      * @cfg {Object} extraParams (Optional) An object containing properties which are used as
23910      * extra parameters to each request made by this object. (defaults to undefined)
23911      */
23912     /**
23913      * @cfg {Object} defaultHeaders (Optional) An object containing request headers which are added
23914      *  to each request made by this object. (defaults to undefined)
23915      */
23916     /**
23917      * @cfg {String} method (Optional) The default HTTP method to be used for requests. (defaults to undefined; if not set but parms are present will use POST, otherwise GET)
23918      */
23919     /**
23920      * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
23921      */
23922      /**
23923      * @cfg {Boolean} autoAbort (Optional) Whether this request should abort any pending requests. (defaults to false)
23924      * @type Boolean
23925      */
23926   
23927
23928     /**
23929      * @cfg {Boolean} disableCaching (Optional) True to add a unique cache-buster param to GET requests. (defaults to true)
23930      * @type Boolean
23931      */
23932     /**
23933      * Return the {@link Roo.data.Connection} object being used by this Proxy.
23934      * @return {Connection} The Connection object. This object may be used to subscribe to events on
23935      * a finer-grained basis than the DataProxy events.
23936      */
23937     getConnection : function(){
23938         return this.useAjax ? Roo.Ajax : this.conn;
23939     },
23940
23941     /**
23942      * Load data from the configured {@link Roo.data.Connection}, read the data object into
23943      * a block of Roo.data.Records using the passed {@link Roo.data.DataReader} implementation, and
23944      * process that block using the passed callback.
23945      * @param {Object} params An object containing properties which are to be used as HTTP parameters
23946      * for the request to the remote server.
23947      * @param {Roo.data.DataReader} reader The Reader object which converts the data
23948      * object into a block of Roo.data.Records.
23949      * @param {Function} callback The function into which to pass the block of Roo.data.Records.
23950      * The function must be passed <ul>
23951      * <li>The Record block object</li>
23952      * <li>The "arg" argument from the load function</li>
23953      * <li>A boolean success indicator</li>
23954      * </ul>
23955      * @param {Object} scope The scope in which to call the callback
23956      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
23957      */
23958     load : function(params, reader, callback, scope, arg){
23959         if(this.fireEvent("beforeload", this, params) !== false){
23960             var  o = {
23961                 params : params || {},
23962                 request: {
23963                     callback : callback,
23964                     scope : scope,
23965                     arg : arg
23966                 },
23967                 reader: reader,
23968                 callback : this.loadResponse,
23969                 scope: this
23970             };
23971             if(this.useAjax){
23972                 Roo.applyIf(o, this.conn);
23973                 if(this.activeRequest){
23974                     Roo.Ajax.abort(this.activeRequest);
23975                 }
23976                 this.activeRequest = Roo.Ajax.request(o);
23977             }else{
23978                 this.conn.request(o);
23979             }
23980         }else{
23981             callback.call(scope||this, null, arg, false);
23982         }
23983     },
23984
23985     // private
23986     loadResponse : function(o, success, response){
23987         delete this.activeRequest;
23988         if(!success){
23989             this.fireEvent("loadexception", this, o, response);
23990             o.request.callback.call(o.request.scope, null, o.request.arg, false);
23991             return;
23992         }
23993         var result;
23994         try {
23995             result = o.reader.read(response);
23996         }catch(e){
23997             this.fireEvent("loadexception", this, o, response, e);
23998             o.request.callback.call(o.request.scope, null, o.request.arg, false);
23999             return;
24000         }
24001         
24002         this.fireEvent("load", this, o, o.request.arg);
24003         o.request.callback.call(o.request.scope, result, o.request.arg, true);
24004     },
24005
24006     // private
24007     update : function(dataSet){
24008
24009     },
24010
24011     // private
24012     updateResponse : function(dataSet){
24013
24014     }
24015 });/*
24016  * Based on:
24017  * Ext JS Library 1.1.1
24018  * Copyright(c) 2006-2007, Ext JS, LLC.
24019  *
24020  * Originally Released Under LGPL - original licence link has changed is not relivant.
24021  *
24022  * Fork - LGPL
24023  * <script type="text/javascript">
24024  */
24025
24026 /**
24027  * @class Roo.data.ScriptTagProxy
24028  * An implementation of Roo.data.DataProxy that reads a data object from a URL which may be in a domain
24029  * other than the originating domain of the running page.<br><br>
24030  * <p>
24031  * <em>Note that if you are retrieving data from a page that is in a domain that is NOT the same as the originating domain
24032  * of the running page, you must use this class, rather than DataProxy.</em><br><br>
24033  * <p>
24034  * The content passed back from a server resource requested by a ScriptTagProxy is executable JavaScript
24035  * source code that is used as the source inside a &lt;script> tag.<br><br>
24036  * <p>
24037  * In order for the browser to process the returned data, the server must wrap the data object
24038  * with a call to a callback function, the name of which is passed as a parameter by the ScriptTagProxy.
24039  * Below is a Java example for a servlet which returns data for either a ScriptTagProxy, or an HttpProxy
24040  * depending on whether the callback name was passed:
24041  * <p>
24042  * <pre><code>
24043 boolean scriptTag = false;
24044 String cb = request.getParameter("callback");
24045 if (cb != null) {
24046     scriptTag = true;
24047     response.setContentType("text/javascript");
24048 } else {
24049     response.setContentType("application/x-json");
24050 }
24051 Writer out = response.getWriter();
24052 if (scriptTag) {
24053     out.write(cb + "(");
24054 }
24055 out.print(dataBlock.toJsonString());
24056 if (scriptTag) {
24057     out.write(");");
24058 }
24059 </pre></code>
24060  *
24061  * @constructor
24062  * @param {Object} config A configuration object.
24063  */
24064 Roo.data.ScriptTagProxy = function(config){
24065     Roo.data.ScriptTagProxy.superclass.constructor.call(this);
24066     Roo.apply(this, config);
24067     this.head = document.getElementsByTagName("head")[0];
24068 };
24069
24070 Roo.data.ScriptTagProxy.TRANS_ID = 1000;
24071
24072 Roo.extend(Roo.data.ScriptTagProxy, Roo.data.DataProxy, {
24073     /**
24074      * @cfg {String} url The URL from which to request the data object.
24075      */
24076     /**
24077      * @cfg {Number} timeout (Optional) The number of milliseconds to wait for a response. Defaults to 30 seconds.
24078      */
24079     timeout : 30000,
24080     /**
24081      * @cfg {String} callbackParam (Optional) The name of the parameter to pass to the server which tells
24082      * the server the name of the callback function set up by the load call to process the returned data object.
24083      * Defaults to "callback".<p>The server-side processing must read this parameter value, and generate
24084      * javascript output which calls this named function passing the data object as its only parameter.
24085      */
24086     callbackParam : "callback",
24087     /**
24088      *  @cfg {Boolean} nocache (Optional) Defaults to true. Disable cacheing by adding a unique parameter
24089      * name to the request.
24090      */
24091     nocache : true,
24092
24093     /**
24094      * Load data from the configured URL, read the data object into
24095      * a block of Roo.data.Records using the passed Roo.data.DataReader implementation, and
24096      * process that block using the passed callback.
24097      * @param {Object} params An object containing properties which are to be used as HTTP parameters
24098      * for the request to the remote server.
24099      * @param {Roo.data.DataReader} reader The Reader object which converts the data
24100      * object into a block of Roo.data.Records.
24101      * @param {Function} callback The function into which to pass the block of Roo.data.Records.
24102      * The function must be passed <ul>
24103      * <li>The Record block object</li>
24104      * <li>The "arg" argument from the load function</li>
24105      * <li>A boolean success indicator</li>
24106      * </ul>
24107      * @param {Object} scope The scope in which to call the callback
24108      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
24109      */
24110     load : function(params, reader, callback, scope, arg){
24111         if(this.fireEvent("beforeload", this, params) !== false){
24112
24113             var p = Roo.urlEncode(Roo.apply(params, this.extraParams));
24114
24115             var url = this.url;
24116             url += (url.indexOf("?") != -1 ? "&" : "?") + p;
24117             if(this.nocache){
24118                 url += "&_dc=" + (new Date().getTime());
24119             }
24120             var transId = ++Roo.data.ScriptTagProxy.TRANS_ID;
24121             var trans = {
24122                 id : transId,
24123                 cb : "stcCallback"+transId,
24124                 scriptId : "stcScript"+transId,
24125                 params : params,
24126                 arg : arg,
24127                 url : url,
24128                 callback : callback,
24129                 scope : scope,
24130                 reader : reader
24131             };
24132             var conn = this;
24133
24134             window[trans.cb] = function(o){
24135                 conn.handleResponse(o, trans);
24136             };
24137
24138             url += String.format("&{0}={1}", this.callbackParam, trans.cb);
24139
24140             if(this.autoAbort !== false){
24141                 this.abort();
24142             }
24143
24144             trans.timeoutId = this.handleFailure.defer(this.timeout, this, [trans]);
24145
24146             var script = document.createElement("script");
24147             script.setAttribute("src", url);
24148             script.setAttribute("type", "text/javascript");
24149             script.setAttribute("id", trans.scriptId);
24150             this.head.appendChild(script);
24151
24152             this.trans = trans;
24153         }else{
24154             callback.call(scope||this, null, arg, false);
24155         }
24156     },
24157
24158     // private
24159     isLoading : function(){
24160         return this.trans ? true : false;
24161     },
24162
24163     /**
24164      * Abort the current server request.
24165      */
24166     abort : function(){
24167         if(this.isLoading()){
24168             this.destroyTrans(this.trans);
24169         }
24170     },
24171
24172     // private
24173     destroyTrans : function(trans, isLoaded){
24174         this.head.removeChild(document.getElementById(trans.scriptId));
24175         clearTimeout(trans.timeoutId);
24176         if(isLoaded){
24177             window[trans.cb] = undefined;
24178             try{
24179                 delete window[trans.cb];
24180             }catch(e){}
24181         }else{
24182             // if hasn't been loaded, wait for load to remove it to prevent script error
24183             window[trans.cb] = function(){
24184                 window[trans.cb] = undefined;
24185                 try{
24186                     delete window[trans.cb];
24187                 }catch(e){}
24188             };
24189         }
24190     },
24191
24192     // private
24193     handleResponse : function(o, trans){
24194         this.trans = false;
24195         this.destroyTrans(trans, true);
24196         var result;
24197         try {
24198             result = trans.reader.readRecords(o);
24199         }catch(e){
24200             this.fireEvent("loadexception", this, o, trans.arg, e);
24201             trans.callback.call(trans.scope||window, null, trans.arg, false);
24202             return;
24203         }
24204         this.fireEvent("load", this, o, trans.arg);
24205         trans.callback.call(trans.scope||window, result, trans.arg, true);
24206     },
24207
24208     // private
24209     handleFailure : function(trans){
24210         this.trans = false;
24211         this.destroyTrans(trans, false);
24212         this.fireEvent("loadexception", this, null, trans.arg);
24213         trans.callback.call(trans.scope||window, null, trans.arg, false);
24214     }
24215 });/*
24216  * Based on:
24217  * Ext JS Library 1.1.1
24218  * Copyright(c) 2006-2007, Ext JS, LLC.
24219  *
24220  * Originally Released Under LGPL - original licence link has changed is not relivant.
24221  *
24222  * Fork - LGPL
24223  * <script type="text/javascript">
24224  */
24225
24226 /**
24227  * @class Roo.data.JsonReader
24228  * @extends Roo.data.DataReader
24229  * Data reader class to create an Array of Roo.data.Record objects from a JSON response
24230  * based on mappings in a provided Roo.data.Record constructor.
24231  * 
24232  * The default behaviour of a store is to send ?_requestMeta=1, unless the class has recieved 'metaData' property
24233  * in the reply previously. 
24234  * 
24235  * <p>
24236  * Example code:
24237  * <pre><code>
24238 var RecordDef = Roo.data.Record.create([
24239     {name: 'name', mapping: 'name'},     // "mapping" property not needed if it's the same as "name"
24240     {name: 'occupation'}                 // This field will use "occupation" as the mapping.
24241 ]);
24242 var myReader = new Roo.data.JsonReader({
24243     totalProperty: "results",    // The property which contains the total dataset size (optional)
24244     root: "rows",                // The property which contains an Array of row objects
24245     id: "id"                     // The property within each row object that provides an ID for the record (optional)
24246 }, RecordDef);
24247 </code></pre>
24248  * <p>
24249  * This would consume a JSON file like this:
24250  * <pre><code>
24251 { 'results': 2, 'rows': [
24252     { 'id': 1, 'name': 'Bill', occupation: 'Gardener' },
24253     { 'id': 2, 'name': 'Ben', occupation: 'Horticulturalist' } ]
24254 }
24255 </code></pre>
24256  * @cfg {String} totalProperty Name of the property from which to retrieve the total number of records
24257  * in the dataset. This is only needed if the whole dataset is not passed in one go, but is being
24258  * paged from the remote server.
24259  * @cfg {String} successProperty Name of the property from which to retrieve the success attribute used by forms.
24260  * @cfg {String} root name of the property which contains the Array of row objects.
24261  * @cfg {String} id Name of the property within a row object that contains a record identifier value.
24262  * @cfg {Array} fields Array of field definition objects
24263  * @constructor
24264  * Create a new JsonReader
24265  * @param {Object} meta Metadata configuration options
24266  * @param {Object} recordType Either an Array of field definition objects,
24267  * or an {@link Roo.data.Record} object created using {@link Roo.data.Record#create}.
24268  */
24269 Roo.data.JsonReader = function(meta, recordType){
24270     
24271     meta = meta || {};
24272     // set some defaults:
24273     Roo.applyIf(meta, {
24274         totalProperty: 'total',
24275         successProperty : 'success',
24276         root : 'data',
24277         id : 'id'
24278     });
24279     
24280     Roo.data.JsonReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24281 };
24282 Roo.extend(Roo.data.JsonReader, Roo.data.DataReader, {
24283     
24284     /**
24285      * @prop {Boolean} metaFromRemote  - if the meta data was loaded from the remote source.
24286      * Used by Store query builder to append _requestMeta to params.
24287      * 
24288      */
24289     metaFromRemote : false,
24290     /**
24291      * This method is only used by a DataProxy which has retrieved data from a remote server.
24292      * @param {Object} response The XHR object which contains the JSON data in its responseText.
24293      * @return {Object} data A data block which is used by an Roo.data.Store object as
24294      * a cache of Roo.data.Records.
24295      */
24296     read : function(response){
24297         var json = response.responseText;
24298        
24299         var o = /* eval:var:o */ eval("("+json+")");
24300         if(!o) {
24301             throw {message: "JsonReader.read: Json object not found"};
24302         }
24303         
24304         if(o.metaData){
24305             
24306             delete this.ef;
24307             this.metaFromRemote = true;
24308             this.meta = o.metaData;
24309             this.recordType = Roo.data.Record.create(o.metaData.fields);
24310             this.onMetaChange(this.meta, this.recordType, o);
24311         }
24312         return this.readRecords(o);
24313     },
24314
24315     // private function a store will implement
24316     onMetaChange : function(meta, recordType, o){
24317
24318     },
24319
24320     /**
24321          * @ignore
24322          */
24323     simpleAccess: function(obj, subsc) {
24324         return obj[subsc];
24325     },
24326
24327         /**
24328          * @ignore
24329          */
24330     getJsonAccessor: function(){
24331         var re = /[\[\.]/;
24332         return function(expr) {
24333             try {
24334                 return(re.test(expr))
24335                     ? new Function("obj", "return obj." + expr)
24336                     : function(obj){
24337                         return obj[expr];
24338                     };
24339             } catch(e){}
24340             return Roo.emptyFn;
24341         };
24342     }(),
24343
24344     /**
24345      * Create a data block containing Roo.data.Records from an XML document.
24346      * @param {Object} o An object which contains an Array of row objects in the property specified
24347      * in the config as 'root, and optionally a property, specified in the config as 'totalProperty'
24348      * which contains the total size of the dataset.
24349      * @return {Object} data A data block which is used by an Roo.data.Store object as
24350      * a cache of Roo.data.Records.
24351      */
24352     readRecords : function(o){
24353         /**
24354          * After any data loads, the raw JSON data is available for further custom processing.
24355          * @type Object
24356          */
24357         this.o = o;
24358         var s = this.meta, Record = this.recordType,
24359             f = Record ? Record.prototype.fields : null, fi = f ? f.items : [], fl = f ? f.length : 0;
24360
24361 //      Generate extraction functions for the totalProperty, the root, the id, and for each field
24362         if (!this.ef) {
24363             if(s.totalProperty) {
24364                     this.getTotal = this.getJsonAccessor(s.totalProperty);
24365                 }
24366                 if(s.successProperty) {
24367                     this.getSuccess = this.getJsonAccessor(s.successProperty);
24368                 }
24369                 this.getRoot = s.root ? this.getJsonAccessor(s.root) : function(p){return p;};
24370                 if (s.id) {
24371                         var g = this.getJsonAccessor(s.id);
24372                         this.getId = function(rec) {
24373                                 var r = g(rec);  
24374                                 return (r === undefined || r === "") ? null : r;
24375                         };
24376                 } else {
24377                         this.getId = function(){return null;};
24378                 }
24379             this.ef = [];
24380             for(var jj = 0; jj < fl; jj++){
24381                 f = fi[jj];
24382                 var map = (f.mapping !== undefined && f.mapping !== null) ? f.mapping : f.name;
24383                 this.ef[jj] = this.getJsonAccessor(map);
24384             }
24385         }
24386
24387         var root = this.getRoot(o), c = root.length, totalRecords = c, success = true;
24388         if(s.totalProperty){
24389             var vt = parseInt(this.getTotal(o), 10);
24390             if(!isNaN(vt)){
24391                 totalRecords = vt;
24392             }
24393         }
24394         if(s.successProperty){
24395             var vs = this.getSuccess(o);
24396             if(vs === false || vs === 'false'){
24397                 success = false;
24398             }
24399         }
24400         var records = [];
24401         for(var i = 0; i < c; i++){
24402                 var n = root[i];
24403             var values = {};
24404             var id = this.getId(n);
24405             for(var j = 0; j < fl; j++){
24406                 f = fi[j];
24407             var v = this.ef[j](n);
24408             if (!f.convert) {
24409                 Roo.log('missing convert for ' + f.name);
24410                 Roo.log(f);
24411                 continue;
24412             }
24413             values[f.name] = f.convert((v !== undefined) ? v : f.defaultValue);
24414             }
24415             var record = new Record(values, id);
24416             record.json = n;
24417             records[i] = record;
24418         }
24419         return {
24420             raw : o,
24421             success : success,
24422             records : records,
24423             totalRecords : totalRecords
24424         };
24425     }
24426 });/*
24427  * Based on:
24428  * Ext JS Library 1.1.1
24429  * Copyright(c) 2006-2007, Ext JS, LLC.
24430  *
24431  * Originally Released Under LGPL - original licence link has changed is not relivant.
24432  *
24433  * Fork - LGPL
24434  * <script type="text/javascript">
24435  */
24436
24437 /**
24438  * @class Roo.data.XmlReader
24439  * @extends Roo.data.DataReader
24440  * Data reader class to create an Array of {@link Roo.data.Record} objects from an XML document
24441  * based on mappings in a provided Roo.data.Record constructor.<br><br>
24442  * <p>
24443  * <em>Note that in order for the browser to parse a returned XML document, the Content-Type
24444  * header in the HTTP response must be set to "text/xml".</em>
24445  * <p>
24446  * Example code:
24447  * <pre><code>
24448 var RecordDef = Roo.data.Record.create([
24449    {name: 'name', mapping: 'name'},     // "mapping" property not needed if it's the same as "name"
24450    {name: 'occupation'}                 // This field will use "occupation" as the mapping.
24451 ]);
24452 var myReader = new Roo.data.XmlReader({
24453    totalRecords: "results", // The element which contains the total dataset size (optional)
24454    record: "row",           // The repeated element which contains row information
24455    id: "id"                 // The element within the row that provides an ID for the record (optional)
24456 }, RecordDef);
24457 </code></pre>
24458  * <p>
24459  * This would consume an XML file like this:
24460  * <pre><code>
24461 &lt;?xml?>
24462 &lt;dataset>
24463  &lt;results>2&lt;/results>
24464  &lt;row>
24465    &lt;id>1&lt;/id>
24466    &lt;name>Bill&lt;/name>
24467    &lt;occupation>Gardener&lt;/occupation>
24468  &lt;/row>
24469  &lt;row>
24470    &lt;id>2&lt;/id>
24471    &lt;name>Ben&lt;/name>
24472    &lt;occupation>Horticulturalist&lt;/occupation>
24473  &lt;/row>
24474 &lt;/dataset>
24475 </code></pre>
24476  * @cfg {String} totalRecords The DomQuery path from which to retrieve the total number of records
24477  * in the dataset. This is only needed if the whole dataset is not passed in one go, but is being
24478  * paged from the remote server.
24479  * @cfg {String} record The DomQuery path to the repeated element which contains record information.
24480  * @cfg {String} success The DomQuery path to the success attribute used by forms.
24481  * @cfg {String} id The DomQuery path relative from the record element to the element that contains
24482  * a record identifier value.
24483  * @constructor
24484  * Create a new XmlReader
24485  * @param {Object} meta Metadata configuration options
24486  * @param {Mixed} recordType The definition of the data record type to produce.  This can be either a valid
24487  * Record subclass created with {@link Roo.data.Record#create}, or an array of objects with which to call
24488  * Roo.data.Record.create.  See the {@link Roo.data.Record} class for more details.
24489  */
24490 Roo.data.XmlReader = function(meta, recordType){
24491     meta = meta || {};
24492     Roo.data.XmlReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24493 };
24494 Roo.extend(Roo.data.XmlReader, Roo.data.DataReader, {
24495     /**
24496      * This method is only used by a DataProxy which has retrieved data from a remote server.
24497          * @param {Object} response The XHR object which contains the parsed XML document.  The response is expected
24498          * to contain a method called 'responseXML' that returns an XML document object.
24499      * @return {Object} records A data block which is used by an {@link Roo.data.Store} as
24500      * a cache of Roo.data.Records.
24501      */
24502     read : function(response){
24503         var doc = response.responseXML;
24504         if(!doc) {
24505             throw {message: "XmlReader.read: XML Document not available"};
24506         }
24507         return this.readRecords(doc);
24508     },
24509
24510     /**
24511      * Create a data block containing Roo.data.Records from an XML document.
24512          * @param {Object} doc A parsed XML document.
24513      * @return {Object} records A data block which is used by an {@link Roo.data.Store} as
24514      * a cache of Roo.data.Records.
24515      */
24516     readRecords : function(doc){
24517         /**
24518          * After any data loads/reads, the raw XML Document is available for further custom processing.
24519          * @type XMLDocument
24520          */
24521         this.xmlData = doc;
24522         var root = doc.documentElement || doc;
24523         var q = Roo.DomQuery;
24524         var recordType = this.recordType, fields = recordType.prototype.fields;
24525         var sid = this.meta.id;
24526         var totalRecords = 0, success = true;
24527         if(this.meta.totalRecords){
24528             totalRecords = q.selectNumber(this.meta.totalRecords, root, 0);
24529         }
24530         
24531         if(this.meta.success){
24532             var sv = q.selectValue(this.meta.success, root, true);
24533             success = sv !== false && sv !== 'false';
24534         }
24535         var records = [];
24536         var ns = q.select(this.meta.record, root);
24537         for(var i = 0, len = ns.length; i < len; i++) {
24538                 var n = ns[i];
24539                 var values = {};
24540                 var id = sid ? q.selectValue(sid, n) : undefined;
24541                 for(var j = 0, jlen = fields.length; j < jlen; j++){
24542                     var f = fields.items[j];
24543                 var v = q.selectValue(f.mapping || f.name, n, f.defaultValue);
24544                     v = f.convert(v);
24545                     values[f.name] = v;
24546                 }
24547                 var record = new recordType(values, id);
24548                 record.node = n;
24549                 records[records.length] = record;
24550             }
24551
24552             return {
24553                 success : success,
24554                 records : records,
24555                 totalRecords : totalRecords || records.length
24556             };
24557     }
24558 });/*
24559  * Based on:
24560  * Ext JS Library 1.1.1
24561  * Copyright(c) 2006-2007, Ext JS, LLC.
24562  *
24563  * Originally Released Under LGPL - original licence link has changed is not relivant.
24564  *
24565  * Fork - LGPL
24566  * <script type="text/javascript">
24567  */
24568
24569 /**
24570  * @class Roo.data.ArrayReader
24571  * @extends Roo.data.DataReader
24572  * Data reader class to create an Array of Roo.data.Record objects from an Array.
24573  * Each element of that Array represents a row of data fields. The
24574  * fields are pulled into a Record object using as a subscript, the <em>mapping</em> property
24575  * of the field definition if it exists, or the field's ordinal position in the definition.<br>
24576  * <p>
24577  * Example code:.
24578  * <pre><code>
24579 var RecordDef = Roo.data.Record.create([
24580     {name: 'name', mapping: 1},         // "mapping" only needed if an "id" field is present which
24581     {name: 'occupation', mapping: 2}    // precludes using the ordinal position as the index.
24582 ]);
24583 var myReader = new Roo.data.ArrayReader({
24584     id: 0                     // The subscript within row Array that provides an ID for the Record (optional)
24585 }, RecordDef);
24586 </code></pre>
24587  * <p>
24588  * This would consume an Array like this:
24589  * <pre><code>
24590 [ [1, 'Bill', 'Gardener'], [2, 'Ben', 'Horticulturalist'] ]
24591   </code></pre>
24592  
24593  * @constructor
24594  * Create a new JsonReader
24595  * @param {Object} meta Metadata configuration options.
24596  * @param {Object|Array} recordType Either an Array of field definition objects
24597  * 
24598  * @cfg {Array} fields Array of field definition objects
24599  * @cfg {String} id Name of the property within a row object that contains a record identifier value.
24600  * as specified to {@link Roo.data.Record#create},
24601  * or an {@link Roo.data.Record} object
24602  *
24603  * 
24604  * created using {@link Roo.data.Record#create}.
24605  */
24606 Roo.data.ArrayReader = function(meta, recordType){
24607     
24608      
24609     Roo.data.ArrayReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24610 };
24611
24612 Roo.extend(Roo.data.ArrayReader, Roo.data.JsonReader, {
24613     /**
24614      * Create a data block containing Roo.data.Records from an XML document.
24615      * @param {Object} o An Array of row objects which represents the dataset.
24616      * @return {Object} A data block which is used by an {@link Roo.data.Store} object as
24617      * a cache of Roo.data.Records.
24618      */
24619     readRecords : function(o){
24620         var sid = this.meta ? this.meta.id : null;
24621         var recordType = this.recordType, fields = recordType.prototype.fields;
24622         var records = [];
24623         var root = o;
24624             for(var i = 0; i < root.length; i++){
24625                     var n = root[i];
24626                 var values = {};
24627                 var id = ((sid || sid === 0) && n[sid] !== undefined && n[sid] !== "" ? n[sid] : null);
24628                 for(var j = 0, jlen = fields.length; j < jlen; j++){
24629                 var f = fields.items[j];
24630                 var k = f.mapping !== undefined && f.mapping !== null ? f.mapping : j;
24631                 var v = n[k] !== undefined ? n[k] : f.defaultValue;
24632                 v = f.convert(v);
24633                 values[f.name] = v;
24634             }
24635                 var record = new recordType(values, id);
24636                 record.json = n;
24637                 records[records.length] = record;
24638             }
24639             return {
24640                 records : records,
24641                 totalRecords : records.length
24642             };
24643     }
24644 });/*
24645  * Based on:
24646  * Ext JS Library 1.1.1
24647  * Copyright(c) 2006-2007, Ext JS, LLC.
24648  *
24649  * Originally Released Under LGPL - original licence link has changed is not relivant.
24650  *
24651  * Fork - LGPL
24652  * <script type="text/javascript">
24653  */
24654
24655
24656 /**
24657  * @class Roo.data.Tree
24658  * @extends Roo.util.Observable
24659  * Represents a tree data structure and bubbles all the events for its nodes. The nodes
24660  * in the tree have most standard DOM functionality.
24661  * @constructor
24662  * @param {Node} root (optional) The root node
24663  */
24664 Roo.data.Tree = function(root){
24665    this.nodeHash = {};
24666    /**
24667     * The root node for this tree
24668     * @type Node
24669     */
24670    this.root = null;
24671    if(root){
24672        this.setRootNode(root);
24673    }
24674    this.addEvents({
24675        /**
24676         * @event append
24677         * Fires when a new child node is appended to a node in this tree.
24678         * @param {Tree} tree The owner tree
24679         * @param {Node} parent The parent node
24680         * @param {Node} node The newly appended node
24681         * @param {Number} index The index of the newly appended node
24682         */
24683        "append" : true,
24684        /**
24685         * @event remove
24686         * Fires when a child node is removed from a node in this tree.
24687         * @param {Tree} tree The owner tree
24688         * @param {Node} parent The parent node
24689         * @param {Node} node The child node removed
24690         */
24691        "remove" : true,
24692        /**
24693         * @event move
24694         * Fires when a node is moved to a new location in the tree
24695         * @param {Tree} tree The owner tree
24696         * @param {Node} node The node moved
24697         * @param {Node} oldParent The old parent of this node
24698         * @param {Node} newParent The new parent of this node
24699         * @param {Number} index The index it was moved to
24700         */
24701        "move" : true,
24702        /**
24703         * @event insert
24704         * Fires when a new child node is inserted in a node in this tree.
24705         * @param {Tree} tree The owner tree
24706         * @param {Node} parent The parent node
24707         * @param {Node} node The child node inserted
24708         * @param {Node} refNode The child node the node was inserted before
24709         */
24710        "insert" : true,
24711        /**
24712         * @event beforeappend
24713         * Fires before a new child is appended to a node in this tree, return false to cancel the append.
24714         * @param {Tree} tree The owner tree
24715         * @param {Node} parent The parent node
24716         * @param {Node} node The child node to be appended
24717         */
24718        "beforeappend" : true,
24719        /**
24720         * @event beforeremove
24721         * Fires before a child is removed from a node in this tree, return false to cancel the remove.
24722         * @param {Tree} tree The owner tree
24723         * @param {Node} parent The parent node
24724         * @param {Node} node The child node to be removed
24725         */
24726        "beforeremove" : true,
24727        /**
24728         * @event beforemove
24729         * Fires before a node is moved to a new location in the tree. Return false to cancel the move.
24730         * @param {Tree} tree The owner tree
24731         * @param {Node} node The node being moved
24732         * @param {Node} oldParent The parent of the node
24733         * @param {Node} newParent The new parent the node is moving to
24734         * @param {Number} index The index it is being moved to
24735         */
24736        "beforemove" : true,
24737        /**
24738         * @event beforeinsert
24739         * Fires before a new child is inserted in a node in this tree, return false to cancel the insert.
24740         * @param {Tree} tree The owner tree
24741         * @param {Node} parent The parent node
24742         * @param {Node} node The child node to be inserted
24743         * @param {Node} refNode The child node the node is being inserted before
24744         */
24745        "beforeinsert" : true
24746    });
24747
24748     Roo.data.Tree.superclass.constructor.call(this);
24749 };
24750
24751 Roo.extend(Roo.data.Tree, Roo.util.Observable, {
24752     pathSeparator: "/",
24753
24754     proxyNodeEvent : function(){
24755         return this.fireEvent.apply(this, arguments);
24756     },
24757
24758     /**
24759      * Returns the root node for this tree.
24760      * @return {Node}
24761      */
24762     getRootNode : function(){
24763         return this.root;
24764     },
24765
24766     /**
24767      * Sets the root node for this tree.
24768      * @param {Node} node
24769      * @return {Node}
24770      */
24771     setRootNode : function(node){
24772         this.root = node;
24773         node.ownerTree = this;
24774         node.isRoot = true;
24775         this.registerNode(node);
24776         return node;
24777     },
24778
24779     /**
24780      * Gets a node in this tree by its id.
24781      * @param {String} id
24782      * @return {Node}
24783      */
24784     getNodeById : function(id){
24785         return this.nodeHash[id];
24786     },
24787
24788     registerNode : function(node){
24789         this.nodeHash[node.id] = node;
24790     },
24791
24792     unregisterNode : function(node){
24793         delete this.nodeHash[node.id];
24794     },
24795
24796     toString : function(){
24797         return "[Tree"+(this.id?" "+this.id:"")+"]";
24798     }
24799 });
24800
24801 /**
24802  * @class Roo.data.Node
24803  * @extends Roo.util.Observable
24804  * @cfg {Boolean} leaf true if this node is a leaf and does not have children
24805  * @cfg {String} id The id for this node. If one is not specified, one is generated.
24806  * @constructor
24807  * @param {Object} attributes The attributes/config for the node
24808  */
24809 Roo.data.Node = function(attributes){
24810     /**
24811      * The attributes supplied for the node. You can use this property to access any custom attributes you supplied.
24812      * @type {Object}
24813      */
24814     this.attributes = attributes || {};
24815     this.leaf = this.attributes.leaf;
24816     /**
24817      * The node id. @type String
24818      */
24819     this.id = this.attributes.id;
24820     if(!this.id){
24821         this.id = Roo.id(null, "ynode-");
24822         this.attributes.id = this.id;
24823     }
24824      
24825     
24826     /**
24827      * All child nodes of this node. @type Array
24828      */
24829     this.childNodes = [];
24830     if(!this.childNodes.indexOf){ // indexOf is a must
24831         this.childNodes.indexOf = function(o){
24832             for(var i = 0, len = this.length; i < len; i++){
24833                 if(this[i] == o) {
24834                     return i;
24835                 }
24836             }
24837             return -1;
24838         };
24839     }
24840     /**
24841      * The parent node for this node. @type Node
24842      */
24843     this.parentNode = null;
24844     /**
24845      * The first direct child node of this node, or null if this node has no child nodes. @type Node
24846      */
24847     this.firstChild = null;
24848     /**
24849      * The last direct child node of this node, or null if this node has no child nodes. @type Node
24850      */
24851     this.lastChild = null;
24852     /**
24853      * The node immediately preceding this node in the tree, or null if there is no sibling node. @type Node
24854      */
24855     this.previousSibling = null;
24856     /**
24857      * The node immediately following this node in the tree, or null if there is no sibling node. @type Node
24858      */
24859     this.nextSibling = null;
24860
24861     this.addEvents({
24862        /**
24863         * @event append
24864         * Fires when a new child node is appended
24865         * @param {Tree} tree The owner tree
24866         * @param {Node} this This node
24867         * @param {Node} node The newly appended node
24868         * @param {Number} index The index of the newly appended node
24869         */
24870        "append" : true,
24871        /**
24872         * @event remove
24873         * Fires when a child node is removed
24874         * @param {Tree} tree The owner tree
24875         * @param {Node} this This node
24876         * @param {Node} node The removed node
24877         */
24878        "remove" : true,
24879        /**
24880         * @event move
24881         * Fires when this node is moved to a new location in the tree
24882         * @param {Tree} tree The owner tree
24883         * @param {Node} this This node
24884         * @param {Node} oldParent The old parent of this node
24885         * @param {Node} newParent The new parent of this node
24886         * @param {Number} index The index it was moved to
24887         */
24888        "move" : true,
24889        /**
24890         * @event insert
24891         * Fires when a new child node is inserted.
24892         * @param {Tree} tree The owner tree
24893         * @param {Node} this This node
24894         * @param {Node} node The child node inserted
24895         * @param {Node} refNode The child node the node was inserted before
24896         */
24897        "insert" : true,
24898        /**
24899         * @event beforeappend
24900         * Fires before a new child is appended, return false to cancel the append.
24901         * @param {Tree} tree The owner tree
24902         * @param {Node} this This node
24903         * @param {Node} node The child node to be appended
24904         */
24905        "beforeappend" : true,
24906        /**
24907         * @event beforeremove
24908         * Fires before a child is removed, return false to cancel the remove.
24909         * @param {Tree} tree The owner tree
24910         * @param {Node} this This node
24911         * @param {Node} node The child node to be removed
24912         */
24913        "beforeremove" : true,
24914        /**
24915         * @event beforemove
24916         * Fires before this node is moved to a new location in the tree. Return false to cancel the move.
24917         * @param {Tree} tree The owner tree
24918         * @param {Node} this This node
24919         * @param {Node} oldParent The parent of this node
24920         * @param {Node} newParent The new parent this node is moving to
24921         * @param {Number} index The index it is being moved to
24922         */
24923        "beforemove" : true,
24924        /**
24925         * @event beforeinsert
24926         * Fires before a new child is inserted, return false to cancel the insert.
24927         * @param {Tree} tree The owner tree
24928         * @param {Node} this This node
24929         * @param {Node} node The child node to be inserted
24930         * @param {Node} refNode The child node the node is being inserted before
24931         */
24932        "beforeinsert" : true
24933    });
24934     this.listeners = this.attributes.listeners;
24935     Roo.data.Node.superclass.constructor.call(this);
24936 };
24937
24938 Roo.extend(Roo.data.Node, Roo.util.Observable, {
24939     fireEvent : function(evtName){
24940         // first do standard event for this node
24941         if(Roo.data.Node.superclass.fireEvent.apply(this, arguments) === false){
24942             return false;
24943         }
24944         // then bubble it up to the tree if the event wasn't cancelled
24945         var ot = this.getOwnerTree();
24946         if(ot){
24947             if(ot.proxyNodeEvent.apply(ot, arguments) === false){
24948                 return false;
24949             }
24950         }
24951         return true;
24952     },
24953
24954     /**
24955      * Returns true if this node is a leaf
24956      * @return {Boolean}
24957      */
24958     isLeaf : function(){
24959         return this.leaf === true;
24960     },
24961
24962     // private
24963     setFirstChild : function(node){
24964         this.firstChild = node;
24965     },
24966
24967     //private
24968     setLastChild : function(node){
24969         this.lastChild = node;
24970     },
24971
24972
24973     /**
24974      * Returns true if this node is the last child of its parent
24975      * @return {Boolean}
24976      */
24977     isLast : function(){
24978        return (!this.parentNode ? true : this.parentNode.lastChild == this);
24979     },
24980
24981     /**
24982      * Returns true if this node is the first child of its parent
24983      * @return {Boolean}
24984      */
24985     isFirst : function(){
24986        return (!this.parentNode ? true : this.parentNode.firstChild == this);
24987     },
24988
24989     hasChildNodes : function(){
24990         return !this.isLeaf() && this.childNodes.length > 0;
24991     },
24992
24993     /**
24994      * Insert node(s) as the last child node of this node.
24995      * @param {Node/Array} node The node or Array of nodes to append
24996      * @return {Node} The appended node if single append, or null if an array was passed
24997      */
24998     appendChild : function(node){
24999         Roo.log('tree.js appendChild');
25000         var multi = false;
25001         if(node instanceof Array){
25002             multi = node;
25003         }else if(arguments.length > 1){
25004             multi = arguments;
25005         }
25006         // if passed an array or multiple args do them one by one
25007         if(multi){
25008             for(var i = 0, len = multi.length; i < len; i++) {
25009                 this.appendChild(multi[i]);
25010             }
25011         }else{
25012             if(this.fireEvent("beforeappend", this.ownerTree, this, node) === false){
25013                 return false;
25014             }
25015             var index = this.childNodes.length;
25016             var oldParent = node.parentNode;
25017             // it's a move, make sure we move it cleanly
25018             if(oldParent){
25019                 if(node.fireEvent("beforemove", node.getOwnerTree(), node, oldParent, this, index) === false){
25020                     return false;
25021                 }
25022                 oldParent.removeChild(node);
25023             }
25024             index = this.childNodes.length;
25025             if(index == 0){
25026                 this.setFirstChild(node);
25027             }
25028             this.childNodes.push(node);
25029             node.parentNode = this;
25030             var ps = this.childNodes[index-1];
25031             if(ps){
25032                 node.previousSibling = ps;
25033                 ps.nextSibling = node;
25034             }else{
25035                 node.previousSibling = null;
25036             }
25037             node.nextSibling = null;
25038             this.setLastChild(node);
25039             node.setOwnerTree(this.getOwnerTree());
25040             this.fireEvent("append", this.ownerTree, this, node, index);
25041             if(oldParent){
25042                 node.fireEvent("move", this.ownerTree, node, oldParent, this, index);
25043             }
25044             return node;
25045         }
25046     },
25047
25048     /**
25049      * Removes a child node from this node.
25050      * @param {Node} node The node to remove
25051      * @return {Node} The removed node
25052      */
25053     removeChild : function(node){
25054         var index = this.childNodes.indexOf(node);
25055         if(index == -1){
25056             return false;
25057         }
25058         if(this.fireEvent("beforeremove", this.ownerTree, this, node) === false){
25059             return false;
25060         }
25061
25062         // remove it from childNodes collection
25063         this.childNodes.splice(index, 1);
25064
25065         // update siblings
25066         if(node.previousSibling){
25067             node.previousSibling.nextSibling = node.nextSibling;
25068         }
25069         if(node.nextSibling){
25070             node.nextSibling.previousSibling = node.previousSibling;
25071         }
25072
25073         // update child refs
25074         if(this.firstChild == node){
25075             this.setFirstChild(node.nextSibling);
25076         }
25077         if(this.lastChild == node){
25078             this.setLastChild(node.previousSibling);
25079         }
25080
25081         node.setOwnerTree(null);
25082         // clear any references from the node
25083         node.parentNode = null;
25084         node.previousSibling = null;
25085         node.nextSibling = null;
25086         this.fireEvent("remove", this.ownerTree, this, node);
25087         return node;
25088     },
25089
25090     /**
25091      * Inserts the first node before the second node in this nodes childNodes collection.
25092      * @param {Node} node The node to insert
25093      * @param {Node} refNode The node to insert before (if null the node is appended)
25094      * @return {Node} The inserted node
25095      */
25096     insertBefore : function(node, refNode){
25097         if(!refNode){ // like standard Dom, refNode can be null for append
25098             return this.appendChild(node);
25099         }
25100         // nothing to do
25101         if(node == refNode){
25102             return false;
25103         }
25104
25105         if(this.fireEvent("beforeinsert", this.ownerTree, this, node, refNode) === false){
25106             return false;
25107         }
25108         var index = this.childNodes.indexOf(refNode);
25109         var oldParent = node.parentNode;
25110         var refIndex = index;
25111
25112         // when moving internally, indexes will change after remove
25113         if(oldParent == this && this.childNodes.indexOf(node) < index){
25114             refIndex--;
25115         }
25116
25117         // it's a move, make sure we move it cleanly
25118         if(oldParent){
25119             if(node.fireEvent("beforemove", node.getOwnerTree(), node, oldParent, this, index, refNode) === false){
25120                 return false;
25121             }
25122             oldParent.removeChild(node);
25123         }
25124         if(refIndex == 0){
25125             this.setFirstChild(node);
25126         }
25127         this.childNodes.splice(refIndex, 0, node);
25128         node.parentNode = this;
25129         var ps = this.childNodes[refIndex-1];
25130         if(ps){
25131             node.previousSibling = ps;
25132             ps.nextSibling = node;
25133         }else{
25134             node.previousSibling = null;
25135         }
25136         node.nextSibling = refNode;
25137         refNode.previousSibling = node;
25138         node.setOwnerTree(this.getOwnerTree());
25139         this.fireEvent("insert", this.ownerTree, this, node, refNode);
25140         if(oldParent){
25141             node.fireEvent("move", this.ownerTree, node, oldParent, this, refIndex, refNode);
25142         }
25143         return node;
25144     },
25145
25146     /**
25147      * Returns the child node at the specified index.
25148      * @param {Number} index
25149      * @return {Node}
25150      */
25151     item : function(index){
25152         return this.childNodes[index];
25153     },
25154
25155     /**
25156      * Replaces one child node in this node with another.
25157      * @param {Node} newChild The replacement node
25158      * @param {Node} oldChild The node to replace
25159      * @return {Node} The replaced node
25160      */
25161     replaceChild : function(newChild, oldChild){
25162         this.insertBefore(newChild, oldChild);
25163         this.removeChild(oldChild);
25164         return oldChild;
25165     },
25166
25167     /**
25168      * Returns the index of a child node
25169      * @param {Node} node
25170      * @return {Number} The index of the node or -1 if it was not found
25171      */
25172     indexOf : function(child){
25173         return this.childNodes.indexOf(child);
25174     },
25175
25176     /**
25177      * Returns the tree this node is in.
25178      * @return {Tree}
25179      */
25180     getOwnerTree : function(){
25181         // if it doesn't have one, look for one
25182         if(!this.ownerTree){
25183             var p = this;
25184             while(p){
25185                 if(p.ownerTree){
25186                     this.ownerTree = p.ownerTree;
25187                     break;
25188                 }
25189                 p = p.parentNode;
25190             }
25191         }
25192         return this.ownerTree;
25193     },
25194
25195     /**
25196      * Returns depth of this node (the root node has a depth of 0)
25197      * @return {Number}
25198      */
25199     getDepth : function(){
25200         var depth = 0;
25201         var p = this;
25202         while(p.parentNode){
25203             ++depth;
25204             p = p.parentNode;
25205         }
25206         return depth;
25207     },
25208
25209     // private
25210     setOwnerTree : function(tree){
25211         // if it's move, we need to update everyone
25212         if(tree != this.ownerTree){
25213             if(this.ownerTree){
25214                 this.ownerTree.unregisterNode(this);
25215             }
25216             this.ownerTree = tree;
25217             var cs = this.childNodes;
25218             for(var i = 0, len = cs.length; i < len; i++) {
25219                 cs[i].setOwnerTree(tree);
25220             }
25221             if(tree){
25222                 tree.registerNode(this);
25223             }
25224         }
25225     },
25226
25227     /**
25228      * Returns the path for this node. The path can be used to expand or select this node programmatically.
25229      * @param {String} attr (optional) The attr to use for the path (defaults to the node's id)
25230      * @return {String} The path
25231      */
25232     getPath : function(attr){
25233         attr = attr || "id";
25234         var p = this.parentNode;
25235         var b = [this.attributes[attr]];
25236         while(p){
25237             b.unshift(p.attributes[attr]);
25238             p = p.parentNode;
25239         }
25240         var sep = this.getOwnerTree().pathSeparator;
25241         return sep + b.join(sep);
25242     },
25243
25244     /**
25245      * Bubbles up the tree from this node, calling the specified function with each node. The scope (<i>this</i>) of
25246      * function call will be the scope provided or the current node. The arguments to the function
25247      * will be the args provided or the current node. If the function returns false at any point,
25248      * the bubble is stopped.
25249      * @param {Function} fn The function to call
25250      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25251      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25252      */
25253     bubble : function(fn, scope, args){
25254         var p = this;
25255         while(p){
25256             if(fn.call(scope || p, args || p) === false){
25257                 break;
25258             }
25259             p = p.parentNode;
25260         }
25261     },
25262
25263     /**
25264      * Cascades down the tree from this node, calling the specified function with each node. The scope (<i>this</i>) of
25265      * function call will be the scope provided or the current node. The arguments to the function
25266      * will be the args provided or the current node. If the function returns false at any point,
25267      * the cascade is stopped on that branch.
25268      * @param {Function} fn The function to call
25269      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25270      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25271      */
25272     cascade : function(fn, scope, args){
25273         if(fn.call(scope || this, args || this) !== false){
25274             var cs = this.childNodes;
25275             for(var i = 0, len = cs.length; i < len; i++) {
25276                 cs[i].cascade(fn, scope, args);
25277             }
25278         }
25279     },
25280
25281     /**
25282      * Interates the child nodes of this node, calling the specified function with each node. The scope (<i>this</i>) of
25283      * function call will be the scope provided or the current node. The arguments to the function
25284      * will be the args provided or the current node. If the function returns false at any point,
25285      * the iteration stops.
25286      * @param {Function} fn The function to call
25287      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25288      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25289      */
25290     eachChild : function(fn, scope, args){
25291         var cs = this.childNodes;
25292         for(var i = 0, len = cs.length; i < len; i++) {
25293                 if(fn.call(scope || this, args || cs[i]) === false){
25294                     break;
25295                 }
25296         }
25297     },
25298
25299     /**
25300      * Finds the first child that has the attribute with the specified value.
25301      * @param {String} attribute The attribute name
25302      * @param {Mixed} value The value to search for
25303      * @return {Node} The found child or null if none was found
25304      */
25305     findChild : function(attribute, value){
25306         var cs = this.childNodes;
25307         for(var i = 0, len = cs.length; i < len; i++) {
25308                 if(cs[i].attributes[attribute] == value){
25309                     return cs[i];
25310                 }
25311         }
25312         return null;
25313     },
25314
25315     /**
25316      * Finds the first child by a custom function. The child matches if the function passed
25317      * returns true.
25318      * @param {Function} fn
25319      * @param {Object} scope (optional)
25320      * @return {Node} The found child or null if none was found
25321      */
25322     findChildBy : function(fn, scope){
25323         var cs = this.childNodes;
25324         for(var i = 0, len = cs.length; i < len; i++) {
25325                 if(fn.call(scope||cs[i], cs[i]) === true){
25326                     return cs[i];
25327                 }
25328         }
25329         return null;
25330     },
25331
25332     /**
25333      * Sorts this nodes children using the supplied sort function
25334      * @param {Function} fn
25335      * @param {Object} scope (optional)
25336      */
25337     sort : function(fn, scope){
25338         var cs = this.childNodes;
25339         var len = cs.length;
25340         if(len > 0){
25341             var sortFn = scope ? function(){fn.apply(scope, arguments);} : fn;
25342             cs.sort(sortFn);
25343             for(var i = 0; i < len; i++){
25344                 var n = cs[i];
25345                 n.previousSibling = cs[i-1];
25346                 n.nextSibling = cs[i+1];
25347                 if(i == 0){
25348                     this.setFirstChild(n);
25349                 }
25350                 if(i == len-1){
25351                     this.setLastChild(n);
25352                 }
25353             }
25354         }
25355     },
25356
25357     /**
25358      * Returns true if this node is an ancestor (at any point) of the passed node.
25359      * @param {Node} node
25360      * @return {Boolean}
25361      */
25362     contains : function(node){
25363         return node.isAncestor(this);
25364     },
25365
25366     /**
25367      * Returns true if the passed node is an ancestor (at any point) of this node.
25368      * @param {Node} node
25369      * @return {Boolean}
25370      */
25371     isAncestor : function(node){
25372         var p = this.parentNode;
25373         while(p){
25374             if(p == node){
25375                 return true;
25376             }
25377             p = p.parentNode;
25378         }
25379         return false;
25380     },
25381
25382     toString : function(){
25383         return "[Node"+(this.id?" "+this.id:"")+"]";
25384     }
25385 });/*
25386  * Based on:
25387  * Ext JS Library 1.1.1
25388  * Copyright(c) 2006-2007, Ext JS, LLC.
25389  *
25390  * Originally Released Under LGPL - original licence link has changed is not relivant.
25391  *
25392  * Fork - LGPL
25393  * <script type="text/javascript">
25394  */
25395  (function(){ 
25396 /**
25397  * @class Roo.Layer
25398  * @extends Roo.Element
25399  * An extended {@link Roo.Element} object that supports a shadow and shim, constrain to viewport and
25400  * automatic maintaining of shadow/shim positions.
25401  * @cfg {Boolean} shim False to disable the iframe shim in browsers which need one (defaults to true)
25402  * @cfg {String/Boolean} shadow True to create a shadow element with default class "x-layer-shadow", or
25403  * you can pass a string with a CSS class name. False turns off the shadow.
25404  * @cfg {Object} dh DomHelper object config to create element with (defaults to {tag: "div", cls: "x-layer"}).
25405  * @cfg {Boolean} constrain False to disable constrain to viewport (defaults to true)
25406  * @cfg {String} cls CSS class to add to the element
25407  * @cfg {Number} zindex Starting z-index (defaults to 11000)
25408  * @cfg {Number} shadowOffset Number of pixels to offset the shadow (defaults to 3)
25409  * @constructor
25410  * @param {Object} config An object with config options.
25411  * @param {String/HTMLElement} existingEl (optional) Uses an existing DOM element. If the element is not found it creates it.
25412  */
25413
25414 Roo.Layer = function(config, existingEl){
25415     config = config || {};
25416     var dh = Roo.DomHelper;
25417     var cp = config.parentEl, pel = cp ? Roo.getDom(cp) : document.body;
25418     if(existingEl){
25419         this.dom = Roo.getDom(existingEl);
25420     }
25421     if(!this.dom){
25422         var o = config.dh || {tag: "div", cls: "x-layer"};
25423         this.dom = dh.append(pel, o);
25424     }
25425     if(config.cls){
25426         this.addClass(config.cls);
25427     }
25428     this.constrain = config.constrain !== false;
25429     this.visibilityMode = Roo.Element.VISIBILITY;
25430     if(config.id){
25431         this.id = this.dom.id = config.id;
25432     }else{
25433         this.id = Roo.id(this.dom);
25434     }
25435     this.zindex = config.zindex || this.getZIndex();
25436     this.position("absolute", this.zindex);
25437     if(config.shadow){
25438         this.shadowOffset = config.shadowOffset || 4;
25439         this.shadow = new Roo.Shadow({
25440             offset : this.shadowOffset,
25441             mode : config.shadow
25442         });
25443     }else{
25444         this.shadowOffset = 0;
25445     }
25446     this.useShim = config.shim !== false && Roo.useShims;
25447     this.useDisplay = config.useDisplay;
25448     this.hide();
25449 };
25450
25451 var supr = Roo.Element.prototype;
25452
25453 // shims are shared among layer to keep from having 100 iframes
25454 var shims = [];
25455
25456 Roo.extend(Roo.Layer, Roo.Element, {
25457
25458     getZIndex : function(){
25459         return this.zindex || parseInt(this.getStyle("z-index"), 10) || 11000;
25460     },
25461
25462     getShim : function(){
25463         if(!this.useShim){
25464             return null;
25465         }
25466         if(this.shim){
25467             return this.shim;
25468         }
25469         var shim = shims.shift();
25470         if(!shim){
25471             shim = this.createShim();
25472             shim.enableDisplayMode('block');
25473             shim.dom.style.display = 'none';
25474             shim.dom.style.visibility = 'visible';
25475         }
25476         var pn = this.dom.parentNode;
25477         if(shim.dom.parentNode != pn){
25478             pn.insertBefore(shim.dom, this.dom);
25479         }
25480         shim.setStyle('z-index', this.getZIndex()-2);
25481         this.shim = shim;
25482         return shim;
25483     },
25484
25485     hideShim : function(){
25486         if(this.shim){
25487             this.shim.setDisplayed(false);
25488             shims.push(this.shim);
25489             delete this.shim;
25490         }
25491     },
25492
25493     disableShadow : function(){
25494         if(this.shadow){
25495             this.shadowDisabled = true;
25496             this.shadow.hide();
25497             this.lastShadowOffset = this.shadowOffset;
25498             this.shadowOffset = 0;
25499         }
25500     },
25501
25502     enableShadow : function(show){
25503         if(this.shadow){
25504             this.shadowDisabled = false;
25505             this.shadowOffset = this.lastShadowOffset;
25506             delete this.lastShadowOffset;
25507             if(show){
25508                 this.sync(true);
25509             }
25510         }
25511     },
25512
25513     // private
25514     // this code can execute repeatedly in milliseconds (i.e. during a drag) so
25515     // code size was sacrificed for effeciency (e.g. no getBox/setBox, no XY calls)
25516     sync : function(doShow){
25517         var sw = this.shadow;
25518         if(!this.updating && this.isVisible() && (sw || this.useShim)){
25519             var sh = this.getShim();
25520
25521             var w = this.getWidth(),
25522                 h = this.getHeight();
25523
25524             var l = this.getLeft(true),
25525                 t = this.getTop(true);
25526
25527             if(sw && !this.shadowDisabled){
25528                 if(doShow && !sw.isVisible()){
25529                     sw.show(this);
25530                 }else{
25531                     sw.realign(l, t, w, h);
25532                 }
25533                 if(sh){
25534                     if(doShow){
25535                        sh.show();
25536                     }
25537                     // fit the shim behind the shadow, so it is shimmed too
25538                     var a = sw.adjusts, s = sh.dom.style;
25539                     s.left = (Math.min(l, l+a.l))+"px";
25540                     s.top = (Math.min(t, t+a.t))+"px";
25541                     s.width = (w+a.w)+"px";
25542                     s.height = (h+a.h)+"px";
25543                 }
25544             }else if(sh){
25545                 if(doShow){
25546                    sh.show();
25547                 }
25548                 sh.setSize(w, h);
25549                 sh.setLeftTop(l, t);
25550             }
25551             
25552         }
25553     },
25554
25555     // private
25556     destroy : function(){
25557         this.hideShim();
25558         if(this.shadow){
25559             this.shadow.hide();
25560         }
25561         this.removeAllListeners();
25562         var pn = this.dom.parentNode;
25563         if(pn){
25564             pn.removeChild(this.dom);
25565         }
25566         Roo.Element.uncache(this.id);
25567     },
25568
25569     remove : function(){
25570         this.destroy();
25571     },
25572
25573     // private
25574     beginUpdate : function(){
25575         this.updating = true;
25576     },
25577
25578     // private
25579     endUpdate : function(){
25580         this.updating = false;
25581         this.sync(true);
25582     },
25583
25584     // private
25585     hideUnders : function(negOffset){
25586         if(this.shadow){
25587             this.shadow.hide();
25588         }
25589         this.hideShim();
25590     },
25591
25592     // private
25593     constrainXY : function(){
25594         if(this.constrain){
25595             var vw = Roo.lib.Dom.getViewWidth(),
25596                 vh = Roo.lib.Dom.getViewHeight();
25597             var s = Roo.get(document).getScroll();
25598
25599             var xy = this.getXY();
25600             var x = xy[0], y = xy[1];   
25601             var w = this.dom.offsetWidth+this.shadowOffset, h = this.dom.offsetHeight+this.shadowOffset;
25602             // only move it if it needs it
25603             var moved = false;
25604             // first validate right/bottom
25605             if((x + w) > vw+s.left){
25606                 x = vw - w - this.shadowOffset;
25607                 moved = true;
25608             }
25609             if((y + h) > vh+s.top){
25610                 y = vh - h - this.shadowOffset;
25611                 moved = true;
25612             }
25613             // then make sure top/left isn't negative
25614             if(x < s.left){
25615                 x = s.left;
25616                 moved = true;
25617             }
25618             if(y < s.top){
25619                 y = s.top;
25620                 moved = true;
25621             }
25622             if(moved){
25623                 if(this.avoidY){
25624                     var ay = this.avoidY;
25625                     if(y <= ay && (y+h) >= ay){
25626                         y = ay-h-5;   
25627                     }
25628                 }
25629                 xy = [x, y];
25630                 this.storeXY(xy);
25631                 supr.setXY.call(this, xy);
25632                 this.sync();
25633             }
25634         }
25635     },
25636
25637     isVisible : function(){
25638         return this.visible;    
25639     },
25640
25641     // private
25642     showAction : function(){
25643         this.visible = true; // track visibility to prevent getStyle calls
25644         if(this.useDisplay === true){
25645             this.setDisplayed("");
25646         }else if(this.lastXY){
25647             supr.setXY.call(this, this.lastXY);
25648         }else if(this.lastLT){
25649             supr.setLeftTop.call(this, this.lastLT[0], this.lastLT[1]);
25650         }
25651     },
25652
25653     // private
25654     hideAction : function(){
25655         this.visible = false;
25656         if(this.useDisplay === true){
25657             this.setDisplayed(false);
25658         }else{
25659             this.setLeftTop(-10000,-10000);
25660         }
25661     },
25662
25663     // overridden Element method
25664     setVisible : function(v, a, d, c, e){
25665         if(v){
25666             this.showAction();
25667         }
25668         if(a && v){
25669             var cb = function(){
25670                 this.sync(true);
25671                 if(c){
25672                     c();
25673                 }
25674             }.createDelegate(this);
25675             supr.setVisible.call(this, true, true, d, cb, e);
25676         }else{
25677             if(!v){
25678                 this.hideUnders(true);
25679             }
25680             var cb = c;
25681             if(a){
25682                 cb = function(){
25683                     this.hideAction();
25684                     if(c){
25685                         c();
25686                     }
25687                 }.createDelegate(this);
25688             }
25689             supr.setVisible.call(this, v, a, d, cb, e);
25690             if(v){
25691                 this.sync(true);
25692             }else if(!a){
25693                 this.hideAction();
25694             }
25695         }
25696     },
25697
25698     storeXY : function(xy){
25699         delete this.lastLT;
25700         this.lastXY = xy;
25701     },
25702
25703     storeLeftTop : function(left, top){
25704         delete this.lastXY;
25705         this.lastLT = [left, top];
25706     },
25707
25708     // private
25709     beforeFx : function(){
25710         this.beforeAction();
25711         return Roo.Layer.superclass.beforeFx.apply(this, arguments);
25712     },
25713
25714     // private
25715     afterFx : function(){
25716         Roo.Layer.superclass.afterFx.apply(this, arguments);
25717         this.sync(this.isVisible());
25718     },
25719
25720     // private
25721     beforeAction : function(){
25722         if(!this.updating && this.shadow){
25723             this.shadow.hide();
25724         }
25725     },
25726
25727     // overridden Element method
25728     setLeft : function(left){
25729         this.storeLeftTop(left, this.getTop(true));
25730         supr.setLeft.apply(this, arguments);
25731         this.sync();
25732     },
25733
25734     setTop : function(top){
25735         this.storeLeftTop(this.getLeft(true), top);
25736         supr.setTop.apply(this, arguments);
25737         this.sync();
25738     },
25739
25740     setLeftTop : function(left, top){
25741         this.storeLeftTop(left, top);
25742         supr.setLeftTop.apply(this, arguments);
25743         this.sync();
25744     },
25745
25746     setXY : function(xy, a, d, c, e){
25747         this.fixDisplay();
25748         this.beforeAction();
25749         this.storeXY(xy);
25750         var cb = this.createCB(c);
25751         supr.setXY.call(this, xy, a, d, cb, e);
25752         if(!a){
25753             cb();
25754         }
25755     },
25756
25757     // private
25758     createCB : function(c){
25759         var el = this;
25760         return function(){
25761             el.constrainXY();
25762             el.sync(true);
25763             if(c){
25764                 c();
25765             }
25766         };
25767     },
25768
25769     // overridden Element method
25770     setX : function(x, a, d, c, e){
25771         this.setXY([x, this.getY()], a, d, c, e);
25772     },
25773
25774     // overridden Element method
25775     setY : function(y, a, d, c, e){
25776         this.setXY([this.getX(), y], a, d, c, e);
25777     },
25778
25779     // overridden Element method
25780     setSize : function(w, h, a, d, c, e){
25781         this.beforeAction();
25782         var cb = this.createCB(c);
25783         supr.setSize.call(this, w, h, a, d, cb, e);
25784         if(!a){
25785             cb();
25786         }
25787     },
25788
25789     // overridden Element method
25790     setWidth : function(w, a, d, c, e){
25791         this.beforeAction();
25792         var cb = this.createCB(c);
25793         supr.setWidth.call(this, w, a, d, cb, e);
25794         if(!a){
25795             cb();
25796         }
25797     },
25798
25799     // overridden Element method
25800     setHeight : function(h, a, d, c, e){
25801         this.beforeAction();
25802         var cb = this.createCB(c);
25803         supr.setHeight.call(this, h, a, d, cb, e);
25804         if(!a){
25805             cb();
25806         }
25807     },
25808
25809     // overridden Element method
25810     setBounds : function(x, y, w, h, a, d, c, e){
25811         this.beforeAction();
25812         var cb = this.createCB(c);
25813         if(!a){
25814             this.storeXY([x, y]);
25815             supr.setXY.call(this, [x, y]);
25816             supr.setSize.call(this, w, h, a, d, cb, e);
25817             cb();
25818         }else{
25819             supr.setBounds.call(this, x, y, w, h, a, d, cb, e);
25820         }
25821         return this;
25822     },
25823     
25824     /**
25825      * Sets the z-index of this layer and adjusts any shadow and shim z-indexes. The layer z-index is automatically
25826      * incremented by two more than the value passed in so that it always shows above any shadow or shim (the shadow
25827      * element, if any, will be assigned z-index + 1, and the shim element, if any, will be assigned the unmodified z-index).
25828      * @param {Number} zindex The new z-index to set
25829      * @return {this} The Layer
25830      */
25831     setZIndex : function(zindex){
25832         this.zindex = zindex;
25833         this.setStyle("z-index", zindex + 2);
25834         if(this.shadow){
25835             this.shadow.setZIndex(zindex + 1);
25836         }
25837         if(this.shim){
25838             this.shim.setStyle("z-index", zindex);
25839         }
25840     }
25841 });
25842 })();/*
25843  * Based on:
25844  * Ext JS Library 1.1.1
25845  * Copyright(c) 2006-2007, Ext JS, LLC.
25846  *
25847  * Originally Released Under LGPL - original licence link has changed is not relivant.
25848  *
25849  * Fork - LGPL
25850  * <script type="text/javascript">
25851  */
25852
25853
25854 /**
25855  * @class Roo.Shadow
25856  * Simple class that can provide a shadow effect for any element.  Note that the element MUST be absolutely positioned,
25857  * and the shadow does not provide any shimming.  This should be used only in simple cases -- for more advanced
25858  * functionality that can also provide the same shadow effect, see the {@link Roo.Layer} class.
25859  * @constructor
25860  * Create a new Shadow
25861  * @param {Object} config The config object
25862  */
25863 Roo.Shadow = function(config){
25864     Roo.apply(this, config);
25865     if(typeof this.mode != "string"){
25866         this.mode = this.defaultMode;
25867     }
25868     var o = this.offset, a = {h: 0};
25869     var rad = Math.floor(this.offset/2);
25870     switch(this.mode.toLowerCase()){ // all this hideous nonsense calculates the various offsets for shadows
25871         case "drop":
25872             a.w = 0;
25873             a.l = a.t = o;
25874             a.t -= 1;
25875             if(Roo.isIE){
25876                 a.l -= this.offset + rad;
25877                 a.t -= this.offset + rad;
25878                 a.w -= rad;
25879                 a.h -= rad;
25880                 a.t += 1;
25881             }
25882         break;
25883         case "sides":
25884             a.w = (o*2);
25885             a.l = -o;
25886             a.t = o-1;
25887             if(Roo.isIE){
25888                 a.l -= (this.offset - rad);
25889                 a.t -= this.offset + rad;
25890                 a.l += 1;
25891                 a.w -= (this.offset - rad)*2;
25892                 a.w -= rad + 1;
25893                 a.h -= 1;
25894             }
25895         break;
25896         case "frame":
25897             a.w = a.h = (o*2);
25898             a.l = a.t = -o;
25899             a.t += 1;
25900             a.h -= 2;
25901             if(Roo.isIE){
25902                 a.l -= (this.offset - rad);
25903                 a.t -= (this.offset - rad);
25904                 a.l += 1;
25905                 a.w -= (this.offset + rad + 1);
25906                 a.h -= (this.offset + rad);
25907                 a.h += 1;
25908             }
25909         break;
25910     };
25911
25912     this.adjusts = a;
25913 };
25914
25915 Roo.Shadow.prototype = {
25916     /**
25917      * @cfg {String} mode
25918      * The shadow display mode.  Supports the following options:<br />
25919      * sides: Shadow displays on both sides and bottom only<br />
25920      * frame: Shadow displays equally on all four sides<br />
25921      * drop: Traditional bottom-right drop shadow (default)
25922      */
25923     /**
25924      * @cfg {String} offset
25925      * The number of pixels to offset the shadow from the element (defaults to 4)
25926      */
25927     offset: 4,
25928
25929     // private
25930     defaultMode: "drop",
25931
25932     /**
25933      * Displays the shadow under the target element
25934      * @param {String/HTMLElement/Element} targetEl The id or element under which the shadow should display
25935      */
25936     show : function(target){
25937         target = Roo.get(target);
25938         if(!this.el){
25939             this.el = Roo.Shadow.Pool.pull();
25940             if(this.el.dom.nextSibling != target.dom){
25941                 this.el.insertBefore(target);
25942             }
25943         }
25944         this.el.setStyle("z-index", this.zIndex || parseInt(target.getStyle("z-index"), 10)-1);
25945         if(Roo.isIE){
25946             this.el.dom.style.filter="progid:DXImageTransform.Microsoft.alpha(opacity=50) progid:DXImageTransform.Microsoft.Blur(pixelradius="+(this.offset)+")";
25947         }
25948         this.realign(
25949             target.getLeft(true),
25950             target.getTop(true),
25951             target.getWidth(),
25952             target.getHeight()
25953         );
25954         this.el.dom.style.display = "block";
25955     },
25956
25957     /**
25958      * Returns true if the shadow is visible, else false
25959      */
25960     isVisible : function(){
25961         return this.el ? true : false;  
25962     },
25963
25964     /**
25965      * Direct alignment when values are already available. Show must be called at least once before
25966      * calling this method to ensure it is initialized.
25967      * @param {Number} left The target element left position
25968      * @param {Number} top The target element top position
25969      * @param {Number} width The target element width
25970      * @param {Number} height The target element height
25971      */
25972     realign : function(l, t, w, h){
25973         if(!this.el){
25974             return;
25975         }
25976         var a = this.adjusts, d = this.el.dom, s = d.style;
25977         var iea = 0;
25978         s.left = (l+a.l)+"px";
25979         s.top = (t+a.t)+"px";
25980         var sw = (w+a.w), sh = (h+a.h), sws = sw +"px", shs = sh + "px";
25981  
25982         if(s.width != sws || s.height != shs){
25983             s.width = sws;
25984             s.height = shs;
25985             if(!Roo.isIE){
25986                 var cn = d.childNodes;
25987                 var sww = Math.max(0, (sw-12))+"px";
25988                 cn[0].childNodes[1].style.width = sww;
25989                 cn[1].childNodes[1].style.width = sww;
25990                 cn[2].childNodes[1].style.width = sww;
25991                 cn[1].style.height = Math.max(0, (sh-12))+"px";
25992             }
25993         }
25994     },
25995
25996     /**
25997      * Hides this shadow
25998      */
25999     hide : function(){
26000         if(this.el){
26001             this.el.dom.style.display = "none";
26002             Roo.Shadow.Pool.push(this.el);
26003             delete this.el;
26004         }
26005     },
26006
26007     /**
26008      * Adjust the z-index of this shadow
26009      * @param {Number} zindex The new z-index
26010      */
26011     setZIndex : function(z){
26012         this.zIndex = z;
26013         if(this.el){
26014             this.el.setStyle("z-index", z);
26015         }
26016     }
26017 };
26018
26019 // Private utility class that manages the internal Shadow cache
26020 Roo.Shadow.Pool = function(){
26021     var p = [];
26022     var markup = Roo.isIE ?
26023                  '<div class="x-ie-shadow"></div>' :
26024                  '<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>';
26025     return {
26026         pull : function(){
26027             var sh = p.shift();
26028             if(!sh){
26029                 sh = Roo.get(Roo.DomHelper.insertHtml("beforeBegin", document.body.firstChild, markup));
26030                 sh.autoBoxAdjust = false;
26031             }
26032             return sh;
26033         },
26034
26035         push : function(sh){
26036             p.push(sh);
26037         }
26038     };
26039 }();/*
26040  * Based on:
26041  * Ext JS Library 1.1.1
26042  * Copyright(c) 2006-2007, Ext JS, LLC.
26043  *
26044  * Originally Released Under LGPL - original licence link has changed is not relivant.
26045  *
26046  * Fork - LGPL
26047  * <script type="text/javascript">
26048  */
26049
26050
26051 /**
26052  * @class Roo.SplitBar
26053  * @extends Roo.util.Observable
26054  * Creates draggable splitter bar functionality from two elements (element to be dragged and element to be resized).
26055  * <br><br>
26056  * Usage:
26057  * <pre><code>
26058 var split = new Roo.SplitBar("elementToDrag", "elementToSize",
26059                    Roo.SplitBar.HORIZONTAL, Roo.SplitBar.LEFT);
26060 split.setAdapter(new Roo.SplitBar.AbsoluteLayoutAdapter("container"));
26061 split.minSize = 100;
26062 split.maxSize = 600;
26063 split.animate = true;
26064 split.on('moved', splitterMoved);
26065 </code></pre>
26066  * @constructor
26067  * Create a new SplitBar
26068  * @param {String/HTMLElement/Roo.Element} dragElement The element to be dragged and act as the SplitBar. 
26069  * @param {String/HTMLElement/Roo.Element} resizingElement The element to be resized based on where the SplitBar element is dragged 
26070  * @param {Number} orientation (optional) Either Roo.SplitBar.HORIZONTAL or Roo.SplitBar.VERTICAL. (Defaults to HORIZONTAL)
26071  * @param {Number} placement (optional) Either Roo.SplitBar.LEFT or Roo.SplitBar.RIGHT for horizontal or  
26072                         Roo.SplitBar.TOP or Roo.SplitBar.BOTTOM for vertical. (By default, this is determined automatically by the initial
26073                         position of the SplitBar).
26074  */
26075 Roo.SplitBar = function(dragElement, resizingElement, orientation, placement, existingProxy){
26076     
26077     /** @private */
26078     this.el = Roo.get(dragElement, true);
26079     this.el.dom.unselectable = "on";
26080     /** @private */
26081     this.resizingEl = Roo.get(resizingElement, true);
26082
26083     /**
26084      * @private
26085      * The orientation of the split. Either Roo.SplitBar.HORIZONTAL or Roo.SplitBar.VERTICAL. (Defaults to HORIZONTAL)
26086      * Note: If this is changed after creating the SplitBar, the placement property must be manually updated
26087      * @type Number
26088      */
26089     this.orientation = orientation || Roo.SplitBar.HORIZONTAL;
26090     
26091     /**
26092      * The minimum size of the resizing element. (Defaults to 0)
26093      * @type Number
26094      */
26095     this.minSize = 0;
26096     
26097     /**
26098      * The maximum size of the resizing element. (Defaults to 2000)
26099      * @type Number
26100      */
26101     this.maxSize = 2000;
26102     
26103     /**
26104      * Whether to animate the transition to the new size
26105      * @type Boolean
26106      */
26107     this.animate = false;
26108     
26109     /**
26110      * Whether to create a transparent shim that overlays the page when dragging, enables dragging across iframes.
26111      * @type Boolean
26112      */
26113     this.useShim = false;
26114     
26115     /** @private */
26116     this.shim = null;
26117     
26118     if(!existingProxy){
26119         /** @private */
26120         this.proxy = Roo.SplitBar.createProxy(this.orientation);
26121     }else{
26122         this.proxy = Roo.get(existingProxy).dom;
26123     }
26124     /** @private */
26125     this.dd = new Roo.dd.DDProxy(this.el.dom.id, "XSplitBars", {dragElId : this.proxy.id});
26126     
26127     /** @private */
26128     this.dd.b4StartDrag = this.onStartProxyDrag.createDelegate(this);
26129     
26130     /** @private */
26131     this.dd.endDrag = this.onEndProxyDrag.createDelegate(this);
26132     
26133     /** @private */
26134     this.dragSpecs = {};
26135     
26136     /**
26137      * @private The adapter to use to positon and resize elements
26138      */
26139     this.adapter = new Roo.SplitBar.BasicLayoutAdapter();
26140     this.adapter.init(this);
26141     
26142     if(this.orientation == Roo.SplitBar.HORIZONTAL){
26143         /** @private */
26144         this.placement = placement || (this.el.getX() > this.resizingEl.getX() ? Roo.SplitBar.LEFT : Roo.SplitBar.RIGHT);
26145         this.el.addClass("x-splitbar-h");
26146     }else{
26147         /** @private */
26148         this.placement = placement || (this.el.getY() > this.resizingEl.getY() ? Roo.SplitBar.TOP : Roo.SplitBar.BOTTOM);
26149         this.el.addClass("x-splitbar-v");
26150     }
26151     
26152     this.addEvents({
26153         /**
26154          * @event resize
26155          * Fires when the splitter is moved (alias for {@link #event-moved})
26156          * @param {Roo.SplitBar} this
26157          * @param {Number} newSize the new width or height
26158          */
26159         "resize" : true,
26160         /**
26161          * @event moved
26162          * Fires when the splitter is moved
26163          * @param {Roo.SplitBar} this
26164          * @param {Number} newSize the new width or height
26165          */
26166         "moved" : true,
26167         /**
26168          * @event beforeresize
26169          * Fires before the splitter is dragged
26170          * @param {Roo.SplitBar} this
26171          */
26172         "beforeresize" : true,
26173
26174         "beforeapply" : true
26175     });
26176
26177     Roo.util.Observable.call(this);
26178 };
26179
26180 Roo.extend(Roo.SplitBar, Roo.util.Observable, {
26181     onStartProxyDrag : function(x, y){
26182         this.fireEvent("beforeresize", this);
26183         if(!this.overlay){
26184             var o = Roo.DomHelper.insertFirst(document.body,  {cls: "x-drag-overlay", html: "&#160;"}, true);
26185             o.unselectable();
26186             o.enableDisplayMode("block");
26187             // all splitbars share the same overlay
26188             Roo.SplitBar.prototype.overlay = o;
26189         }
26190         this.overlay.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
26191         this.overlay.show();
26192         Roo.get(this.proxy).setDisplayed("block");
26193         var size = this.adapter.getElementSize(this);
26194         this.activeMinSize = this.getMinimumSize();;
26195         this.activeMaxSize = this.getMaximumSize();;
26196         var c1 = size - this.activeMinSize;
26197         var c2 = Math.max(this.activeMaxSize - size, 0);
26198         if(this.orientation == Roo.SplitBar.HORIZONTAL){
26199             this.dd.resetConstraints();
26200             this.dd.setXConstraint(
26201                 this.placement == Roo.SplitBar.LEFT ? c1 : c2, 
26202                 this.placement == Roo.SplitBar.LEFT ? c2 : c1
26203             );
26204             this.dd.setYConstraint(0, 0);
26205         }else{
26206             this.dd.resetConstraints();
26207             this.dd.setXConstraint(0, 0);
26208             this.dd.setYConstraint(
26209                 this.placement == Roo.SplitBar.TOP ? c1 : c2, 
26210                 this.placement == Roo.SplitBar.TOP ? c2 : c1
26211             );
26212          }
26213         this.dragSpecs.startSize = size;
26214         this.dragSpecs.startPoint = [x, y];
26215         Roo.dd.DDProxy.prototype.b4StartDrag.call(this.dd, x, y);
26216     },
26217     
26218     /** 
26219      * @private Called after the drag operation by the DDProxy
26220      */
26221     onEndProxyDrag : function(e){
26222         Roo.get(this.proxy).setDisplayed(false);
26223         var endPoint = Roo.lib.Event.getXY(e);
26224         if(this.overlay){
26225             this.overlay.hide();
26226         }
26227         var newSize;
26228         if(this.orientation == Roo.SplitBar.HORIZONTAL){
26229             newSize = this.dragSpecs.startSize + 
26230                 (this.placement == Roo.SplitBar.LEFT ?
26231                     endPoint[0] - this.dragSpecs.startPoint[0] :
26232                     this.dragSpecs.startPoint[0] - endPoint[0]
26233                 );
26234         }else{
26235             newSize = this.dragSpecs.startSize + 
26236                 (this.placement == Roo.SplitBar.TOP ?
26237                     endPoint[1] - this.dragSpecs.startPoint[1] :
26238                     this.dragSpecs.startPoint[1] - endPoint[1]
26239                 );
26240         }
26241         newSize = Math.min(Math.max(newSize, this.activeMinSize), this.activeMaxSize);
26242         if(newSize != this.dragSpecs.startSize){
26243             if(this.fireEvent('beforeapply', this, newSize) !== false){
26244                 this.adapter.setElementSize(this, newSize);
26245                 this.fireEvent("moved", this, newSize);
26246                 this.fireEvent("resize", this, newSize);
26247             }
26248         }
26249     },
26250     
26251     /**
26252      * Get the adapter this SplitBar uses
26253      * @return The adapter object
26254      */
26255     getAdapter : function(){
26256         return this.adapter;
26257     },
26258     
26259     /**
26260      * Set the adapter this SplitBar uses
26261      * @param {Object} adapter A SplitBar adapter object
26262      */
26263     setAdapter : function(adapter){
26264         this.adapter = adapter;
26265         this.adapter.init(this);
26266     },
26267     
26268     /**
26269      * Gets the minimum size for the resizing element
26270      * @return {Number} The minimum size
26271      */
26272     getMinimumSize : function(){
26273         return this.minSize;
26274     },
26275     
26276     /**
26277      * Sets the minimum size for the resizing element
26278      * @param {Number} minSize The minimum size
26279      */
26280     setMinimumSize : function(minSize){
26281         this.minSize = minSize;
26282     },
26283     
26284     /**
26285      * Gets the maximum size for the resizing element
26286      * @return {Number} The maximum size
26287      */
26288     getMaximumSize : function(){
26289         return this.maxSize;
26290     },
26291     
26292     /**
26293      * Sets the maximum size for the resizing element
26294      * @param {Number} maxSize The maximum size
26295      */
26296     setMaximumSize : function(maxSize){
26297         this.maxSize = maxSize;
26298     },
26299     
26300     /**
26301      * Sets the initialize size for the resizing element
26302      * @param {Number} size The initial size
26303      */
26304     setCurrentSize : function(size){
26305         var oldAnimate = this.animate;
26306         this.animate = false;
26307         this.adapter.setElementSize(this, size);
26308         this.animate = oldAnimate;
26309     },
26310     
26311     /**
26312      * Destroy this splitbar. 
26313      * @param {Boolean} removeEl True to remove the element
26314      */
26315     destroy : function(removeEl){
26316         if(this.shim){
26317             this.shim.remove();
26318         }
26319         this.dd.unreg();
26320         this.proxy.parentNode.removeChild(this.proxy);
26321         if(removeEl){
26322             this.el.remove();
26323         }
26324     }
26325 });
26326
26327 /**
26328  * @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.
26329  */
26330 Roo.SplitBar.createProxy = function(dir){
26331     var proxy = new Roo.Element(document.createElement("div"));
26332     proxy.unselectable();
26333     var cls = 'x-splitbar-proxy';
26334     proxy.addClass(cls + ' ' + (dir == Roo.SplitBar.HORIZONTAL ? cls +'-h' : cls + '-v'));
26335     document.body.appendChild(proxy.dom);
26336     return proxy.dom;
26337 };
26338
26339 /** 
26340  * @class Roo.SplitBar.BasicLayoutAdapter
26341  * Default Adapter. It assumes the splitter and resizing element are not positioned
26342  * elements and only gets/sets the width of the element. Generally used for table based layouts.
26343  */
26344 Roo.SplitBar.BasicLayoutAdapter = function(){
26345 };
26346
26347 Roo.SplitBar.BasicLayoutAdapter.prototype = {
26348     // do nothing for now
26349     init : function(s){
26350     
26351     },
26352     /**
26353      * Called before drag operations to get the current size of the resizing element. 
26354      * @param {Roo.SplitBar} s The SplitBar using this adapter
26355      */
26356      getElementSize : function(s){
26357         if(s.orientation == Roo.SplitBar.HORIZONTAL){
26358             return s.resizingEl.getWidth();
26359         }else{
26360             return s.resizingEl.getHeight();
26361         }
26362     },
26363     
26364     /**
26365      * Called after drag operations to set the size of the resizing element.
26366      * @param {Roo.SplitBar} s The SplitBar using this adapter
26367      * @param {Number} newSize The new size to set
26368      * @param {Function} onComplete A function to be invoked when resizing is complete
26369      */
26370     setElementSize : function(s, newSize, onComplete){
26371         if(s.orientation == Roo.SplitBar.HORIZONTAL){
26372             if(!s.animate){
26373                 s.resizingEl.setWidth(newSize);
26374                 if(onComplete){
26375                     onComplete(s, newSize);
26376                 }
26377             }else{
26378                 s.resizingEl.setWidth(newSize, true, .1, onComplete, 'easeOut');
26379             }
26380         }else{
26381             
26382             if(!s.animate){
26383                 s.resizingEl.setHeight(newSize);
26384                 if(onComplete){
26385                     onComplete(s, newSize);
26386                 }
26387             }else{
26388                 s.resizingEl.setHeight(newSize, true, .1, onComplete, 'easeOut');
26389             }
26390         }
26391     }
26392 };
26393
26394 /** 
26395  *@class Roo.SplitBar.AbsoluteLayoutAdapter
26396  * @extends Roo.SplitBar.BasicLayoutAdapter
26397  * Adapter that  moves the splitter element to align with the resized sizing element. 
26398  * Used with an absolute positioned SplitBar.
26399  * @param {String/HTMLElement/Roo.Element} container The container that wraps around the absolute positioned content. If it's
26400  * document.body, make sure you assign an id to the body element.
26401  */
26402 Roo.SplitBar.AbsoluteLayoutAdapter = function(container){
26403     this.basic = new Roo.SplitBar.BasicLayoutAdapter();
26404     this.container = Roo.get(container);
26405 };
26406
26407 Roo.SplitBar.AbsoluteLayoutAdapter.prototype = {
26408     init : function(s){
26409         this.basic.init(s);
26410     },
26411     
26412     getElementSize : function(s){
26413         return this.basic.getElementSize(s);
26414     },
26415     
26416     setElementSize : function(s, newSize, onComplete){
26417         this.basic.setElementSize(s, newSize, this.moveSplitter.createDelegate(this, [s]));
26418     },
26419     
26420     moveSplitter : function(s){
26421         var yes = Roo.SplitBar;
26422         switch(s.placement){
26423             case yes.LEFT:
26424                 s.el.setX(s.resizingEl.getRight());
26425                 break;
26426             case yes.RIGHT:
26427                 s.el.setStyle("right", (this.container.getWidth() - s.resizingEl.getLeft()) + "px");
26428                 break;
26429             case yes.TOP:
26430                 s.el.setY(s.resizingEl.getBottom());
26431                 break;
26432             case yes.BOTTOM:
26433                 s.el.setY(s.resizingEl.getTop() - s.el.getHeight());
26434                 break;
26435         }
26436     }
26437 };
26438
26439 /**
26440  * Orientation constant - Create a vertical SplitBar
26441  * @static
26442  * @type Number
26443  */
26444 Roo.SplitBar.VERTICAL = 1;
26445
26446 /**
26447  * Orientation constant - Create a horizontal SplitBar
26448  * @static
26449  * @type Number
26450  */
26451 Roo.SplitBar.HORIZONTAL = 2;
26452
26453 /**
26454  * Placement constant - The resizing element is to the left of the splitter element
26455  * @static
26456  * @type Number
26457  */
26458 Roo.SplitBar.LEFT = 1;
26459
26460 /**
26461  * Placement constant - The resizing element is to the right of the splitter element
26462  * @static
26463  * @type Number
26464  */
26465 Roo.SplitBar.RIGHT = 2;
26466
26467 /**
26468  * Placement constant - The resizing element is positioned above the splitter element
26469  * @static
26470  * @type Number
26471  */
26472 Roo.SplitBar.TOP = 3;
26473
26474 /**
26475  * Placement constant - The resizing element is positioned under splitter element
26476  * @static
26477  * @type Number
26478  */
26479 Roo.SplitBar.BOTTOM = 4;
26480 /*
26481  * Based on:
26482  * Ext JS Library 1.1.1
26483  * Copyright(c) 2006-2007, Ext JS, LLC.
26484  *
26485  * Originally Released Under LGPL - original licence link has changed is not relivant.
26486  *
26487  * Fork - LGPL
26488  * <script type="text/javascript">
26489  */
26490
26491 /**
26492  * @class Roo.View
26493  * @extends Roo.util.Observable
26494  * Create a "View" for an element based on a data model or UpdateManager and the supplied DomHelper template. 
26495  * This class also supports single and multi selection modes. <br>
26496  * Create a data model bound view:
26497  <pre><code>
26498  var store = new Roo.data.Store(...);
26499
26500  var view = new Roo.View({
26501     el : "my-element",
26502     tpl : '&lt;div id="{0}"&gt;{2} - {1}&lt;/div&gt;', // auto create template
26503  
26504     singleSelect: true,
26505     selectedClass: "ydataview-selected",
26506     store: store
26507  });
26508
26509  // listen for node click?
26510  view.on("click", function(vw, index, node, e){
26511  alert('Node "' + node.id + '" at index: ' + index + " was clicked.");
26512  });
26513
26514  // load XML data
26515  dataModel.load("foobar.xml");
26516  </code></pre>
26517  For an example of creating a JSON/UpdateManager view, see {@link Roo.JsonView}.
26518  * <br><br>
26519  * <b>Note: The root of your template must be a single node. Table/row implementations may work but are not supported due to
26520  * IE"s limited insertion support with tables and Opera"s faulty event bubbling.</b>
26521  * 
26522  * Note: old style constructor is still suported (container, template, config)
26523  * 
26524  * @constructor
26525  * Create a new View
26526  * @param {Object} config The config object
26527  * 
26528  */
26529 Roo.View = function(config, depreciated_tpl, depreciated_config){
26530     
26531     this.parent = false;
26532     
26533     if (typeof(depreciated_tpl) == 'undefined') {
26534         // new way.. - universal constructor.
26535         Roo.apply(this, config);
26536         this.el  = Roo.get(this.el);
26537     } else {
26538         // old format..
26539         this.el  = Roo.get(config);
26540         this.tpl = depreciated_tpl;
26541         Roo.apply(this, depreciated_config);
26542     }
26543     this.wrapEl  = this.el.wrap().wrap();
26544     ///this.el = this.wrapEla.appendChild(document.createElement("div"));
26545     
26546     
26547     if(typeof(this.tpl) == "string"){
26548         this.tpl = new Roo.Template(this.tpl);
26549     } else {
26550         // support xtype ctors..
26551         this.tpl = new Roo.factory(this.tpl, Roo);
26552     }
26553     
26554     
26555     this.tpl.compile();
26556     
26557     /** @private */
26558     this.addEvents({
26559         /**
26560          * @event beforeclick
26561          * Fires before a click is processed. Returns false to cancel the default action.
26562          * @param {Roo.View} this
26563          * @param {Number} index The index of the target node
26564          * @param {HTMLElement} node The target node
26565          * @param {Roo.EventObject} e The raw event object
26566          */
26567             "beforeclick" : true,
26568         /**
26569          * @event click
26570          * Fires when a template node is clicked.
26571          * @param {Roo.View} this
26572          * @param {Number} index The index of the target node
26573          * @param {HTMLElement} node The target node
26574          * @param {Roo.EventObject} e The raw event object
26575          */
26576             "click" : true,
26577         /**
26578          * @event dblclick
26579          * Fires when a template node is double clicked.
26580          * @param {Roo.View} this
26581          * @param {Number} index The index of the target node
26582          * @param {HTMLElement} node The target node
26583          * @param {Roo.EventObject} e The raw event object
26584          */
26585             "dblclick" : true,
26586         /**
26587          * @event contextmenu
26588          * Fires when a template node is right clicked.
26589          * @param {Roo.View} this
26590          * @param {Number} index The index of the target node
26591          * @param {HTMLElement} node The target node
26592          * @param {Roo.EventObject} e The raw event object
26593          */
26594             "contextmenu" : true,
26595         /**
26596          * @event selectionchange
26597          * Fires when the selected nodes change.
26598          * @param {Roo.View} this
26599          * @param {Array} selections Array of the selected nodes
26600          */
26601             "selectionchange" : true,
26602     
26603         /**
26604          * @event beforeselect
26605          * Fires before a selection is made. If any handlers return false, the selection is cancelled.
26606          * @param {Roo.View} this
26607          * @param {HTMLElement} node The node to be selected
26608          * @param {Array} selections Array of currently selected nodes
26609          */
26610             "beforeselect" : true,
26611         /**
26612          * @event preparedata
26613          * Fires on every row to render, to allow you to change the data.
26614          * @param {Roo.View} this
26615          * @param {Object} data to be rendered (change this)
26616          */
26617           "preparedata" : true
26618           
26619           
26620         });
26621
26622
26623
26624     this.el.on({
26625         "click": this.onClick,
26626         "dblclick": this.onDblClick,
26627         "contextmenu": this.onContextMenu,
26628         scope:this
26629     });
26630
26631     this.selections = [];
26632     this.nodes = [];
26633     this.cmp = new Roo.CompositeElementLite([]);
26634     if(this.store){
26635         this.store = Roo.factory(this.store, Roo.data);
26636         this.setStore(this.store, true);
26637     }
26638     
26639     if ( this.footer && this.footer.xtype) {
26640            
26641          var fctr = this.wrapEl.appendChild(document.createElement("div"));
26642         
26643         this.footer.dataSource = this.store;
26644         this.footer.container = fctr;
26645         this.footer = Roo.factory(this.footer, Roo);
26646         fctr.insertFirst(this.el);
26647         
26648         // this is a bit insane - as the paging toolbar seems to detach the el..
26649 //        dom.parentNode.parentNode.parentNode
26650          // they get detached?
26651     }
26652     
26653     
26654     Roo.View.superclass.constructor.call(this);
26655     
26656     
26657 };
26658
26659 Roo.extend(Roo.View, Roo.util.Observable, {
26660     
26661      /**
26662      * @cfg {Roo.data.Store} store Data store to load data from.
26663      */
26664     store : false,
26665     
26666     /**
26667      * @cfg {String|Roo.Element} el The container element.
26668      */
26669     el : '',
26670     
26671     /**
26672      * @cfg {String|Roo.Template} tpl The template used by this View 
26673      */
26674     tpl : false,
26675     /**
26676      * @cfg {String} dataName the named area of the template to use as the data area
26677      *                          Works with domtemplates roo-name="name"
26678      */
26679     dataName: false,
26680     /**
26681      * @cfg {String} selectedClass The css class to add to selected nodes
26682      */
26683     selectedClass : "x-view-selected",
26684      /**
26685      * @cfg {String} emptyText The empty text to show when nothing is loaded.
26686      */
26687     emptyText : "",
26688     
26689     /**
26690      * @cfg {String} text to display on mask (default Loading)
26691      */
26692     mask : false,
26693     /**
26694      * @cfg {Boolean} multiSelect Allow multiple selection
26695      */
26696     multiSelect : false,
26697     /**
26698      * @cfg {Boolean} singleSelect Allow single selection
26699      */
26700     singleSelect:  false,
26701     
26702     /**
26703      * @cfg {Boolean} toggleSelect - selecting 
26704      */
26705     toggleSelect : false,
26706     
26707     /**
26708      * @cfg {Boolean} tickable - selecting 
26709      */
26710     tickable : false,
26711     
26712     /**
26713      * Returns the element this view is bound to.
26714      * @return {Roo.Element}
26715      */
26716     getEl : function(){
26717         return this.wrapEl;
26718     },
26719     
26720     
26721
26722     /**
26723      * Refreshes the view. - called by datachanged on the store. - do not call directly.
26724      */
26725     refresh : function(){
26726         //Roo.log('refresh');
26727         var t = this.tpl;
26728         
26729         // if we are using something like 'domtemplate', then
26730         // the what gets used is:
26731         // t.applySubtemplate(NAME, data, wrapping data..)
26732         // the outer template then get' applied with
26733         //     the store 'extra data'
26734         // and the body get's added to the
26735         //      roo-name="data" node?
26736         //      <span class='roo-tpl-{name}'></span> ?????
26737         
26738         
26739         
26740         this.clearSelections();
26741         this.el.update("");
26742         var html = [];
26743         var records = this.store.getRange();
26744         if(records.length < 1) {
26745             
26746             // is this valid??  = should it render a template??
26747             
26748             this.el.update(this.emptyText);
26749             return;
26750         }
26751         var el = this.el;
26752         if (this.dataName) {
26753             this.el.update(t.apply(this.store.meta)); //????
26754             el = this.el.child('.roo-tpl-' + this.dataName);
26755         }
26756         
26757         for(var i = 0, len = records.length; i < len; i++){
26758             var data = this.prepareData(records[i].data, i, records[i]);
26759             this.fireEvent("preparedata", this, data, i, records[i]);
26760             
26761             var d = Roo.apply({}, data);
26762             
26763             if(this.tickable){
26764                 Roo.apply(d, {'roo-id' : Roo.id()});
26765                 
26766                 var _this = this;
26767             
26768                 Roo.each(this.parent.item, function(item){
26769                     if(item[_this.parent.valueField] != data[_this.parent.valueField]){
26770                         return;
26771                     }
26772                     Roo.apply(d, {'roo-data-checked' : 'checked'});
26773                 });
26774             }
26775             
26776             html[html.length] = Roo.util.Format.trim(
26777                 this.dataName ?
26778                     t.applySubtemplate(this.dataName, d, this.store.meta) :
26779                     t.apply(d)
26780             );
26781         }
26782         
26783         
26784         
26785         el.update(html.join(""));
26786         this.nodes = el.dom.childNodes;
26787         this.updateIndexes(0);
26788     },
26789     
26790
26791     /**
26792      * Function to override to reformat the data that is sent to
26793      * the template for each node.
26794      * DEPRICATED - use the preparedata event handler.
26795      * @param {Array/Object} data The raw data (array of colData for a data model bound view or
26796      * a JSON object for an UpdateManager bound view).
26797      */
26798     prepareData : function(data, index, record)
26799     {
26800         this.fireEvent("preparedata", this, data, index, record);
26801         return data;
26802     },
26803
26804     onUpdate : function(ds, record){
26805         // Roo.log('on update');   
26806         this.clearSelections();
26807         var index = this.store.indexOf(record);
26808         var n = this.nodes[index];
26809         this.tpl.insertBefore(n, this.prepareData(record.data, index, record));
26810         n.parentNode.removeChild(n);
26811         this.updateIndexes(index, index);
26812     },
26813
26814     
26815     
26816 // --------- FIXME     
26817     onAdd : function(ds, records, index)
26818     {
26819         //Roo.log(['on Add', ds, records, index] );        
26820         this.clearSelections();
26821         if(this.nodes.length == 0){
26822             this.refresh();
26823             return;
26824         }
26825         var n = this.nodes[index];
26826         for(var i = 0, len = records.length; i < len; i++){
26827             var d = this.prepareData(records[i].data, i, records[i]);
26828             if(n){
26829                 this.tpl.insertBefore(n, d);
26830             }else{
26831                 
26832                 this.tpl.append(this.el, d);
26833             }
26834         }
26835         this.updateIndexes(index);
26836     },
26837
26838     onRemove : function(ds, record, index){
26839        // Roo.log('onRemove');
26840         this.clearSelections();
26841         var el = this.dataName  ?
26842             this.el.child('.roo-tpl-' + this.dataName) :
26843             this.el; 
26844         
26845         el.dom.removeChild(this.nodes[index]);
26846         this.updateIndexes(index);
26847     },
26848
26849     /**
26850      * Refresh an individual node.
26851      * @param {Number} index
26852      */
26853     refreshNode : function(index){
26854         this.onUpdate(this.store, this.store.getAt(index));
26855     },
26856
26857     updateIndexes : function(startIndex, endIndex){
26858         var ns = this.nodes;
26859         startIndex = startIndex || 0;
26860         endIndex = endIndex || ns.length - 1;
26861         for(var i = startIndex; i <= endIndex; i++){
26862             ns[i].nodeIndex = i;
26863         }
26864     },
26865
26866     /**
26867      * Changes the data store this view uses and refresh the view.
26868      * @param {Store} store
26869      */
26870     setStore : function(store, initial){
26871         if(!initial && this.store){
26872             this.store.un("datachanged", this.refresh);
26873             this.store.un("add", this.onAdd);
26874             this.store.un("remove", this.onRemove);
26875             this.store.un("update", this.onUpdate);
26876             this.store.un("clear", this.refresh);
26877             this.store.un("beforeload", this.onBeforeLoad);
26878             this.store.un("load", this.onLoad);
26879             this.store.un("loadexception", this.onLoad);
26880         }
26881         if(store){
26882           
26883             store.on("datachanged", this.refresh, this);
26884             store.on("add", this.onAdd, this);
26885             store.on("remove", this.onRemove, this);
26886             store.on("update", this.onUpdate, this);
26887             store.on("clear", this.refresh, this);
26888             store.on("beforeload", this.onBeforeLoad, this);
26889             store.on("load", this.onLoad, this);
26890             store.on("loadexception", this.onLoad, this);
26891         }
26892         
26893         if(store){
26894             this.refresh();
26895         }
26896     },
26897     /**
26898      * onbeforeLoad - masks the loading area.
26899      *
26900      */
26901     onBeforeLoad : function(store,opts)
26902     {
26903          //Roo.log('onBeforeLoad');   
26904         if (!opts.add) {
26905             this.el.update("");
26906         }
26907         this.el.mask(this.mask ? this.mask : "Loading" ); 
26908     },
26909     onLoad : function ()
26910     {
26911         this.el.unmask();
26912     },
26913     
26914
26915     /**
26916      * Returns the template node the passed child belongs to or null if it doesn't belong to one.
26917      * @param {HTMLElement} node
26918      * @return {HTMLElement} The template node
26919      */
26920     findItemFromChild : function(node){
26921         var el = this.dataName  ?
26922             this.el.child('.roo-tpl-' + this.dataName,true) :
26923             this.el.dom; 
26924         
26925         if(!node || node.parentNode == el){
26926                     return node;
26927             }
26928             var p = node.parentNode;
26929             while(p && p != el){
26930             if(p.parentNode == el){
26931                 return p;
26932             }
26933             p = p.parentNode;
26934         }
26935             return null;
26936     },
26937
26938     /** @ignore */
26939     onClick : function(e){
26940         var item = this.findItemFromChild(e.getTarget());
26941         if(item){
26942             var index = this.indexOf(item);
26943             if(this.onItemClick(item, index, e) !== false){
26944                 this.fireEvent("click", this, index, item, e);
26945             }
26946         }else{
26947             this.clearSelections();
26948         }
26949     },
26950
26951     /** @ignore */
26952     onContextMenu : function(e){
26953         var item = this.findItemFromChild(e.getTarget());
26954         if(item){
26955             this.fireEvent("contextmenu", this, this.indexOf(item), item, e);
26956         }
26957     },
26958
26959     /** @ignore */
26960     onDblClick : function(e){
26961         var item = this.findItemFromChild(e.getTarget());
26962         if(item){
26963             this.fireEvent("dblclick", this, this.indexOf(item), item, e);
26964         }
26965     },
26966
26967     onItemClick : function(item, index, e)
26968     {
26969         if(this.fireEvent("beforeclick", this, index, item, e) === false){
26970             return false;
26971         }
26972         if (this.toggleSelect) {
26973             var m = this.isSelected(item) ? 'unselect' : 'select';
26974             //Roo.log(m);
26975             var _t = this;
26976             _t[m](item, true, false);
26977             return true;
26978         }
26979         if(this.multiSelect || this.singleSelect){
26980             if(this.multiSelect && e.shiftKey && this.lastSelection){
26981                 this.select(this.getNodes(this.indexOf(this.lastSelection), index), false);
26982             }else{
26983                 this.select(item, this.multiSelect && e.ctrlKey);
26984                 this.lastSelection = item;
26985             }
26986             
26987             if(!this.tickable){
26988                 e.preventDefault();
26989             }
26990             
26991         }
26992         return true;
26993     },
26994
26995     /**
26996      * Get the number of selected nodes.
26997      * @return {Number}
26998      */
26999     getSelectionCount : function(){
27000         return this.selections.length;
27001     },
27002
27003     /**
27004      * Get the currently selected nodes.
27005      * @return {Array} An array of HTMLElements
27006      */
27007     getSelectedNodes : function(){
27008         return this.selections;
27009     },
27010
27011     /**
27012      * Get the indexes of the selected nodes.
27013      * @return {Array}
27014      */
27015     getSelectedIndexes : function(){
27016         var indexes = [], s = this.selections;
27017         for(var i = 0, len = s.length; i < len; i++){
27018             indexes.push(s[i].nodeIndex);
27019         }
27020         return indexes;
27021     },
27022
27023     /**
27024      * Clear all selections
27025      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange event
27026      */
27027     clearSelections : function(suppressEvent){
27028         if(this.nodes && (this.multiSelect || this.singleSelect) && this.selections.length > 0){
27029             this.cmp.elements = this.selections;
27030             this.cmp.removeClass(this.selectedClass);
27031             this.selections = [];
27032             if(!suppressEvent){
27033                 this.fireEvent("selectionchange", this, this.selections);
27034             }
27035         }
27036     },
27037
27038     /**
27039      * Returns true if the passed node is selected
27040      * @param {HTMLElement/Number} node The node or node index
27041      * @return {Boolean}
27042      */
27043     isSelected : function(node){
27044         var s = this.selections;
27045         if(s.length < 1){
27046             return false;
27047         }
27048         node = this.getNode(node);
27049         return s.indexOf(node) !== -1;
27050     },
27051
27052     /**
27053      * Selects nodes.
27054      * @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
27055      * @param {Boolean} keepExisting (optional) true to keep existing selections
27056      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange vent
27057      */
27058     select : function(nodeInfo, keepExisting, suppressEvent){
27059         if(nodeInfo instanceof Array){
27060             if(!keepExisting){
27061                 this.clearSelections(true);
27062             }
27063             for(var i = 0, len = nodeInfo.length; i < len; i++){
27064                 this.select(nodeInfo[i], true, true);
27065             }
27066             return;
27067         } 
27068         var node = this.getNode(nodeInfo);
27069         if(!node || this.isSelected(node)){
27070             return; // already selected.
27071         }
27072         if(!keepExisting){
27073             this.clearSelections(true);
27074         }
27075         
27076         if(this.fireEvent("beforeselect", this, node, this.selections) !== false){
27077             Roo.fly(node).addClass(this.selectedClass);
27078             this.selections.push(node);
27079             if(!suppressEvent){
27080                 this.fireEvent("selectionchange", this, this.selections);
27081             }
27082         }
27083         
27084         
27085     },
27086       /**
27087      * Unselects nodes.
27088      * @param {Array/HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node, id of a template node or an array of any of those to select
27089      * @param {Boolean} keepExisting (optional) true IGNORED (for campatibility with select)
27090      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange vent
27091      */
27092     unselect : function(nodeInfo, keepExisting, suppressEvent)
27093     {
27094         if(nodeInfo instanceof Array){
27095             Roo.each(this.selections, function(s) {
27096                 this.unselect(s, nodeInfo);
27097             }, this);
27098             return;
27099         }
27100         var node = this.getNode(nodeInfo);
27101         if(!node || !this.isSelected(node)){
27102             //Roo.log("not selected");
27103             return; // not selected.
27104         }
27105         // fireevent???
27106         var ns = [];
27107         Roo.each(this.selections, function(s) {
27108             if (s == node ) {
27109                 Roo.fly(node).removeClass(this.selectedClass);
27110
27111                 return;
27112             }
27113             ns.push(s);
27114         },this);
27115         
27116         this.selections= ns;
27117         this.fireEvent("selectionchange", this, this.selections);
27118     },
27119
27120     /**
27121      * Gets a template node.
27122      * @param {HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node or the id of a template node
27123      * @return {HTMLElement} The node or null if it wasn't found
27124      */
27125     getNode : function(nodeInfo){
27126         if(typeof nodeInfo == "string"){
27127             return document.getElementById(nodeInfo);
27128         }else if(typeof nodeInfo == "number"){
27129             return this.nodes[nodeInfo];
27130         }
27131         return nodeInfo;
27132     },
27133
27134     /**
27135      * Gets a range template nodes.
27136      * @param {Number} startIndex
27137      * @param {Number} endIndex
27138      * @return {Array} An array of nodes
27139      */
27140     getNodes : function(start, end){
27141         var ns = this.nodes;
27142         start = start || 0;
27143         end = typeof end == "undefined" ? ns.length - 1 : end;
27144         var nodes = [];
27145         if(start <= end){
27146             for(var i = start; i <= end; i++){
27147                 nodes.push(ns[i]);
27148             }
27149         } else{
27150             for(var i = start; i >= end; i--){
27151                 nodes.push(ns[i]);
27152             }
27153         }
27154         return nodes;
27155     },
27156
27157     /**
27158      * Finds the index of the passed node
27159      * @param {HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node or the id of a template node
27160      * @return {Number} The index of the node or -1
27161      */
27162     indexOf : function(node){
27163         node = this.getNode(node);
27164         if(typeof node.nodeIndex == "number"){
27165             return node.nodeIndex;
27166         }
27167         var ns = this.nodes;
27168         for(var i = 0, len = ns.length; i < len; i++){
27169             if(ns[i] == node){
27170                 return i;
27171             }
27172         }
27173         return -1;
27174     }
27175 });
27176 /*
27177  * Based on:
27178  * Ext JS Library 1.1.1
27179  * Copyright(c) 2006-2007, Ext JS, LLC.
27180  *
27181  * Originally Released Under LGPL - original licence link has changed is not relivant.
27182  *
27183  * Fork - LGPL
27184  * <script type="text/javascript">
27185  */
27186
27187 /**
27188  * @class Roo.JsonView
27189  * @extends Roo.View
27190  * Shortcut class to create a JSON + {@link Roo.UpdateManager} template view. Usage:
27191 <pre><code>
27192 var view = new Roo.JsonView({
27193     container: "my-element",
27194     tpl: '&lt;div id="{id}"&gt;{foo} - {bar}&lt;/div&gt;', // auto create template
27195     multiSelect: true, 
27196     jsonRoot: "data" 
27197 });
27198
27199 // listen for node click?
27200 view.on("click", function(vw, index, node, e){
27201     alert('Node "' + node.id + '" at index: ' + index + " was clicked.");
27202 });
27203
27204 // direct load of JSON data
27205 view.load("foobar.php");
27206
27207 // Example from my blog list
27208 var tpl = new Roo.Template(
27209     '&lt;div class="entry"&gt;' +
27210     '&lt;a class="entry-title" href="{link}"&gt;{title}&lt;/a&gt;' +
27211     "&lt;h4&gt;{date} by {author} | {comments} Comments&lt;/h4&gt;{description}" +
27212     "&lt;/div&gt;&lt;hr /&gt;"
27213 );
27214
27215 var moreView = new Roo.JsonView({
27216     container :  "entry-list", 
27217     template : tpl,
27218     jsonRoot: "posts"
27219 });
27220 moreView.on("beforerender", this.sortEntries, this);
27221 moreView.load({
27222     url: "/blog/get-posts.php",
27223     params: "allposts=true",
27224     text: "Loading Blog Entries..."
27225 });
27226 </code></pre>
27227
27228 * Note: old code is supported with arguments : (container, template, config)
27229
27230
27231  * @constructor
27232  * Create a new JsonView
27233  * 
27234  * @param {Object} config The config object
27235  * 
27236  */
27237 Roo.JsonView = function(config, depreciated_tpl, depreciated_config){
27238     
27239     
27240     Roo.JsonView.superclass.constructor.call(this, config, depreciated_tpl, depreciated_config);
27241
27242     var um = this.el.getUpdateManager();
27243     um.setRenderer(this);
27244     um.on("update", this.onLoad, this);
27245     um.on("failure", this.onLoadException, this);
27246
27247     /**
27248      * @event beforerender
27249      * Fires before rendering of the downloaded JSON data.
27250      * @param {Roo.JsonView} this
27251      * @param {Object} data The JSON data loaded
27252      */
27253     /**
27254      * @event load
27255      * Fires when data is loaded.
27256      * @param {Roo.JsonView} this
27257      * @param {Object} data The JSON data loaded
27258      * @param {Object} response The raw Connect response object
27259      */
27260     /**
27261      * @event loadexception
27262      * Fires when loading fails.
27263      * @param {Roo.JsonView} this
27264      * @param {Object} response The raw Connect response object
27265      */
27266     this.addEvents({
27267         'beforerender' : true,
27268         'load' : true,
27269         'loadexception' : true
27270     });
27271 };
27272 Roo.extend(Roo.JsonView, Roo.View, {
27273     /**
27274      * @type {String} The root property in the loaded JSON object that contains the data
27275      */
27276     jsonRoot : "",
27277
27278     /**
27279      * Refreshes the view.
27280      */
27281     refresh : function(){
27282         this.clearSelections();
27283         this.el.update("");
27284         var html = [];
27285         var o = this.jsonData;
27286         if(o && o.length > 0){
27287             for(var i = 0, len = o.length; i < len; i++){
27288                 var data = this.prepareData(o[i], i, o);
27289                 html[html.length] = this.tpl.apply(data);
27290             }
27291         }else{
27292             html.push(this.emptyText);
27293         }
27294         this.el.update(html.join(""));
27295         this.nodes = this.el.dom.childNodes;
27296         this.updateIndexes(0);
27297     },
27298
27299     /**
27300      * 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.
27301      * @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:
27302      <pre><code>
27303      view.load({
27304          url: "your-url.php",
27305          params: {param1: "foo", param2: "bar"}, // or a URL encoded string
27306          callback: yourFunction,
27307          scope: yourObject, //(optional scope)
27308          discardUrl: false,
27309          nocache: false,
27310          text: "Loading...",
27311          timeout: 30,
27312          scripts: false
27313      });
27314      </code></pre>
27315      * The only required property is <i>url</i>. The optional properties <i>nocache</i>, <i>text</i> and <i>scripts</i>
27316      * 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.
27317      * @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}
27318      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
27319      * @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.
27320      */
27321     load : function(){
27322         var um = this.el.getUpdateManager();
27323         um.update.apply(um, arguments);
27324     },
27325
27326     // note - render is a standard framework call...
27327     // using it for the response is really flaky... - it's called by UpdateManager normally, except when called by the XComponent/addXtype.
27328     render : function(el, response){
27329         
27330         this.clearSelections();
27331         this.el.update("");
27332         var o;
27333         try{
27334             if (response != '') {
27335                 o = Roo.util.JSON.decode(response.responseText);
27336                 if(this.jsonRoot){
27337                     
27338                     o = o[this.jsonRoot];
27339                 }
27340             }
27341         } catch(e){
27342         }
27343         /**
27344          * The current JSON data or null
27345          */
27346         this.jsonData = o;
27347         this.beforeRender();
27348         this.refresh();
27349     },
27350
27351 /**
27352  * Get the number of records in the current JSON dataset
27353  * @return {Number}
27354  */
27355     getCount : function(){
27356         return this.jsonData ? this.jsonData.length : 0;
27357     },
27358
27359 /**
27360  * Returns the JSON object for the specified node(s)
27361  * @param {HTMLElement/Array} node The node or an array of nodes
27362  * @return {Object/Array} If you pass in an array, you get an array back, otherwise
27363  * you get the JSON object for the node
27364  */
27365     getNodeData : function(node){
27366         if(node instanceof Array){
27367             var data = [];
27368             for(var i = 0, len = node.length; i < len; i++){
27369                 data.push(this.getNodeData(node[i]));
27370             }
27371             return data;
27372         }
27373         return this.jsonData[this.indexOf(node)] || null;
27374     },
27375
27376     beforeRender : function(){
27377         this.snapshot = this.jsonData;
27378         if(this.sortInfo){
27379             this.sort.apply(this, this.sortInfo);
27380         }
27381         this.fireEvent("beforerender", this, this.jsonData);
27382     },
27383
27384     onLoad : function(el, o){
27385         this.fireEvent("load", this, this.jsonData, o);
27386     },
27387
27388     onLoadException : function(el, o){
27389         this.fireEvent("loadexception", this, o);
27390     },
27391
27392 /**
27393  * Filter the data by a specific property.
27394  * @param {String} property A property on your JSON objects
27395  * @param {String/RegExp} value Either string that the property values
27396  * should start with, or a RegExp to test against the property
27397  */
27398     filter : function(property, value){
27399         if(this.jsonData){
27400             var data = [];
27401             var ss = this.snapshot;
27402             if(typeof value == "string"){
27403                 var vlen = value.length;
27404                 if(vlen == 0){
27405                     this.clearFilter();
27406                     return;
27407                 }
27408                 value = value.toLowerCase();
27409                 for(var i = 0, len = ss.length; i < len; i++){
27410                     var o = ss[i];
27411                     if(o[property].substr(0, vlen).toLowerCase() == value){
27412                         data.push(o);
27413                     }
27414                 }
27415             } else if(value.exec){ // regex?
27416                 for(var i = 0, len = ss.length; i < len; i++){
27417                     var o = ss[i];
27418                     if(value.test(o[property])){
27419                         data.push(o);
27420                     }
27421                 }
27422             } else{
27423                 return;
27424             }
27425             this.jsonData = data;
27426             this.refresh();
27427         }
27428     },
27429
27430 /**
27431  * Filter by a function. The passed function will be called with each
27432  * object in the current dataset. If the function returns true the value is kept,
27433  * otherwise it is filtered.
27434  * @param {Function} fn
27435  * @param {Object} scope (optional) The scope of the function (defaults to this JsonView)
27436  */
27437     filterBy : function(fn, scope){
27438         if(this.jsonData){
27439             var data = [];
27440             var ss = this.snapshot;
27441             for(var i = 0, len = ss.length; i < len; i++){
27442                 var o = ss[i];
27443                 if(fn.call(scope || this, o)){
27444                     data.push(o);
27445                 }
27446             }
27447             this.jsonData = data;
27448             this.refresh();
27449         }
27450     },
27451
27452 /**
27453  * Clears the current filter.
27454  */
27455     clearFilter : function(){
27456         if(this.snapshot && this.jsonData != this.snapshot){
27457             this.jsonData = this.snapshot;
27458             this.refresh();
27459         }
27460     },
27461
27462
27463 /**
27464  * Sorts the data for this view and refreshes it.
27465  * @param {String} property A property on your JSON objects to sort on
27466  * @param {String} direction (optional) "desc" or "asc" (defaults to "asc")
27467  * @param {Function} sortType (optional) A function to call to convert the data to a sortable value.
27468  */
27469     sort : function(property, dir, sortType){
27470         this.sortInfo = Array.prototype.slice.call(arguments, 0);
27471         if(this.jsonData){
27472             var p = property;
27473             var dsc = dir && dir.toLowerCase() == "desc";
27474             var f = function(o1, o2){
27475                 var v1 = sortType ? sortType(o1[p]) : o1[p];
27476                 var v2 = sortType ? sortType(o2[p]) : o2[p];
27477                 ;
27478                 if(v1 < v2){
27479                     return dsc ? +1 : -1;
27480                 } else if(v1 > v2){
27481                     return dsc ? -1 : +1;
27482                 } else{
27483                     return 0;
27484                 }
27485             };
27486             this.jsonData.sort(f);
27487             this.refresh();
27488             if(this.jsonData != this.snapshot){
27489                 this.snapshot.sort(f);
27490             }
27491         }
27492     }
27493 });/*
27494  * Based on:
27495  * Ext JS Library 1.1.1
27496  * Copyright(c) 2006-2007, Ext JS, LLC.
27497  *
27498  * Originally Released Under LGPL - original licence link has changed is not relivant.
27499  *
27500  * Fork - LGPL
27501  * <script type="text/javascript">
27502  */
27503  
27504
27505 /**
27506  * @class Roo.ColorPalette
27507  * @extends Roo.Component
27508  * Simple color palette class for choosing colors.  The palette can be rendered to any container.<br />
27509  * Here's an example of typical usage:
27510  * <pre><code>
27511 var cp = new Roo.ColorPalette({value:'993300'});  // initial selected color
27512 cp.render('my-div');
27513
27514 cp.on('select', function(palette, selColor){
27515     // do something with selColor
27516 });
27517 </code></pre>
27518  * @constructor
27519  * Create a new ColorPalette
27520  * @param {Object} config The config object
27521  */
27522 Roo.ColorPalette = function(config){
27523     Roo.ColorPalette.superclass.constructor.call(this, config);
27524     this.addEvents({
27525         /**
27526              * @event select
27527              * Fires when a color is selected
27528              * @param {ColorPalette} this
27529              * @param {String} color The 6-digit color hex code (without the # symbol)
27530              */
27531         select: true
27532     });
27533
27534     if(this.handler){
27535         this.on("select", this.handler, this.scope, true);
27536     }
27537 };
27538 Roo.extend(Roo.ColorPalette, Roo.Component, {
27539     /**
27540      * @cfg {String} itemCls
27541      * The CSS class to apply to the containing element (defaults to "x-color-palette")
27542      */
27543     itemCls : "x-color-palette",
27544     /**
27545      * @cfg {String} value
27546      * The initial color to highlight (should be a valid 6-digit color hex code without the # symbol).  Note that
27547      * the hex codes are case-sensitive.
27548      */
27549     value : null,
27550     clickEvent:'click',
27551     // private
27552     ctype: "Roo.ColorPalette",
27553
27554     /**
27555      * @cfg {Boolean} allowReselect If set to true then reselecting a color that is already selected fires the selection event
27556      */
27557     allowReselect : false,
27558
27559     /**
27560      * <p>An array of 6-digit color hex code strings (without the # symbol).  This array can contain any number
27561      * of colors, and each hex code should be unique.  The width of the palette is controlled via CSS by adjusting
27562      * the width property of the 'x-color-palette' class (or assigning a custom class), so you can balance the number
27563      * of colors with the width setting until the box is symmetrical.</p>
27564      * <p>You can override individual colors if needed:</p>
27565      * <pre><code>
27566 var cp = new Roo.ColorPalette();
27567 cp.colors[0] = "FF0000";  // change the first box to red
27568 </code></pre>
27569
27570 Or you can provide a custom array of your own for complete control:
27571 <pre><code>
27572 var cp = new Roo.ColorPalette();
27573 cp.colors = ["000000", "993300", "333300"];
27574 </code></pre>
27575      * @type Array
27576      */
27577     colors : [
27578         "000000", "993300", "333300", "003300", "003366", "000080", "333399", "333333",
27579         "800000", "FF6600", "808000", "008000", "008080", "0000FF", "666699", "808080",
27580         "FF0000", "FF9900", "99CC00", "339966", "33CCCC", "3366FF", "800080", "969696",
27581         "FF00FF", "FFCC00", "FFFF00", "00FF00", "00FFFF", "00CCFF", "993366", "C0C0C0",
27582         "FF99CC", "FFCC99", "FFFF99", "CCFFCC", "CCFFFF", "99CCFF", "CC99FF", "FFFFFF"
27583     ],
27584
27585     // private
27586     onRender : function(container, position){
27587         var t = new Roo.MasterTemplate(
27588             '<tpl><a href="#" class="color-{0}" hidefocus="on"><em><span style="background:#{0}" unselectable="on">&#160;</span></em></a></tpl>'
27589         );
27590         var c = this.colors;
27591         for(var i = 0, len = c.length; i < len; i++){
27592             t.add([c[i]]);
27593         }
27594         var el = document.createElement("div");
27595         el.className = this.itemCls;
27596         t.overwrite(el);
27597         container.dom.insertBefore(el, position);
27598         this.el = Roo.get(el);
27599         this.el.on(this.clickEvent, this.handleClick,  this, {delegate: "a"});
27600         if(this.clickEvent != 'click'){
27601             this.el.on('click', Roo.emptyFn,  this, {delegate: "a", preventDefault:true});
27602         }
27603     },
27604
27605     // private
27606     afterRender : function(){
27607         Roo.ColorPalette.superclass.afterRender.call(this);
27608         if(this.value){
27609             var s = this.value;
27610             this.value = null;
27611             this.select(s);
27612         }
27613     },
27614
27615     // private
27616     handleClick : function(e, t){
27617         e.preventDefault();
27618         if(!this.disabled){
27619             var c = t.className.match(/(?:^|\s)color-(.{6})(?:\s|$)/)[1];
27620             this.select(c.toUpperCase());
27621         }
27622     },
27623
27624     /**
27625      * Selects the specified color in the palette (fires the select event)
27626      * @param {String} color A valid 6-digit color hex code (# will be stripped if included)
27627      */
27628     select : function(color){
27629         color = color.replace("#", "");
27630         if(color != this.value || this.allowReselect){
27631             var el = this.el;
27632             if(this.value){
27633                 el.child("a.color-"+this.value).removeClass("x-color-palette-sel");
27634             }
27635             el.child("a.color-"+color).addClass("x-color-palette-sel");
27636             this.value = color;
27637             this.fireEvent("select", this, color);
27638         }
27639     }
27640 });/*
27641  * Based on:
27642  * Ext JS Library 1.1.1
27643  * Copyright(c) 2006-2007, Ext JS, LLC.
27644  *
27645  * Originally Released Under LGPL - original licence link has changed is not relivant.
27646  *
27647  * Fork - LGPL
27648  * <script type="text/javascript">
27649  */
27650  
27651 /**
27652  * @class Roo.DatePicker
27653  * @extends Roo.Component
27654  * Simple date picker class.
27655  * @constructor
27656  * Create a new DatePicker
27657  * @param {Object} config The config object
27658  */
27659 Roo.DatePicker = function(config){
27660     Roo.DatePicker.superclass.constructor.call(this, config);
27661
27662     this.value = config && config.value ?
27663                  config.value.clearTime() : new Date().clearTime();
27664
27665     this.addEvents({
27666         /**
27667              * @event select
27668              * Fires when a date is selected
27669              * @param {DatePicker} this
27670              * @param {Date} date The selected date
27671              */
27672         'select': true,
27673         /**
27674              * @event monthchange
27675              * Fires when the displayed month changes 
27676              * @param {DatePicker} this
27677              * @param {Date} date The selected month
27678              */
27679         'monthchange': true
27680     });
27681
27682     if(this.handler){
27683         this.on("select", this.handler,  this.scope || this);
27684     }
27685     // build the disabledDatesRE
27686     if(!this.disabledDatesRE && this.disabledDates){
27687         var dd = this.disabledDates;
27688         var re = "(?:";
27689         for(var i = 0; i < dd.length; i++){
27690             re += dd[i];
27691             if(i != dd.length-1) {
27692                 re += "|";
27693             }
27694         }
27695         this.disabledDatesRE = new RegExp(re + ")");
27696     }
27697 };
27698
27699 Roo.extend(Roo.DatePicker, Roo.Component, {
27700     /**
27701      * @cfg {String} todayText
27702      * The text to display on the button that selects the current date (defaults to "Today")
27703      */
27704     todayText : "Today",
27705     /**
27706      * @cfg {String} okText
27707      * The text to display on the ok button
27708      */
27709     okText : "&#160;OK&#160;", // &#160; to give the user extra clicking room
27710     /**
27711      * @cfg {String} cancelText
27712      * The text to display on the cancel button
27713      */
27714     cancelText : "Cancel",
27715     /**
27716      * @cfg {String} todayTip
27717      * The tooltip to display for the button that selects the current date (defaults to "{current date} (Spacebar)")
27718      */
27719     todayTip : "{0} (Spacebar)",
27720     /**
27721      * @cfg {Date} minDate
27722      * Minimum allowable date (JavaScript date object, defaults to null)
27723      */
27724     minDate : null,
27725     /**
27726      * @cfg {Date} maxDate
27727      * Maximum allowable date (JavaScript date object, defaults to null)
27728      */
27729     maxDate : null,
27730     /**
27731      * @cfg {String} minText
27732      * The error text to display if the minDate validation fails (defaults to "This date is before the minimum date")
27733      */
27734     minText : "This date is before the minimum date",
27735     /**
27736      * @cfg {String} maxText
27737      * The error text to display if the maxDate validation fails (defaults to "This date is after the maximum date")
27738      */
27739     maxText : "This date is after the maximum date",
27740     /**
27741      * @cfg {String} format
27742      * The default date format string which can be overriden for localization support.  The format must be
27743      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
27744      */
27745     format : "m/d/y",
27746     /**
27747      * @cfg {Array} disabledDays
27748      * An array of days to disable, 0-based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
27749      */
27750     disabledDays : null,
27751     /**
27752      * @cfg {String} disabledDaysText
27753      * The tooltip to display when the date falls on a disabled day (defaults to "")
27754      */
27755     disabledDaysText : "",
27756     /**
27757      * @cfg {RegExp} disabledDatesRE
27758      * JavaScript regular expression used to disable a pattern of dates (defaults to null)
27759      */
27760     disabledDatesRE : null,
27761     /**
27762      * @cfg {String} disabledDatesText
27763      * The tooltip text to display when the date falls on a disabled date (defaults to "")
27764      */
27765     disabledDatesText : "",
27766     /**
27767      * @cfg {Boolean} constrainToViewport
27768      * True to constrain the date picker to the viewport (defaults to true)
27769      */
27770     constrainToViewport : true,
27771     /**
27772      * @cfg {Array} monthNames
27773      * An array of textual month names which can be overriden for localization support (defaults to Date.monthNames)
27774      */
27775     monthNames : Date.monthNames,
27776     /**
27777      * @cfg {Array} dayNames
27778      * An array of textual day names which can be overriden for localization support (defaults to Date.dayNames)
27779      */
27780     dayNames : Date.dayNames,
27781     /**
27782      * @cfg {String} nextText
27783      * The next month navigation button tooltip (defaults to 'Next Month (Control+Right)')
27784      */
27785     nextText: 'Next Month (Control+Right)',
27786     /**
27787      * @cfg {String} prevText
27788      * The previous month navigation button tooltip (defaults to 'Previous Month (Control+Left)')
27789      */
27790     prevText: 'Previous Month (Control+Left)',
27791     /**
27792      * @cfg {String} monthYearText
27793      * The header month selector tooltip (defaults to 'Choose a month (Control+Up/Down to move years)')
27794      */
27795     monthYearText: 'Choose a month (Control+Up/Down to move years)',
27796     /**
27797      * @cfg {Number} startDay
27798      * Day index at which the week should begin, 0-based (defaults to 0, which is Sunday)
27799      */
27800     startDay : 0,
27801     /**
27802      * @cfg {Bool} showClear
27803      * Show a clear button (usefull for date form elements that can be blank.)
27804      */
27805     
27806     showClear: false,
27807     
27808     /**
27809      * Sets the value of the date field
27810      * @param {Date} value The date to set
27811      */
27812     setValue : function(value){
27813         var old = this.value;
27814         
27815         if (typeof(value) == 'string') {
27816          
27817             value = Date.parseDate(value, this.format);
27818         }
27819         if (!value) {
27820             value = new Date();
27821         }
27822         
27823         this.value = value.clearTime(true);
27824         if(this.el){
27825             this.update(this.value);
27826         }
27827     },
27828
27829     /**
27830      * Gets the current selected value of the date field
27831      * @return {Date} The selected date
27832      */
27833     getValue : function(){
27834         return this.value;
27835     },
27836
27837     // private
27838     focus : function(){
27839         if(this.el){
27840             this.update(this.activeDate);
27841         }
27842     },
27843
27844     // privateval
27845     onRender : function(container, position){
27846         
27847         var m = [
27848              '<table cellspacing="0">',
27849                 '<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>',
27850                 '<tr><td colspan="3"><table class="x-date-inner" cellspacing="0"><thead><tr>'];
27851         var dn = this.dayNames;
27852         for(var i = 0; i < 7; i++){
27853             var d = this.startDay+i;
27854             if(d > 6){
27855                 d = d-7;
27856             }
27857             m.push("<th><span>", dn[d].substr(0,1), "</span></th>");
27858         }
27859         m[m.length] = "</tr></thead><tbody><tr>";
27860         for(var i = 0; i < 42; i++) {
27861             if(i % 7 == 0 && i != 0){
27862                 m[m.length] = "</tr><tr>";
27863             }
27864             m[m.length] = '<td><a href="#" hidefocus="on" class="x-date-date" tabIndex="1"><em><span></span></em></a></td>';
27865         }
27866         m[m.length] = '</tr></tbody></table></td></tr><tr>'+
27867             '<td colspan="3" class="x-date-bottom" align="center"></td></tr></table><div class="x-date-mp"></div>';
27868
27869         var el = document.createElement("div");
27870         el.className = "x-date-picker";
27871         el.innerHTML = m.join("");
27872
27873         container.dom.insertBefore(el, position);
27874
27875         this.el = Roo.get(el);
27876         this.eventEl = Roo.get(el.firstChild);
27877
27878         new Roo.util.ClickRepeater(this.el.child("td.x-date-left a"), {
27879             handler: this.showPrevMonth,
27880             scope: this,
27881             preventDefault:true,
27882             stopDefault:true
27883         });
27884
27885         new Roo.util.ClickRepeater(this.el.child("td.x-date-right a"), {
27886             handler: this.showNextMonth,
27887             scope: this,
27888             preventDefault:true,
27889             stopDefault:true
27890         });
27891
27892         this.eventEl.on("mousewheel", this.handleMouseWheel,  this);
27893
27894         this.monthPicker = this.el.down('div.x-date-mp');
27895         this.monthPicker.enableDisplayMode('block');
27896         
27897         var kn = new Roo.KeyNav(this.eventEl, {
27898             "left" : function(e){
27899                 e.ctrlKey ?
27900                     this.showPrevMonth() :
27901                     this.update(this.activeDate.add("d", -1));
27902             },
27903
27904             "right" : function(e){
27905                 e.ctrlKey ?
27906                     this.showNextMonth() :
27907                     this.update(this.activeDate.add("d", 1));
27908             },
27909
27910             "up" : function(e){
27911                 e.ctrlKey ?
27912                     this.showNextYear() :
27913                     this.update(this.activeDate.add("d", -7));
27914             },
27915
27916             "down" : function(e){
27917                 e.ctrlKey ?
27918                     this.showPrevYear() :
27919                     this.update(this.activeDate.add("d", 7));
27920             },
27921
27922             "pageUp" : function(e){
27923                 this.showNextMonth();
27924             },
27925
27926             "pageDown" : function(e){
27927                 this.showPrevMonth();
27928             },
27929
27930             "enter" : function(e){
27931                 e.stopPropagation();
27932                 return true;
27933             },
27934
27935             scope : this
27936         });
27937
27938         this.eventEl.on("click", this.handleDateClick,  this, {delegate: "a.x-date-date"});
27939
27940         this.eventEl.addKeyListener(Roo.EventObject.SPACE, this.selectToday,  this);
27941
27942         this.el.unselectable();
27943         
27944         this.cells = this.el.select("table.x-date-inner tbody td");
27945         this.textNodes = this.el.query("table.x-date-inner tbody span");
27946
27947         this.mbtn = new Roo.Button(this.el.child("td.x-date-middle", true), {
27948             text: "&#160;",
27949             tooltip: this.monthYearText
27950         });
27951
27952         this.mbtn.on('click', this.showMonthPicker, this);
27953         this.mbtn.el.child(this.mbtn.menuClassTarget).addClass("x-btn-with-menu");
27954
27955
27956         var today = (new Date()).dateFormat(this.format);
27957         
27958         var baseTb = new Roo.Toolbar(this.el.child("td.x-date-bottom", true));
27959         if (this.showClear) {
27960             baseTb.add( new Roo.Toolbar.Fill());
27961         }
27962         baseTb.add({
27963             text: String.format(this.todayText, today),
27964             tooltip: String.format(this.todayTip, today),
27965             handler: this.selectToday,
27966             scope: this
27967         });
27968         
27969         //var todayBtn = new Roo.Button(this.el.child("td.x-date-bottom", true), {
27970             
27971         //});
27972         if (this.showClear) {
27973             
27974             baseTb.add( new Roo.Toolbar.Fill());
27975             baseTb.add({
27976                 text: '&#160;',
27977                 cls: 'x-btn-icon x-btn-clear',
27978                 handler: function() {
27979                     //this.value = '';
27980                     this.fireEvent("select", this, '');
27981                 },
27982                 scope: this
27983             });
27984         }
27985         
27986         
27987         if(Roo.isIE){
27988             this.el.repaint();
27989         }
27990         this.update(this.value);
27991     },
27992
27993     createMonthPicker : function(){
27994         if(!this.monthPicker.dom.firstChild){
27995             var buf = ['<table border="0" cellspacing="0">'];
27996             for(var i = 0; i < 6; i++){
27997                 buf.push(
27998                     '<tr><td class="x-date-mp-month"><a href="#">', this.monthNames[i].substr(0, 3), '</a></td>',
27999                     '<td class="x-date-mp-month x-date-mp-sep"><a href="#">', this.monthNames[i+6].substr(0, 3), '</a></td>',
28000                     i == 0 ?
28001                     '<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>' :
28002                     '<td class="x-date-mp-year"><a href="#"></a></td><td class="x-date-mp-year"><a href="#"></a></td></tr>'
28003                 );
28004             }
28005             buf.push(
28006                 '<tr class="x-date-mp-btns"><td colspan="4"><button type="button" class="x-date-mp-ok">',
28007                     this.okText,
28008                     '</button><button type="button" class="x-date-mp-cancel">',
28009                     this.cancelText,
28010                     '</button></td></tr>',
28011                 '</table>'
28012             );
28013             this.monthPicker.update(buf.join(''));
28014             this.monthPicker.on('click', this.onMonthClick, this);
28015             this.monthPicker.on('dblclick', this.onMonthDblClick, this);
28016
28017             this.mpMonths = this.monthPicker.select('td.x-date-mp-month');
28018             this.mpYears = this.monthPicker.select('td.x-date-mp-year');
28019
28020             this.mpMonths.each(function(m, a, i){
28021                 i += 1;
28022                 if((i%2) == 0){
28023                     m.dom.xmonth = 5 + Math.round(i * .5);
28024                 }else{
28025                     m.dom.xmonth = Math.round((i-1) * .5);
28026                 }
28027             });
28028         }
28029     },
28030
28031     showMonthPicker : function(){
28032         this.createMonthPicker();
28033         var size = this.el.getSize();
28034         this.monthPicker.setSize(size);
28035         this.monthPicker.child('table').setSize(size);
28036
28037         this.mpSelMonth = (this.activeDate || this.value).getMonth();
28038         this.updateMPMonth(this.mpSelMonth);
28039         this.mpSelYear = (this.activeDate || this.value).getFullYear();
28040         this.updateMPYear(this.mpSelYear);
28041
28042         this.monthPicker.slideIn('t', {duration:.2});
28043     },
28044
28045     updateMPYear : function(y){
28046         this.mpyear = y;
28047         var ys = this.mpYears.elements;
28048         for(var i = 1; i <= 10; i++){
28049             var td = ys[i-1], y2;
28050             if((i%2) == 0){
28051                 y2 = y + Math.round(i * .5);
28052                 td.firstChild.innerHTML = y2;
28053                 td.xyear = y2;
28054             }else{
28055                 y2 = y - (5-Math.round(i * .5));
28056                 td.firstChild.innerHTML = y2;
28057                 td.xyear = y2;
28058             }
28059             this.mpYears.item(i-1)[y2 == this.mpSelYear ? 'addClass' : 'removeClass']('x-date-mp-sel');
28060         }
28061     },
28062
28063     updateMPMonth : function(sm){
28064         this.mpMonths.each(function(m, a, i){
28065             m[m.dom.xmonth == sm ? 'addClass' : 'removeClass']('x-date-mp-sel');
28066         });
28067     },
28068
28069     selectMPMonth: function(m){
28070         
28071     },
28072
28073     onMonthClick : function(e, t){
28074         e.stopEvent();
28075         var el = new Roo.Element(t), pn;
28076         if(el.is('button.x-date-mp-cancel')){
28077             this.hideMonthPicker();
28078         }
28079         else if(el.is('button.x-date-mp-ok')){
28080             this.update(new Date(this.mpSelYear, this.mpSelMonth, (this.activeDate || this.value).getDate()));
28081             this.hideMonthPicker();
28082         }
28083         else if(pn = el.up('td.x-date-mp-month', 2)){
28084             this.mpMonths.removeClass('x-date-mp-sel');
28085             pn.addClass('x-date-mp-sel');
28086             this.mpSelMonth = pn.dom.xmonth;
28087         }
28088         else if(pn = el.up('td.x-date-mp-year', 2)){
28089             this.mpYears.removeClass('x-date-mp-sel');
28090             pn.addClass('x-date-mp-sel');
28091             this.mpSelYear = pn.dom.xyear;
28092         }
28093         else if(el.is('a.x-date-mp-prev')){
28094             this.updateMPYear(this.mpyear-10);
28095         }
28096         else if(el.is('a.x-date-mp-next')){
28097             this.updateMPYear(this.mpyear+10);
28098         }
28099     },
28100
28101     onMonthDblClick : function(e, t){
28102         e.stopEvent();
28103         var el = new Roo.Element(t), pn;
28104         if(pn = el.up('td.x-date-mp-month', 2)){
28105             this.update(new Date(this.mpSelYear, pn.dom.xmonth, (this.activeDate || this.value).getDate()));
28106             this.hideMonthPicker();
28107         }
28108         else if(pn = el.up('td.x-date-mp-year', 2)){
28109             this.update(new Date(pn.dom.xyear, this.mpSelMonth, (this.activeDate || this.value).getDate()));
28110             this.hideMonthPicker();
28111         }
28112     },
28113
28114     hideMonthPicker : function(disableAnim){
28115         if(this.monthPicker){
28116             if(disableAnim === true){
28117                 this.monthPicker.hide();
28118             }else{
28119                 this.monthPicker.slideOut('t', {duration:.2});
28120             }
28121         }
28122     },
28123
28124     // private
28125     showPrevMonth : function(e){
28126         this.update(this.activeDate.add("mo", -1));
28127     },
28128
28129     // private
28130     showNextMonth : function(e){
28131         this.update(this.activeDate.add("mo", 1));
28132     },
28133
28134     // private
28135     showPrevYear : function(){
28136         this.update(this.activeDate.add("y", -1));
28137     },
28138
28139     // private
28140     showNextYear : function(){
28141         this.update(this.activeDate.add("y", 1));
28142     },
28143
28144     // private
28145     handleMouseWheel : function(e){
28146         var delta = e.getWheelDelta();
28147         if(delta > 0){
28148             this.showPrevMonth();
28149             e.stopEvent();
28150         } else if(delta < 0){
28151             this.showNextMonth();
28152             e.stopEvent();
28153         }
28154     },
28155
28156     // private
28157     handleDateClick : function(e, t){
28158         e.stopEvent();
28159         if(t.dateValue && !Roo.fly(t.parentNode).hasClass("x-date-disabled")){
28160             this.setValue(new Date(t.dateValue));
28161             this.fireEvent("select", this, this.value);
28162         }
28163     },
28164
28165     // private
28166     selectToday : function(){
28167         this.setValue(new Date().clearTime());
28168         this.fireEvent("select", this, this.value);
28169     },
28170
28171     // private
28172     update : function(date)
28173     {
28174         var vd = this.activeDate;
28175         this.activeDate = date;
28176         if(vd && this.el){
28177             var t = date.getTime();
28178             if(vd.getMonth() == date.getMonth() && vd.getFullYear() == date.getFullYear()){
28179                 this.cells.removeClass("x-date-selected");
28180                 this.cells.each(function(c){
28181                    if(c.dom.firstChild.dateValue == t){
28182                        c.addClass("x-date-selected");
28183                        setTimeout(function(){
28184                             try{c.dom.firstChild.focus();}catch(e){}
28185                        }, 50);
28186                        return false;
28187                    }
28188                 });
28189                 return;
28190             }
28191         }
28192         
28193         var days = date.getDaysInMonth();
28194         var firstOfMonth = date.getFirstDateOfMonth();
28195         var startingPos = firstOfMonth.getDay()-this.startDay;
28196
28197         if(startingPos <= this.startDay){
28198             startingPos += 7;
28199         }
28200
28201         var pm = date.add("mo", -1);
28202         var prevStart = pm.getDaysInMonth()-startingPos;
28203
28204         var cells = this.cells.elements;
28205         var textEls = this.textNodes;
28206         days += startingPos;
28207
28208         // convert everything to numbers so it's fast
28209         var day = 86400000;
28210         var d = (new Date(pm.getFullYear(), pm.getMonth(), prevStart)).clearTime();
28211         var today = new Date().clearTime().getTime();
28212         var sel = date.clearTime().getTime();
28213         var min = this.minDate ? this.minDate.clearTime() : Number.NEGATIVE_INFINITY;
28214         var max = this.maxDate ? this.maxDate.clearTime() : Number.POSITIVE_INFINITY;
28215         var ddMatch = this.disabledDatesRE;
28216         var ddText = this.disabledDatesText;
28217         var ddays = this.disabledDays ? this.disabledDays.join("") : false;
28218         var ddaysText = this.disabledDaysText;
28219         var format = this.format;
28220
28221         var setCellClass = function(cal, cell){
28222             cell.title = "";
28223             var t = d.getTime();
28224             cell.firstChild.dateValue = t;
28225             if(t == today){
28226                 cell.className += " x-date-today";
28227                 cell.title = cal.todayText;
28228             }
28229             if(t == sel){
28230                 cell.className += " x-date-selected";
28231                 setTimeout(function(){
28232                     try{cell.firstChild.focus();}catch(e){}
28233                 }, 50);
28234             }
28235             // disabling
28236             if(t < min) {
28237                 cell.className = " x-date-disabled";
28238                 cell.title = cal.minText;
28239                 return;
28240             }
28241             if(t > max) {
28242                 cell.className = " x-date-disabled";
28243                 cell.title = cal.maxText;
28244                 return;
28245             }
28246             if(ddays){
28247                 if(ddays.indexOf(d.getDay()) != -1){
28248                     cell.title = ddaysText;
28249                     cell.className = " x-date-disabled";
28250                 }
28251             }
28252             if(ddMatch && format){
28253                 var fvalue = d.dateFormat(format);
28254                 if(ddMatch.test(fvalue)){
28255                     cell.title = ddText.replace("%0", fvalue);
28256                     cell.className = " x-date-disabled";
28257                 }
28258             }
28259         };
28260
28261         var i = 0;
28262         for(; i < startingPos; i++) {
28263             textEls[i].innerHTML = (++prevStart);
28264             d.setDate(d.getDate()+1);
28265             cells[i].className = "x-date-prevday";
28266             setCellClass(this, cells[i]);
28267         }
28268         for(; i < days; i++){
28269             intDay = i - startingPos + 1;
28270             textEls[i].innerHTML = (intDay);
28271             d.setDate(d.getDate()+1);
28272             cells[i].className = "x-date-active";
28273             setCellClass(this, cells[i]);
28274         }
28275         var extraDays = 0;
28276         for(; i < 42; i++) {
28277              textEls[i].innerHTML = (++extraDays);
28278              d.setDate(d.getDate()+1);
28279              cells[i].className = "x-date-nextday";
28280              setCellClass(this, cells[i]);
28281         }
28282
28283         this.mbtn.setText(this.monthNames[date.getMonth()] + " " + date.getFullYear());
28284         this.fireEvent('monthchange', this, date);
28285         
28286         if(!this.internalRender){
28287             var main = this.el.dom.firstChild;
28288             var w = main.offsetWidth;
28289             this.el.setWidth(w + this.el.getBorderWidth("lr"));
28290             Roo.fly(main).setWidth(w);
28291             this.internalRender = true;
28292             // opera does not respect the auto grow header center column
28293             // then, after it gets a width opera refuses to recalculate
28294             // without a second pass
28295             if(Roo.isOpera && !this.secondPass){
28296                 main.rows[0].cells[1].style.width = (w - (main.rows[0].cells[0].offsetWidth+main.rows[0].cells[2].offsetWidth)) + "px";
28297                 this.secondPass = true;
28298                 this.update.defer(10, this, [date]);
28299             }
28300         }
28301         
28302         
28303     }
28304 });        /*
28305  * Based on:
28306  * Ext JS Library 1.1.1
28307  * Copyright(c) 2006-2007, Ext JS, LLC.
28308  *
28309  * Originally Released Under LGPL - original licence link has changed is not relivant.
28310  *
28311  * Fork - LGPL
28312  * <script type="text/javascript">
28313  */
28314 /**
28315  * @class Roo.TabPanel
28316  * @extends Roo.util.Observable
28317  * A lightweight tab container.
28318  * <br><br>
28319  * Usage:
28320  * <pre><code>
28321 // basic tabs 1, built from existing content
28322 var tabs = new Roo.TabPanel("tabs1");
28323 tabs.addTab("script", "View Script");
28324 tabs.addTab("markup", "View Markup");
28325 tabs.activate("script");
28326
28327 // more advanced tabs, built from javascript
28328 var jtabs = new Roo.TabPanel("jtabs");
28329 jtabs.addTab("jtabs-1", "Normal Tab", "My content was added during construction.");
28330
28331 // set up the UpdateManager
28332 var tab2 = jtabs.addTab("jtabs-2", "Ajax Tab 1");
28333 var updater = tab2.getUpdateManager();
28334 updater.setDefaultUrl("ajax1.htm");
28335 tab2.on('activate', updater.refresh, updater, true);
28336
28337 // Use setUrl for Ajax loading
28338 var tab3 = jtabs.addTab("jtabs-3", "Ajax Tab 2");
28339 tab3.setUrl("ajax2.htm", null, true);
28340
28341 // Disabled tab
28342 var tab4 = jtabs.addTab("tabs1-5", "Disabled Tab", "Can't see me cause I'm disabled");
28343 tab4.disable();
28344
28345 jtabs.activate("jtabs-1");
28346  * </code></pre>
28347  * @constructor
28348  * Create a new TabPanel.
28349  * @param {String/HTMLElement/Roo.Element} container The id, DOM element or Roo.Element container where this TabPanel is to be rendered.
28350  * @param {Object/Boolean} config Config object to set any properties for this TabPanel, or true to render the tabs on the bottom.
28351  */
28352 Roo.TabPanel = function(container, config){
28353     /**
28354     * The container element for this TabPanel.
28355     * @type Roo.Element
28356     */
28357     this.el = Roo.get(container, true);
28358     if(config){
28359         if(typeof config == "boolean"){
28360             this.tabPosition = config ? "bottom" : "top";
28361         }else{
28362             Roo.apply(this, config);
28363         }
28364     }
28365     if(this.tabPosition == "bottom"){
28366         this.bodyEl = Roo.get(this.createBody(this.el.dom));
28367         this.el.addClass("x-tabs-bottom");
28368     }
28369     this.stripWrap = Roo.get(this.createStrip(this.el.dom), true);
28370     this.stripEl = Roo.get(this.createStripList(this.stripWrap.dom), true);
28371     this.stripBody = Roo.get(this.stripWrap.dom.firstChild.firstChild, true);
28372     if(Roo.isIE){
28373         Roo.fly(this.stripWrap.dom.firstChild).setStyle("overflow-x", "hidden");
28374     }
28375     if(this.tabPosition != "bottom"){
28376         /** The body element that contains {@link Roo.TabPanelItem} bodies. +
28377          * @type Roo.Element
28378          */
28379         this.bodyEl = Roo.get(this.createBody(this.el.dom));
28380         this.el.addClass("x-tabs-top");
28381     }
28382     this.items = [];
28383
28384     this.bodyEl.setStyle("position", "relative");
28385
28386     this.active = null;
28387     this.activateDelegate = this.activate.createDelegate(this);
28388
28389     this.addEvents({
28390         /**
28391          * @event tabchange
28392          * Fires when the active tab changes
28393          * @param {Roo.TabPanel} this
28394          * @param {Roo.TabPanelItem} activePanel The new active tab
28395          */
28396         "tabchange": true,
28397         /**
28398          * @event beforetabchange
28399          * Fires before the active tab changes, set cancel to true on the "e" parameter to cancel the change
28400          * @param {Roo.TabPanel} this
28401          * @param {Object} e Set cancel to true on this object to cancel the tab change
28402          * @param {Roo.TabPanelItem} tab The tab being changed to
28403          */
28404         "beforetabchange" : true
28405     });
28406
28407     Roo.EventManager.onWindowResize(this.onResize, this);
28408     this.cpad = this.el.getPadding("lr");
28409     this.hiddenCount = 0;
28410
28411
28412     // toolbar on the tabbar support...
28413     if (this.toolbar) {
28414         var tcfg = this.toolbar;
28415         tcfg.container = this.stripEl.child('td.x-tab-strip-toolbar');  
28416         this.toolbar = new Roo.Toolbar(tcfg);
28417         if (Roo.isSafari) {
28418             var tbl = tcfg.container.child('table', true);
28419             tbl.setAttribute('width', '100%');
28420         }
28421         
28422     }
28423    
28424
28425
28426     Roo.TabPanel.superclass.constructor.call(this);
28427 };
28428
28429 Roo.extend(Roo.TabPanel, Roo.util.Observable, {
28430     /*
28431      *@cfg {String} tabPosition "top" or "bottom" (defaults to "top")
28432      */
28433     tabPosition : "top",
28434     /*
28435      *@cfg {Number} currentTabWidth The width of the current tab (defaults to 0)
28436      */
28437     currentTabWidth : 0,
28438     /*
28439      *@cfg {Number} minTabWidth The minimum width of a tab (defaults to 40) (ignored if {@link #resizeTabs} is not true)
28440      */
28441     minTabWidth : 40,
28442     /*
28443      *@cfg {Number} maxTabWidth The maximum width of a tab (defaults to 250) (ignored if {@link #resizeTabs} is not true)
28444      */
28445     maxTabWidth : 250,
28446     /*
28447      *@cfg {Number} preferredTabWidth The preferred (default) width of a tab (defaults to 175) (ignored if {@link #resizeTabs} is not true)
28448      */
28449     preferredTabWidth : 175,
28450     /*
28451      *@cfg {Boolean} resizeTabs True to enable dynamic tab resizing (defaults to false)
28452      */
28453     resizeTabs : false,
28454     /*
28455      *@cfg {Boolean} monitorResize Set this to true to turn on window resize monitoring (ignored if {@link #resizeTabs} is not true) (defaults to true)
28456      */
28457     monitorResize : true,
28458     /*
28459      *@cfg {Object} toolbar xtype description of toolbar to show at the right of the tab bar. 
28460      */
28461     toolbar : false,
28462
28463     /**
28464      * Creates a new {@link Roo.TabPanelItem} by looking for an existing element with the provided id -- if it's not found it creates one.
28465      * @param {String} id The id of the div to use <b>or create</b>
28466      * @param {String} text The text for the tab
28467      * @param {String} content (optional) Content to put in the TabPanelItem body
28468      * @param {Boolean} closable (optional) True to create a close icon on the tab
28469      * @return {Roo.TabPanelItem} The created TabPanelItem
28470      */
28471     addTab : function(id, text, content, closable){
28472         var item = new Roo.TabPanelItem(this, id, text, closable);
28473         this.addTabItem(item);
28474         if(content){
28475             item.setContent(content);
28476         }
28477         return item;
28478     },
28479
28480     /**
28481      * Returns the {@link Roo.TabPanelItem} with the specified id/index
28482      * @param {String/Number} id The id or index of the TabPanelItem to fetch.
28483      * @return {Roo.TabPanelItem}
28484      */
28485     getTab : function(id){
28486         return this.items[id];
28487     },
28488
28489     /**
28490      * Hides the {@link Roo.TabPanelItem} with the specified id/index
28491      * @param {String/Number} id The id or index of the TabPanelItem to hide.
28492      */
28493     hideTab : function(id){
28494         var t = this.items[id];
28495         if(!t.isHidden()){
28496            t.setHidden(true);
28497            this.hiddenCount++;
28498            this.autoSizeTabs();
28499         }
28500     },
28501
28502     /**
28503      * "Unhides" the {@link Roo.TabPanelItem} with the specified id/index.
28504      * @param {String/Number} id The id or index of the TabPanelItem to unhide.
28505      */
28506     unhideTab : function(id){
28507         var t = this.items[id];
28508         if(t.isHidden()){
28509            t.setHidden(false);
28510            this.hiddenCount--;
28511            this.autoSizeTabs();
28512         }
28513     },
28514
28515     /**
28516      * Adds an existing {@link Roo.TabPanelItem}.
28517      * @param {Roo.TabPanelItem} item The TabPanelItem to add
28518      */
28519     addTabItem : function(item){
28520         this.items[item.id] = item;
28521         this.items.push(item);
28522         if(this.resizeTabs){
28523            item.setWidth(this.currentTabWidth || this.preferredTabWidth);
28524            this.autoSizeTabs();
28525         }else{
28526             item.autoSize();
28527         }
28528     },
28529
28530     /**
28531      * Removes a {@link Roo.TabPanelItem}.
28532      * @param {String/Number} id The id or index of the TabPanelItem to remove.
28533      */
28534     removeTab : function(id){
28535         var items = this.items;
28536         var tab = items[id];
28537         if(!tab) { return; }
28538         var index = items.indexOf(tab);
28539         if(this.active == tab && items.length > 1){
28540             var newTab = this.getNextAvailable(index);
28541             if(newTab) {
28542                 newTab.activate();
28543             }
28544         }
28545         this.stripEl.dom.removeChild(tab.pnode.dom);
28546         if(tab.bodyEl.dom.parentNode == this.bodyEl.dom){ // if it was moved already prevent error
28547             this.bodyEl.dom.removeChild(tab.bodyEl.dom);
28548         }
28549         items.splice(index, 1);
28550         delete this.items[tab.id];
28551         tab.fireEvent("close", tab);
28552         tab.purgeListeners();
28553         this.autoSizeTabs();
28554     },
28555
28556     getNextAvailable : function(start){
28557         var items = this.items;
28558         var index = start;
28559         // look for a next tab that will slide over to
28560         // replace the one being removed
28561         while(index < items.length){
28562             var item = items[++index];
28563             if(item && !item.isHidden()){
28564                 return item;
28565             }
28566         }
28567         // if one isn't found select the previous tab (on the left)
28568         index = start;
28569         while(index >= 0){
28570             var item = items[--index];
28571             if(item && !item.isHidden()){
28572                 return item;
28573             }
28574         }
28575         return null;
28576     },
28577
28578     /**
28579      * Disables a {@link Roo.TabPanelItem}. It cannot be the active tab, if it is this call is ignored.
28580      * @param {String/Number} id The id or index of the TabPanelItem to disable.
28581      */
28582     disableTab : function(id){
28583         var tab = this.items[id];
28584         if(tab && this.active != tab){
28585             tab.disable();
28586         }
28587     },
28588
28589     /**
28590      * Enables a {@link Roo.TabPanelItem} that is disabled.
28591      * @param {String/Number} id The id or index of the TabPanelItem to enable.
28592      */
28593     enableTab : function(id){
28594         var tab = this.items[id];
28595         tab.enable();
28596     },
28597
28598     /**
28599      * Activates a {@link Roo.TabPanelItem}. The currently active one will be deactivated.
28600      * @param {String/Number} id The id or index of the TabPanelItem to activate.
28601      * @return {Roo.TabPanelItem} The TabPanelItem.
28602      */
28603     activate : function(id){
28604         var tab = this.items[id];
28605         if(!tab){
28606             return null;
28607         }
28608         if(tab == this.active || tab.disabled){
28609             return tab;
28610         }
28611         var e = {};
28612         this.fireEvent("beforetabchange", this, e, tab);
28613         if(e.cancel !== true && !tab.disabled){
28614             if(this.active){
28615                 this.active.hide();
28616             }
28617             this.active = this.items[id];
28618             this.active.show();
28619             this.fireEvent("tabchange", this, this.active);
28620         }
28621         return tab;
28622     },
28623
28624     /**
28625      * Gets the active {@link Roo.TabPanelItem}.
28626      * @return {Roo.TabPanelItem} The active TabPanelItem or null if none are active.
28627      */
28628     getActiveTab : function(){
28629         return this.active;
28630     },
28631
28632     /**
28633      * Updates the tab body element to fit the height of the container element
28634      * for overflow scrolling
28635      * @param {Number} targetHeight (optional) Override the starting height from the elements height
28636      */
28637     syncHeight : function(targetHeight){
28638         var height = (targetHeight || this.el.getHeight())-this.el.getBorderWidth("tb")-this.el.getPadding("tb");
28639         var bm = this.bodyEl.getMargins();
28640         var newHeight = height-(this.stripWrap.getHeight()||0)-(bm.top+bm.bottom);
28641         this.bodyEl.setHeight(newHeight);
28642         return newHeight;
28643     },
28644
28645     onResize : function(){
28646         if(this.monitorResize){
28647             this.autoSizeTabs();
28648         }
28649     },
28650
28651     /**
28652      * Disables tab resizing while tabs are being added (if {@link #resizeTabs} is false this does nothing)
28653      */
28654     beginUpdate : function(){
28655         this.updating = true;
28656     },
28657
28658     /**
28659      * Stops an update and resizes the tabs (if {@link #resizeTabs} is false this does nothing)
28660      */
28661     endUpdate : function(){
28662         this.updating = false;
28663         this.autoSizeTabs();
28664     },
28665
28666     /**
28667      * Manual call to resize the tabs (if {@link #resizeTabs} is false this does nothing)
28668      */
28669     autoSizeTabs : function(){
28670         var count = this.items.length;
28671         var vcount = count - this.hiddenCount;
28672         if(!this.resizeTabs || count < 1 || vcount < 1 || this.updating) {
28673             return;
28674         }
28675         var w = Math.max(this.el.getWidth() - this.cpad, 10);
28676         var availWidth = Math.floor(w / vcount);
28677         var b = this.stripBody;
28678         if(b.getWidth() > w){
28679             var tabs = this.items;
28680             this.setTabWidth(Math.max(availWidth, this.minTabWidth)-2);
28681             if(availWidth < this.minTabWidth){
28682                 /*if(!this.sleft){    // incomplete scrolling code
28683                     this.createScrollButtons();
28684                 }
28685                 this.showScroll();
28686                 this.stripClip.setWidth(w - (this.sleft.getWidth()+this.sright.getWidth()));*/
28687             }
28688         }else{
28689             if(this.currentTabWidth < this.preferredTabWidth){
28690                 this.setTabWidth(Math.min(availWidth, this.preferredTabWidth)-2);
28691             }
28692         }
28693     },
28694
28695     /**
28696      * Returns the number of tabs in this TabPanel.
28697      * @return {Number}
28698      */
28699      getCount : function(){
28700          return this.items.length;
28701      },
28702
28703     /**
28704      * Resizes all the tabs to the passed width
28705      * @param {Number} The new width
28706      */
28707     setTabWidth : function(width){
28708         this.currentTabWidth = width;
28709         for(var i = 0, len = this.items.length; i < len; i++) {
28710                 if(!this.items[i].isHidden()) {
28711                 this.items[i].setWidth(width);
28712             }
28713         }
28714     },
28715
28716     /**
28717      * Destroys this TabPanel
28718      * @param {Boolean} removeEl (optional) True to remove the element from the DOM as well (defaults to undefined)
28719      */
28720     destroy : function(removeEl){
28721         Roo.EventManager.removeResizeListener(this.onResize, this);
28722         for(var i = 0, len = this.items.length; i < len; i++){
28723             this.items[i].purgeListeners();
28724         }
28725         if(removeEl === true){
28726             this.el.update("");
28727             this.el.remove();
28728         }
28729     }
28730 });
28731
28732 /**
28733  * @class Roo.TabPanelItem
28734  * @extends Roo.util.Observable
28735  * Represents an individual item (tab plus body) in a TabPanel.
28736  * @param {Roo.TabPanel} tabPanel The {@link Roo.TabPanel} this TabPanelItem belongs to
28737  * @param {String} id The id of this TabPanelItem
28738  * @param {String} text The text for the tab of this TabPanelItem
28739  * @param {Boolean} closable True to allow this TabPanelItem to be closable (defaults to false)
28740  */
28741 Roo.TabPanelItem = function(tabPanel, id, text, closable){
28742     /**
28743      * The {@link Roo.TabPanel} this TabPanelItem belongs to
28744      * @type Roo.TabPanel
28745      */
28746     this.tabPanel = tabPanel;
28747     /**
28748      * The id for this TabPanelItem
28749      * @type String
28750      */
28751     this.id = id;
28752     /** @private */
28753     this.disabled = false;
28754     /** @private */
28755     this.text = text;
28756     /** @private */
28757     this.loaded = false;
28758     this.closable = closable;
28759
28760     /**
28761      * The body element for this TabPanelItem.
28762      * @type Roo.Element
28763      */
28764     this.bodyEl = Roo.get(tabPanel.createItemBody(tabPanel.bodyEl.dom, id));
28765     this.bodyEl.setVisibilityMode(Roo.Element.VISIBILITY);
28766     this.bodyEl.setStyle("display", "block");
28767     this.bodyEl.setStyle("zoom", "1");
28768     this.hideAction();
28769
28770     var els = tabPanel.createStripElements(tabPanel.stripEl.dom, text, closable);
28771     /** @private */
28772     this.el = Roo.get(els.el, true);
28773     this.inner = Roo.get(els.inner, true);
28774     this.textEl = Roo.get(this.el.dom.firstChild.firstChild.firstChild, true);
28775     this.pnode = Roo.get(els.el.parentNode, true);
28776     this.el.on("mousedown", this.onTabMouseDown, this);
28777     this.el.on("click", this.onTabClick, this);
28778     /** @private */
28779     if(closable){
28780         var c = Roo.get(els.close, true);
28781         c.dom.title = this.closeText;
28782         c.addClassOnOver("close-over");
28783         c.on("click", this.closeClick, this);
28784      }
28785
28786     this.addEvents({
28787          /**
28788          * @event activate
28789          * Fires when this tab becomes the active tab.
28790          * @param {Roo.TabPanel} tabPanel The parent TabPanel
28791          * @param {Roo.TabPanelItem} this
28792          */
28793         "activate": true,
28794         /**
28795          * @event beforeclose
28796          * Fires before this tab is closed. To cancel the close, set cancel to true on e (e.cancel = true).
28797          * @param {Roo.TabPanelItem} this
28798          * @param {Object} e Set cancel to true on this object to cancel the close.
28799          */
28800         "beforeclose": true,
28801         /**
28802          * @event close
28803          * Fires when this tab is closed.
28804          * @param {Roo.TabPanelItem} this
28805          */
28806          "close": true,
28807         /**
28808          * @event deactivate
28809          * Fires when this tab is no longer the active tab.
28810          * @param {Roo.TabPanel} tabPanel The parent TabPanel
28811          * @param {Roo.TabPanelItem} this
28812          */
28813          "deactivate" : true
28814     });
28815     this.hidden = false;
28816
28817     Roo.TabPanelItem.superclass.constructor.call(this);
28818 };
28819
28820 Roo.extend(Roo.TabPanelItem, Roo.util.Observable, {
28821     purgeListeners : function(){
28822        Roo.util.Observable.prototype.purgeListeners.call(this);
28823        this.el.removeAllListeners();
28824     },
28825     /**
28826      * Shows this TabPanelItem -- this <b>does not</b> deactivate the currently active TabPanelItem.
28827      */
28828     show : function(){
28829         this.pnode.addClass("on");
28830         this.showAction();
28831         if(Roo.isOpera){
28832             this.tabPanel.stripWrap.repaint();
28833         }
28834         this.fireEvent("activate", this.tabPanel, this);
28835     },
28836
28837     /**
28838      * Returns true if this tab is the active tab.
28839      * @return {Boolean}
28840      */
28841     isActive : function(){
28842         return this.tabPanel.getActiveTab() == this;
28843     },
28844
28845     /**
28846      * Hides this TabPanelItem -- if you don't activate another TabPanelItem this could look odd.
28847      */
28848     hide : function(){
28849         this.pnode.removeClass("on");
28850         this.hideAction();
28851         this.fireEvent("deactivate", this.tabPanel, this);
28852     },
28853
28854     hideAction : function(){
28855         this.bodyEl.hide();
28856         this.bodyEl.setStyle("position", "absolute");
28857         this.bodyEl.setLeft("-20000px");
28858         this.bodyEl.setTop("-20000px");
28859     },
28860
28861     showAction : function(){
28862         this.bodyEl.setStyle("position", "relative");
28863         this.bodyEl.setTop("");
28864         this.bodyEl.setLeft("");
28865         this.bodyEl.show();
28866     },
28867
28868     /**
28869      * Set the tooltip for the tab.
28870      * @param {String} tooltip The tab's tooltip
28871      */
28872     setTooltip : function(text){
28873         if(Roo.QuickTips && Roo.QuickTips.isEnabled()){
28874             this.textEl.dom.qtip = text;
28875             this.textEl.dom.removeAttribute('title');
28876         }else{
28877             this.textEl.dom.title = text;
28878         }
28879     },
28880
28881     onTabClick : function(e){
28882         e.preventDefault();
28883         this.tabPanel.activate(this.id);
28884     },
28885
28886     onTabMouseDown : function(e){
28887         e.preventDefault();
28888         this.tabPanel.activate(this.id);
28889     },
28890
28891     getWidth : function(){
28892         return this.inner.getWidth();
28893     },
28894
28895     setWidth : function(width){
28896         var iwidth = width - this.pnode.getPadding("lr");
28897         this.inner.setWidth(iwidth);
28898         this.textEl.setWidth(iwidth-this.inner.getPadding("lr"));
28899         this.pnode.setWidth(width);
28900     },
28901
28902     /**
28903      * Show or hide the tab
28904      * @param {Boolean} hidden True to hide or false to show.
28905      */
28906     setHidden : function(hidden){
28907         this.hidden = hidden;
28908         this.pnode.setStyle("display", hidden ? "none" : "");
28909     },
28910
28911     /**
28912      * Returns true if this tab is "hidden"
28913      * @return {Boolean}
28914      */
28915     isHidden : function(){
28916         return this.hidden;
28917     },
28918
28919     /**
28920      * Returns the text for this tab
28921      * @return {String}
28922      */
28923     getText : function(){
28924         return this.text;
28925     },
28926
28927     autoSize : function(){
28928         //this.el.beginMeasure();
28929         this.textEl.setWidth(1);
28930         /*
28931          *  #2804 [new] Tabs in Roojs
28932          *  increase the width by 2-4 pixels to prevent the ellipssis showing in chrome
28933          */
28934         this.setWidth(this.textEl.dom.scrollWidth+this.pnode.getPadding("lr")+this.inner.getPadding("lr") + 2);
28935         //this.el.endMeasure();
28936     },
28937
28938     /**
28939      * Sets the text for the tab (Note: this also sets the tooltip text)
28940      * @param {String} text The tab's text and tooltip
28941      */
28942     setText : function(text){
28943         this.text = text;
28944         this.textEl.update(text);
28945         this.setTooltip(text);
28946         if(!this.tabPanel.resizeTabs){
28947             this.autoSize();
28948         }
28949     },
28950     /**
28951      * Activates this TabPanelItem -- this <b>does</b> deactivate the currently active TabPanelItem.
28952      */
28953     activate : function(){
28954         this.tabPanel.activate(this.id);
28955     },
28956
28957     /**
28958      * Disables this TabPanelItem -- this does nothing if this is the active TabPanelItem.
28959      */
28960     disable : function(){
28961         if(this.tabPanel.active != this){
28962             this.disabled = true;
28963             this.pnode.addClass("disabled");
28964         }
28965     },
28966
28967     /**
28968      * Enables this TabPanelItem if it was previously disabled.
28969      */
28970     enable : function(){
28971         this.disabled = false;
28972         this.pnode.removeClass("disabled");
28973     },
28974
28975     /**
28976      * Sets the content for this TabPanelItem.
28977      * @param {String} content The content
28978      * @param {Boolean} loadScripts true to look for and load scripts
28979      */
28980     setContent : function(content, loadScripts){
28981         this.bodyEl.update(content, loadScripts);
28982     },
28983
28984     /**
28985      * Gets the {@link Roo.UpdateManager} for the body of this TabPanelItem. Enables you to perform Ajax updates.
28986      * @return {Roo.UpdateManager} The UpdateManager
28987      */
28988     getUpdateManager : function(){
28989         return this.bodyEl.getUpdateManager();
28990     },
28991
28992     /**
28993      * Set a URL to be used to load the content for this TabPanelItem.
28994      * @param {String/Function} url The URL to load the content from, or a function to call to get the URL
28995      * @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)
28996      * @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)
28997      * @return {Roo.UpdateManager} The UpdateManager
28998      */
28999     setUrl : function(url, params, loadOnce){
29000         if(this.refreshDelegate){
29001             this.un('activate', this.refreshDelegate);
29002         }
29003         this.refreshDelegate = this._handleRefresh.createDelegate(this, [url, params, loadOnce]);
29004         this.on("activate", this.refreshDelegate);
29005         return this.bodyEl.getUpdateManager();
29006     },
29007
29008     /** @private */
29009     _handleRefresh : function(url, params, loadOnce){
29010         if(!loadOnce || !this.loaded){
29011             var updater = this.bodyEl.getUpdateManager();
29012             updater.update(url, params, this._setLoaded.createDelegate(this));
29013         }
29014     },
29015
29016     /**
29017      *   Forces a content refresh from the URL specified in the {@link #setUrl} method.
29018      *   Will fail silently if the setUrl method has not been called.
29019      *   This does not activate the panel, just updates its content.
29020      */
29021     refresh : function(){
29022         if(this.refreshDelegate){
29023            this.loaded = false;
29024            this.refreshDelegate();
29025         }
29026     },
29027
29028     /** @private */
29029     _setLoaded : function(){
29030         this.loaded = true;
29031     },
29032
29033     /** @private */
29034     closeClick : function(e){
29035         var o = {};
29036         e.stopEvent();
29037         this.fireEvent("beforeclose", this, o);
29038         if(o.cancel !== true){
29039             this.tabPanel.removeTab(this.id);
29040         }
29041     },
29042     /**
29043      * The text displayed in the tooltip for the close icon.
29044      * @type String
29045      */
29046     closeText : "Close this tab"
29047 });
29048
29049 /** @private */
29050 Roo.TabPanel.prototype.createStrip = function(container){
29051     var strip = document.createElement("div");
29052     strip.className = "x-tabs-wrap";
29053     container.appendChild(strip);
29054     return strip;
29055 };
29056 /** @private */
29057 Roo.TabPanel.prototype.createStripList = function(strip){
29058     // div wrapper for retard IE
29059     // returns the "tr" element.
29060     strip.innerHTML = '<div class="x-tabs-strip-wrap">'+
29061         '<table class="x-tabs-strip" cellspacing="0" cellpadding="0" border="0"><tbody><tr>'+
29062         '<td class="x-tab-strip-toolbar"></td></tr></tbody></table></div>';
29063     return strip.firstChild.firstChild.firstChild.firstChild;
29064 };
29065 /** @private */
29066 Roo.TabPanel.prototype.createBody = function(container){
29067     var body = document.createElement("div");
29068     Roo.id(body, "tab-body");
29069     Roo.fly(body).addClass("x-tabs-body");
29070     container.appendChild(body);
29071     return body;
29072 };
29073 /** @private */
29074 Roo.TabPanel.prototype.createItemBody = function(bodyEl, id){
29075     var body = Roo.getDom(id);
29076     if(!body){
29077         body = document.createElement("div");
29078         body.id = id;
29079     }
29080     Roo.fly(body).addClass("x-tabs-item-body");
29081     bodyEl.insertBefore(body, bodyEl.firstChild);
29082     return body;
29083 };
29084 /** @private */
29085 Roo.TabPanel.prototype.createStripElements = function(stripEl, text, closable){
29086     var td = document.createElement("td");
29087     stripEl.insertBefore(td, stripEl.childNodes[stripEl.childNodes.length-1]);
29088     //stripEl.appendChild(td);
29089     if(closable){
29090         td.className = "x-tabs-closable";
29091         if(!this.closeTpl){
29092             this.closeTpl = new Roo.Template(
29093                '<a href="#" class="x-tabs-right"><span class="x-tabs-left"><em class="x-tabs-inner">' +
29094                '<span unselectable="on"' + (this.disableTooltips ? '' : ' title="{text}"') +' class="x-tabs-text">{text}</span>' +
29095                '<div unselectable="on" class="close-icon">&#160;</div></em></span></a>'
29096             );
29097         }
29098         var el = this.closeTpl.overwrite(td, {"text": text});
29099         var close = el.getElementsByTagName("div")[0];
29100         var inner = el.getElementsByTagName("em")[0];
29101         return {"el": el, "close": close, "inner": inner};
29102     } else {
29103         if(!this.tabTpl){
29104             this.tabTpl = new Roo.Template(
29105                '<a href="#" class="x-tabs-right"><span class="x-tabs-left"><em class="x-tabs-inner">' +
29106                '<span unselectable="on"' + (this.disableTooltips ? '' : ' title="{text}"') +' class="x-tabs-text">{text}</span></em></span></a>'
29107             );
29108         }
29109         var el = this.tabTpl.overwrite(td, {"text": text});
29110         var inner = el.getElementsByTagName("em")[0];
29111         return {"el": el, "inner": inner};
29112     }
29113 };/*
29114  * Based on:
29115  * Ext JS Library 1.1.1
29116  * Copyright(c) 2006-2007, Ext JS, LLC.
29117  *
29118  * Originally Released Under LGPL - original licence link has changed is not relivant.
29119  *
29120  * Fork - LGPL
29121  * <script type="text/javascript">
29122  */
29123
29124 /**
29125  * @class Roo.Button
29126  * @extends Roo.util.Observable
29127  * Simple Button class
29128  * @cfg {String} text The button text
29129  * @cfg {String} icon The path to an image to display in the button (the image will be set as the background-image
29130  * CSS property of the button by default, so if you want a mixed icon/text button, set cls:"x-btn-text-icon")
29131  * @cfg {Function} handler A function called when the button is clicked (can be used instead of click event)
29132  * @cfg {Object} scope The scope of the handler
29133  * @cfg {Number} minWidth The minimum width for this button (used to give a set of buttons a common width)
29134  * @cfg {String/Object} tooltip The tooltip for the button - can be a string or QuickTips config object
29135  * @cfg {Boolean} hidden True to start hidden (defaults to false)
29136  * @cfg {Boolean} disabled True to start disabled (defaults to false)
29137  * @cfg {Boolean} pressed True to start pressed (only if enableToggle = true)
29138  * @cfg {String} toggleGroup The group this toggle button is a member of (only 1 per group can be pressed, only
29139    applies if enableToggle = true)
29140  * @cfg {String/HTMLElement/Element} renderTo The element to append the button to
29141  * @cfg {Boolean/Object} repeat True to repeat fire the click event while the mouse is down. This can also be
29142   an {@link Roo.util.ClickRepeater} config object (defaults to false).
29143  * @constructor
29144  * Create a new button
29145  * @param {Object} config The config object
29146  */
29147 Roo.Button = function(renderTo, config)
29148 {
29149     if (!config) {
29150         config = renderTo;
29151         renderTo = config.renderTo || false;
29152     }
29153     
29154     Roo.apply(this, config);
29155     this.addEvents({
29156         /**
29157              * @event click
29158              * Fires when this button is clicked
29159              * @param {Button} this
29160              * @param {EventObject} e The click event
29161              */
29162             "click" : true,
29163         /**
29164              * @event toggle
29165              * Fires when the "pressed" state of this button changes (only if enableToggle = true)
29166              * @param {Button} this
29167              * @param {Boolean} pressed
29168              */
29169             "toggle" : true,
29170         /**
29171              * @event mouseover
29172              * Fires when the mouse hovers over the button
29173              * @param {Button} this
29174              * @param {Event} e The event object
29175              */
29176         'mouseover' : true,
29177         /**
29178              * @event mouseout
29179              * Fires when the mouse exits the button
29180              * @param {Button} this
29181              * @param {Event} e The event object
29182              */
29183         'mouseout': true,
29184          /**
29185              * @event render
29186              * Fires when the button is rendered
29187              * @param {Button} this
29188              */
29189         'render': true
29190     });
29191     if(this.menu){
29192         this.menu = Roo.menu.MenuMgr.get(this.menu);
29193     }
29194     // register listeners first!!  - so render can be captured..
29195     Roo.util.Observable.call(this);
29196     if(renderTo){
29197         this.render(renderTo);
29198     }
29199     
29200   
29201 };
29202
29203 Roo.extend(Roo.Button, Roo.util.Observable, {
29204     /**
29205      * 
29206      */
29207     
29208     /**
29209      * Read-only. True if this button is hidden
29210      * @type Boolean
29211      */
29212     hidden : false,
29213     /**
29214      * Read-only. True if this button is disabled
29215      * @type Boolean
29216      */
29217     disabled : false,
29218     /**
29219      * Read-only. True if this button is pressed (only if enableToggle = true)
29220      * @type Boolean
29221      */
29222     pressed : false,
29223
29224     /**
29225      * @cfg {Number} tabIndex 
29226      * The DOM tabIndex for this button (defaults to undefined)
29227      */
29228     tabIndex : undefined,
29229
29230     /**
29231      * @cfg {Boolean} enableToggle
29232      * True to enable pressed/not pressed toggling (defaults to false)
29233      */
29234     enableToggle: false,
29235     /**
29236      * @cfg {Mixed} menu
29237      * Standard menu attribute consisting of a reference to a menu object, a menu id or a menu config blob (defaults to undefined).
29238      */
29239     menu : undefined,
29240     /**
29241      * @cfg {String} menuAlign
29242      * The position to align the menu to (see {@link Roo.Element#alignTo} for more details, defaults to 'tl-bl?').
29243      */
29244     menuAlign : "tl-bl?",
29245
29246     /**
29247      * @cfg {String} iconCls
29248      * A css class which sets a background image to be used as the icon for this button (defaults to undefined).
29249      */
29250     iconCls : undefined,
29251     /**
29252      * @cfg {String} type
29253      * The button's type, corresponding to the DOM input element type attribute.  Either "submit," "reset" or "button" (default).
29254      */
29255     type : 'button',
29256
29257     // private
29258     menuClassTarget: 'tr',
29259
29260     /**
29261      * @cfg {String} clickEvent
29262      * The type of event to map to the button's event handler (defaults to 'click')
29263      */
29264     clickEvent : 'click',
29265
29266     /**
29267      * @cfg {Boolean} handleMouseEvents
29268      * False to disable visual cues on mouseover, mouseout and mousedown (defaults to true)
29269      */
29270     handleMouseEvents : true,
29271
29272     /**
29273      * @cfg {String} tooltipType
29274      * The type of tooltip to use. Either "qtip" (default) for QuickTips or "title" for title attribute.
29275      */
29276     tooltipType : 'qtip',
29277
29278     /**
29279      * @cfg {String} cls
29280      * A CSS class to apply to the button's main element.
29281      */
29282     
29283     /**
29284      * @cfg {Roo.Template} template (Optional)
29285      * An {@link Roo.Template} with which to create the Button's main element. This Template must
29286      * contain numeric substitution parameter 0 if it is to display the tRoo property. Changing the template could
29287      * require code modifications if required elements (e.g. a button) aren't present.
29288      */
29289
29290     // private
29291     render : function(renderTo){
29292         var btn;
29293         if(this.hideParent){
29294             this.parentEl = Roo.get(renderTo);
29295         }
29296         if(!this.dhconfig){
29297             if(!this.template){
29298                 if(!Roo.Button.buttonTemplate){
29299                     // hideous table template
29300                     Roo.Button.buttonTemplate = new Roo.Template(
29301                         '<table border="0" cellpadding="0" cellspacing="0" class="x-btn-wrap"><tbody><tr>',
29302                         '<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>',
29303                         "</tr></tbody></table>");
29304                 }
29305                 this.template = Roo.Button.buttonTemplate;
29306             }
29307             btn = this.template.append(renderTo, [this.text || '&#160;', this.type], true);
29308             var btnEl = btn.child("button:first");
29309             btnEl.on('focus', this.onFocus, this);
29310             btnEl.on('blur', this.onBlur, this);
29311             if(this.cls){
29312                 btn.addClass(this.cls);
29313             }
29314             if(this.icon){
29315                 btnEl.setStyle('background-image', 'url(' +this.icon +')');
29316             }
29317             if(this.iconCls){
29318                 btnEl.addClass(this.iconCls);
29319                 if(!this.cls){
29320                     btn.addClass(this.text ? 'x-btn-text-icon' : 'x-btn-icon');
29321                 }
29322             }
29323             if(this.tabIndex !== undefined){
29324                 btnEl.dom.tabIndex = this.tabIndex;
29325             }
29326             if(this.tooltip){
29327                 if(typeof this.tooltip == 'object'){
29328                     Roo.QuickTips.tips(Roo.apply({
29329                           target: btnEl.id
29330                     }, this.tooltip));
29331                 } else {
29332                     btnEl.dom[this.tooltipType] = this.tooltip;
29333                 }
29334             }
29335         }else{
29336             btn = Roo.DomHelper.append(Roo.get(renderTo).dom, this.dhconfig, true);
29337         }
29338         this.el = btn;
29339         if(this.id){
29340             this.el.dom.id = this.el.id = this.id;
29341         }
29342         if(this.menu){
29343             this.el.child(this.menuClassTarget).addClass("x-btn-with-menu");
29344             this.menu.on("show", this.onMenuShow, this);
29345             this.menu.on("hide", this.onMenuHide, this);
29346         }
29347         btn.addClass("x-btn");
29348         if(Roo.isIE && !Roo.isIE7){
29349             this.autoWidth.defer(1, this);
29350         }else{
29351             this.autoWidth();
29352         }
29353         if(this.handleMouseEvents){
29354             btn.on("mouseover", this.onMouseOver, this);
29355             btn.on("mouseout", this.onMouseOut, this);
29356             btn.on("mousedown", this.onMouseDown, this);
29357         }
29358         btn.on(this.clickEvent, this.onClick, this);
29359         //btn.on("mouseup", this.onMouseUp, this);
29360         if(this.hidden){
29361             this.hide();
29362         }
29363         if(this.disabled){
29364             this.disable();
29365         }
29366         Roo.ButtonToggleMgr.register(this);
29367         if(this.pressed){
29368             this.el.addClass("x-btn-pressed");
29369         }
29370         if(this.repeat){
29371             var repeater = new Roo.util.ClickRepeater(btn,
29372                 typeof this.repeat == "object" ? this.repeat : {}
29373             );
29374             repeater.on("click", this.onClick,  this);
29375         }
29376         
29377         this.fireEvent('render', this);
29378         
29379     },
29380     /**
29381      * Returns the button's underlying element
29382      * @return {Roo.Element} The element
29383      */
29384     getEl : function(){
29385         return this.el;  
29386     },
29387     
29388     /**
29389      * Destroys this Button and removes any listeners.
29390      */
29391     destroy : function(){
29392         Roo.ButtonToggleMgr.unregister(this);
29393         this.el.removeAllListeners();
29394         this.purgeListeners();
29395         this.el.remove();
29396     },
29397
29398     // private
29399     autoWidth : function(){
29400         if(this.el){
29401             this.el.setWidth("auto");
29402             if(Roo.isIE7 && Roo.isStrict){
29403                 var ib = this.el.child('button');
29404                 if(ib && ib.getWidth() > 20){
29405                     ib.clip();
29406                     ib.setWidth(Roo.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));
29407                 }
29408             }
29409             if(this.minWidth){
29410                 if(this.hidden){
29411                     this.el.beginMeasure();
29412                 }
29413                 if(this.el.getWidth() < this.minWidth){
29414                     this.el.setWidth(this.minWidth);
29415                 }
29416                 if(this.hidden){
29417                     this.el.endMeasure();
29418                 }
29419             }
29420         }
29421     },
29422
29423     /**
29424      * Assigns this button's click handler
29425      * @param {Function} handler The function to call when the button is clicked
29426      * @param {Object} scope (optional) Scope for the function passed in
29427      */
29428     setHandler : function(handler, scope){
29429         this.handler = handler;
29430         this.scope = scope;  
29431     },
29432     
29433     /**
29434      * Sets this button's text
29435      * @param {String} text The button text
29436      */
29437     setText : function(text){
29438         this.text = text;
29439         if(this.el){
29440             this.el.child("td.x-btn-center button.x-btn-text").update(text);
29441         }
29442         this.autoWidth();
29443     },
29444     
29445     /**
29446      * Gets the text for this button
29447      * @return {String} The button text
29448      */
29449     getText : function(){
29450         return this.text;  
29451     },
29452     
29453     /**
29454      * Show this button
29455      */
29456     show: function(){
29457         this.hidden = false;
29458         if(this.el){
29459             this[this.hideParent? 'parentEl' : 'el'].setStyle("display", "");
29460         }
29461     },
29462     
29463     /**
29464      * Hide this button
29465      */
29466     hide: function(){
29467         this.hidden = true;
29468         if(this.el){
29469             this[this.hideParent? 'parentEl' : 'el'].setStyle("display", "none");
29470         }
29471     },
29472     
29473     /**
29474      * Convenience function for boolean show/hide
29475      * @param {Boolean} visible True to show, false to hide
29476      */
29477     setVisible: function(visible){
29478         if(visible) {
29479             this.show();
29480         }else{
29481             this.hide();
29482         }
29483     },
29484     
29485     /**
29486      * If a state it passed, it becomes the pressed state otherwise the current state is toggled.
29487      * @param {Boolean} state (optional) Force a particular state
29488      */
29489     toggle : function(state){
29490         state = state === undefined ? !this.pressed : state;
29491         if(state != this.pressed){
29492             if(state){
29493                 this.el.addClass("x-btn-pressed");
29494                 this.pressed = true;
29495                 this.fireEvent("toggle", this, true);
29496             }else{
29497                 this.el.removeClass("x-btn-pressed");
29498                 this.pressed = false;
29499                 this.fireEvent("toggle", this, false);
29500             }
29501             if(this.toggleHandler){
29502                 this.toggleHandler.call(this.scope || this, this, state);
29503             }
29504         }
29505     },
29506     
29507     /**
29508      * Focus the button
29509      */
29510     focus : function(){
29511         this.el.child('button:first').focus();
29512     },
29513     
29514     /**
29515      * Disable this button
29516      */
29517     disable : function(){
29518         if(this.el){
29519             this.el.addClass("x-btn-disabled");
29520         }
29521         this.disabled = true;
29522     },
29523     
29524     /**
29525      * Enable this button
29526      */
29527     enable : function(){
29528         if(this.el){
29529             this.el.removeClass("x-btn-disabled");
29530         }
29531         this.disabled = false;
29532     },
29533
29534     /**
29535      * Convenience function for boolean enable/disable
29536      * @param {Boolean} enabled True to enable, false to disable
29537      */
29538     setDisabled : function(v){
29539         this[v !== true ? "enable" : "disable"]();
29540     },
29541
29542     // private
29543     onClick : function(e)
29544     {
29545         if(e){
29546             e.preventDefault();
29547         }
29548         if(e.button != 0){
29549             return;
29550         }
29551         if(!this.disabled){
29552             if(this.enableToggle){
29553                 this.toggle();
29554             }
29555             if(this.menu && !this.menu.isVisible()){
29556                 this.menu.show(this.el, this.menuAlign);
29557             }
29558             this.fireEvent("click", this, e);
29559             if(this.handler){
29560                 this.el.removeClass("x-btn-over");
29561                 this.handler.call(this.scope || this, this, e);
29562             }
29563         }
29564     },
29565     // private
29566     onMouseOver : function(e){
29567         if(!this.disabled){
29568             this.el.addClass("x-btn-over");
29569             this.fireEvent('mouseover', this, e);
29570         }
29571     },
29572     // private
29573     onMouseOut : function(e){
29574         if(!e.within(this.el,  true)){
29575             this.el.removeClass("x-btn-over");
29576             this.fireEvent('mouseout', this, e);
29577         }
29578     },
29579     // private
29580     onFocus : function(e){
29581         if(!this.disabled){
29582             this.el.addClass("x-btn-focus");
29583         }
29584     },
29585     // private
29586     onBlur : function(e){
29587         this.el.removeClass("x-btn-focus");
29588     },
29589     // private
29590     onMouseDown : function(e){
29591         if(!this.disabled && e.button == 0){
29592             this.el.addClass("x-btn-click");
29593             Roo.get(document).on('mouseup', this.onMouseUp, this);
29594         }
29595     },
29596     // private
29597     onMouseUp : function(e){
29598         if(e.button == 0){
29599             this.el.removeClass("x-btn-click");
29600             Roo.get(document).un('mouseup', this.onMouseUp, this);
29601         }
29602     },
29603     // private
29604     onMenuShow : function(e){
29605         this.el.addClass("x-btn-menu-active");
29606     },
29607     // private
29608     onMenuHide : function(e){
29609         this.el.removeClass("x-btn-menu-active");
29610     }   
29611 });
29612
29613 // Private utility class used by Button
29614 Roo.ButtonToggleMgr = function(){
29615    var groups = {};
29616    
29617    function toggleGroup(btn, state){
29618        if(state){
29619            var g = groups[btn.toggleGroup];
29620            for(var i = 0, l = g.length; i < l; i++){
29621                if(g[i] != btn){
29622                    g[i].toggle(false);
29623                }
29624            }
29625        }
29626    }
29627    
29628    return {
29629        register : function(btn){
29630            if(!btn.toggleGroup){
29631                return;
29632            }
29633            var g = groups[btn.toggleGroup];
29634            if(!g){
29635                g = groups[btn.toggleGroup] = [];
29636            }
29637            g.push(btn);
29638            btn.on("toggle", toggleGroup);
29639        },
29640        
29641        unregister : function(btn){
29642            if(!btn.toggleGroup){
29643                return;
29644            }
29645            var g = groups[btn.toggleGroup];
29646            if(g){
29647                g.remove(btn);
29648                btn.un("toggle", toggleGroup);
29649            }
29650        }
29651    };
29652 }();/*
29653  * Based on:
29654  * Ext JS Library 1.1.1
29655  * Copyright(c) 2006-2007, Ext JS, LLC.
29656  *
29657  * Originally Released Under LGPL - original licence link has changed is not relivant.
29658  *
29659  * Fork - LGPL
29660  * <script type="text/javascript">
29661  */
29662  
29663 /**
29664  * @class Roo.SplitButton
29665  * @extends Roo.Button
29666  * A split button that provides a built-in dropdown arrow that can fire an event separately from the default
29667  * click event of the button.  Typically this would be used to display a dropdown menu that provides additional
29668  * options to the primary button action, but any custom handler can provide the arrowclick implementation.
29669  * @cfg {Function} arrowHandler A function called when the arrow button is clicked (can be used instead of click event)
29670  * @cfg {String} arrowTooltip The title attribute of the arrow
29671  * @constructor
29672  * Create a new menu button
29673  * @param {String/HTMLElement/Element} renderTo The element to append the button to
29674  * @param {Object} config The config object
29675  */
29676 Roo.SplitButton = function(renderTo, config){
29677     Roo.SplitButton.superclass.constructor.call(this, renderTo, config);
29678     /**
29679      * @event arrowclick
29680      * Fires when this button's arrow is clicked
29681      * @param {SplitButton} this
29682      * @param {EventObject} e The click event
29683      */
29684     this.addEvents({"arrowclick":true});
29685 };
29686
29687 Roo.extend(Roo.SplitButton, Roo.Button, {
29688     render : function(renderTo){
29689         // this is one sweet looking template!
29690         var tpl = new Roo.Template(
29691             '<table cellspacing="0" class="x-btn-menu-wrap x-btn"><tr><td>',
29692             '<table cellspacing="0" class="x-btn-wrap x-btn-menu-text-wrap"><tbody>',
29693             '<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>',
29694             "</tbody></table></td><td>",
29695             '<table cellspacing="0" class="x-btn-wrap x-btn-menu-arrow-wrap"><tbody>',
29696             '<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>',
29697             "</tbody></table></td></tr></table>"
29698         );
29699         var btn = tpl.append(renderTo, [this.text, this.type], true);
29700         var btnEl = btn.child("button");
29701         if(this.cls){
29702             btn.addClass(this.cls);
29703         }
29704         if(this.icon){
29705             btnEl.setStyle('background-image', 'url(' +this.icon +')');
29706         }
29707         if(this.iconCls){
29708             btnEl.addClass(this.iconCls);
29709             if(!this.cls){
29710                 btn.addClass(this.text ? 'x-btn-text-icon' : 'x-btn-icon');
29711             }
29712         }
29713         this.el = btn;
29714         if(this.handleMouseEvents){
29715             btn.on("mouseover", this.onMouseOver, this);
29716             btn.on("mouseout", this.onMouseOut, this);
29717             btn.on("mousedown", this.onMouseDown, this);
29718             btn.on("mouseup", this.onMouseUp, this);
29719         }
29720         btn.on(this.clickEvent, this.onClick, this);
29721         if(this.tooltip){
29722             if(typeof this.tooltip == 'object'){
29723                 Roo.QuickTips.tips(Roo.apply({
29724                       target: btnEl.id
29725                 }, this.tooltip));
29726             } else {
29727                 btnEl.dom[this.tooltipType] = this.tooltip;
29728             }
29729         }
29730         if(this.arrowTooltip){
29731             btn.child("button:nth(2)").dom[this.tooltipType] = this.arrowTooltip;
29732         }
29733         if(this.hidden){
29734             this.hide();
29735         }
29736         if(this.disabled){
29737             this.disable();
29738         }
29739         if(this.pressed){
29740             this.el.addClass("x-btn-pressed");
29741         }
29742         if(Roo.isIE && !Roo.isIE7){
29743             this.autoWidth.defer(1, this);
29744         }else{
29745             this.autoWidth();
29746         }
29747         if(this.menu){
29748             this.menu.on("show", this.onMenuShow, this);
29749             this.menu.on("hide", this.onMenuHide, this);
29750         }
29751         this.fireEvent('render', this);
29752     },
29753
29754     // private
29755     autoWidth : function(){
29756         if(this.el){
29757             var tbl = this.el.child("table:first");
29758             var tbl2 = this.el.child("table:last");
29759             this.el.setWidth("auto");
29760             tbl.setWidth("auto");
29761             if(Roo.isIE7 && Roo.isStrict){
29762                 var ib = this.el.child('button:first');
29763                 if(ib && ib.getWidth() > 20){
29764                     ib.clip();
29765                     ib.setWidth(Roo.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));
29766                 }
29767             }
29768             if(this.minWidth){
29769                 if(this.hidden){
29770                     this.el.beginMeasure();
29771                 }
29772                 if((tbl.getWidth()+tbl2.getWidth()) < this.minWidth){
29773                     tbl.setWidth(this.minWidth-tbl2.getWidth());
29774                 }
29775                 if(this.hidden){
29776                     this.el.endMeasure();
29777                 }
29778             }
29779             this.el.setWidth(tbl.getWidth()+tbl2.getWidth());
29780         } 
29781     },
29782     /**
29783      * Sets this button's click handler
29784      * @param {Function} handler The function to call when the button is clicked
29785      * @param {Object} scope (optional) Scope for the function passed above
29786      */
29787     setHandler : function(handler, scope){
29788         this.handler = handler;
29789         this.scope = scope;  
29790     },
29791     
29792     /**
29793      * Sets this button's arrow click handler
29794      * @param {Function} handler The function to call when the arrow is clicked
29795      * @param {Object} scope (optional) Scope for the function passed above
29796      */
29797     setArrowHandler : function(handler, scope){
29798         this.arrowHandler = handler;
29799         this.scope = scope;  
29800     },
29801     
29802     /**
29803      * Focus the button
29804      */
29805     focus : function(){
29806         if(this.el){
29807             this.el.child("button:first").focus();
29808         }
29809     },
29810
29811     // private
29812     onClick : function(e){
29813         e.preventDefault();
29814         if(!this.disabled){
29815             if(e.getTarget(".x-btn-menu-arrow-wrap")){
29816                 if(this.menu && !this.menu.isVisible()){
29817                     this.menu.show(this.el, this.menuAlign);
29818                 }
29819                 this.fireEvent("arrowclick", this, e);
29820                 if(this.arrowHandler){
29821                     this.arrowHandler.call(this.scope || this, this, e);
29822                 }
29823             }else{
29824                 this.fireEvent("click", this, e);
29825                 if(this.handler){
29826                     this.handler.call(this.scope || this, this, e);
29827                 }
29828             }
29829         }
29830     },
29831     // private
29832     onMouseDown : function(e){
29833         if(!this.disabled){
29834             Roo.fly(e.getTarget("table")).addClass("x-btn-click");
29835         }
29836     },
29837     // private
29838     onMouseUp : function(e){
29839         Roo.fly(e.getTarget("table")).removeClass("x-btn-click");
29840     }   
29841 });
29842
29843
29844 // backwards compat
29845 Roo.MenuButton = Roo.SplitButton;/*
29846  * Based on:
29847  * Ext JS Library 1.1.1
29848  * Copyright(c) 2006-2007, Ext JS, LLC.
29849  *
29850  * Originally Released Under LGPL - original licence link has changed is not relivant.
29851  *
29852  * Fork - LGPL
29853  * <script type="text/javascript">
29854  */
29855
29856 /**
29857  * @class Roo.Toolbar
29858  * Basic Toolbar class.
29859  * @constructor
29860  * Creates a new Toolbar
29861  * @param {Object} container The config object
29862  */ 
29863 Roo.Toolbar = function(container, buttons, config)
29864 {
29865     /// old consturctor format still supported..
29866     if(container instanceof Array){ // omit the container for later rendering
29867         buttons = container;
29868         config = buttons;
29869         container = null;
29870     }
29871     if (typeof(container) == 'object' && container.xtype) {
29872         config = container;
29873         container = config.container;
29874         buttons = config.buttons || []; // not really - use items!!
29875     }
29876     var xitems = [];
29877     if (config && config.items) {
29878         xitems = config.items;
29879         delete config.items;
29880     }
29881     Roo.apply(this, config);
29882     this.buttons = buttons;
29883     
29884     if(container){
29885         this.render(container);
29886     }
29887     this.xitems = xitems;
29888     Roo.each(xitems, function(b) {
29889         this.add(b);
29890     }, this);
29891     
29892 };
29893
29894 Roo.Toolbar.prototype = {
29895     /**
29896      * @cfg {Array} items
29897      * array of button configs or elements to add (will be converted to a MixedCollection)
29898      */
29899     
29900     /**
29901      * @cfg {String/HTMLElement/Element} container
29902      * The id or element that will contain the toolbar
29903      */
29904     // private
29905     render : function(ct){
29906         this.el = Roo.get(ct);
29907         if(this.cls){
29908             this.el.addClass(this.cls);
29909         }
29910         // using a table allows for vertical alignment
29911         // 100% width is needed by Safari...
29912         this.el.update('<div class="x-toolbar x-small-editor"><table cellspacing="0"><tr></tr></table></div>');
29913         this.tr = this.el.child("tr", true);
29914         var autoId = 0;
29915         this.items = new Roo.util.MixedCollection(false, function(o){
29916             return o.id || ("item" + (++autoId));
29917         });
29918         if(this.buttons){
29919             this.add.apply(this, this.buttons);
29920             delete this.buttons;
29921         }
29922     },
29923
29924     /**
29925      * Adds element(s) to the toolbar -- this function takes a variable number of 
29926      * arguments of mixed type and adds them to the toolbar.
29927      * @param {Mixed} arg1 The following types of arguments are all valid:<br />
29928      * <ul>
29929      * <li>{@link Roo.Toolbar.Button} config: A valid button config object (equivalent to {@link #addButton})</li>
29930      * <li>HtmlElement: Any standard HTML element (equivalent to {@link #addElement})</li>
29931      * <li>Field: Any form field (equivalent to {@link #addField})</li>
29932      * <li>Item: Any subclass of {@link Roo.Toolbar.Item} (equivalent to {@link #addItem})</li>
29933      * <li>String: Any generic string (gets wrapped in a {@link Roo.Toolbar.TextItem}, equivalent to {@link #addText}).
29934      * Note that there are a few special strings that are treated differently as explained nRoo.</li>
29935      * <li>'separator' or '-': Creates a separator element (equivalent to {@link #addSeparator})</li>
29936      * <li>' ': Creates a spacer element (equivalent to {@link #addSpacer})</li>
29937      * <li>'->': Creates a fill element (equivalent to {@link #addFill})</li>
29938      * </ul>
29939      * @param {Mixed} arg2
29940      * @param {Mixed} etc.
29941      */
29942     add : function(){
29943         var a = arguments, l = a.length;
29944         for(var i = 0; i < l; i++){
29945             this._add(a[i]);
29946         }
29947     },
29948     // private..
29949     _add : function(el) {
29950         
29951         if (el.xtype) {
29952             el = Roo.factory(el, typeof(Roo.Toolbar[el.xtype]) == 'undefined' ? Roo.form : Roo.Toolbar);
29953         }
29954         
29955         if (el.applyTo){ // some kind of form field
29956             return this.addField(el);
29957         } 
29958         if (el.render){ // some kind of Toolbar.Item
29959             return this.addItem(el);
29960         }
29961         if (typeof el == "string"){ // string
29962             if(el == "separator" || el == "-"){
29963                 return this.addSeparator();
29964             }
29965             if (el == " "){
29966                 return this.addSpacer();
29967             }
29968             if(el == "->"){
29969                 return this.addFill();
29970             }
29971             return this.addText(el);
29972             
29973         }
29974         if(el.tagName){ // element
29975             return this.addElement(el);
29976         }
29977         if(typeof el == "object"){ // must be button config?
29978             return this.addButton(el);
29979         }
29980         // and now what?!?!
29981         return false;
29982         
29983     },
29984     
29985     /**
29986      * Add an Xtype element
29987      * @param {Object} xtype Xtype Object
29988      * @return {Object} created Object
29989      */
29990     addxtype : function(e){
29991         return this.add(e);  
29992     },
29993     
29994     /**
29995      * Returns the Element for this toolbar.
29996      * @return {Roo.Element}
29997      */
29998     getEl : function(){
29999         return this.el;  
30000     },
30001     
30002     /**
30003      * Adds a separator
30004      * @return {Roo.Toolbar.Item} The separator item
30005      */
30006     addSeparator : function(){
30007         return this.addItem(new Roo.Toolbar.Separator());
30008     },
30009
30010     /**
30011      * Adds a spacer element
30012      * @return {Roo.Toolbar.Spacer} The spacer item
30013      */
30014     addSpacer : function(){
30015         return this.addItem(new Roo.Toolbar.Spacer());
30016     },
30017
30018     /**
30019      * Adds a fill element that forces subsequent additions to the right side of the toolbar
30020      * @return {Roo.Toolbar.Fill} The fill item
30021      */
30022     addFill : function(){
30023         return this.addItem(new Roo.Toolbar.Fill());
30024     },
30025
30026     /**
30027      * Adds any standard HTML element to the toolbar
30028      * @param {String/HTMLElement/Element} el The element or id of the element to add
30029      * @return {Roo.Toolbar.Item} The element's item
30030      */
30031     addElement : function(el){
30032         return this.addItem(new Roo.Toolbar.Item(el));
30033     },
30034     /**
30035      * Collection of items on the toolbar.. (only Toolbar Items, so use fields to retrieve fields)
30036      * @type Roo.util.MixedCollection  
30037      */
30038     items : false,
30039      
30040     /**
30041      * Adds any Toolbar.Item or subclass
30042      * @param {Roo.Toolbar.Item} item
30043      * @return {Roo.Toolbar.Item} The item
30044      */
30045     addItem : function(item){
30046         var td = this.nextBlock();
30047         item.render(td);
30048         this.items.add(item);
30049         return item;
30050     },
30051     
30052     /**
30053      * Adds a button (or buttons). See {@link Roo.Toolbar.Button} for more info on the config.
30054      * @param {Object/Array} config A button config or array of configs
30055      * @return {Roo.Toolbar.Button/Array}
30056      */
30057     addButton : function(config){
30058         if(config instanceof Array){
30059             var buttons = [];
30060             for(var i = 0, len = config.length; i < len; i++) {
30061                 buttons.push(this.addButton(config[i]));
30062             }
30063             return buttons;
30064         }
30065         var b = config;
30066         if(!(config instanceof Roo.Toolbar.Button)){
30067             b = config.split ?
30068                 new Roo.Toolbar.SplitButton(config) :
30069                 new Roo.Toolbar.Button(config);
30070         }
30071         var td = this.nextBlock();
30072         b.render(td);
30073         this.items.add(b);
30074         return b;
30075     },
30076     
30077     /**
30078      * Adds text to the toolbar
30079      * @param {String} text The text to add
30080      * @return {Roo.Toolbar.Item} The element's item
30081      */
30082     addText : function(text){
30083         return this.addItem(new Roo.Toolbar.TextItem(text));
30084     },
30085     
30086     /**
30087      * Inserts any {@link Roo.Toolbar.Item}/{@link Roo.Toolbar.Button} at the specified index.
30088      * @param {Number} index The index where the item is to be inserted
30089      * @param {Object/Roo.Toolbar.Item/Roo.Toolbar.Button (may be Array)} item The button, or button config object to be inserted.
30090      * @return {Roo.Toolbar.Button/Item}
30091      */
30092     insertButton : function(index, item){
30093         if(item instanceof Array){
30094             var buttons = [];
30095             for(var i = 0, len = item.length; i < len; i++) {
30096                buttons.push(this.insertButton(index + i, item[i]));
30097             }
30098             return buttons;
30099         }
30100         if (!(item instanceof Roo.Toolbar.Button)){
30101            item = new Roo.Toolbar.Button(item);
30102         }
30103         var td = document.createElement("td");
30104         this.tr.insertBefore(td, this.tr.childNodes[index]);
30105         item.render(td);
30106         this.items.insert(index, item);
30107         return item;
30108     },
30109     
30110     /**
30111      * Adds a new element to the toolbar from the passed {@link Roo.DomHelper} config.
30112      * @param {Object} config
30113      * @return {Roo.Toolbar.Item} The element's item
30114      */
30115     addDom : function(config, returnEl){
30116         var td = this.nextBlock();
30117         Roo.DomHelper.overwrite(td, config);
30118         var ti = new Roo.Toolbar.Item(td.firstChild);
30119         ti.render(td);
30120         this.items.add(ti);
30121         return ti;
30122     },
30123
30124     /**
30125      * Collection of fields on the toolbar.. usefull for quering (value is false if there are no fields)
30126      * @type Roo.util.MixedCollection  
30127      */
30128     fields : false,
30129     
30130     /**
30131      * Adds a dynamically rendered Roo.form field (TextField, ComboBox, etc).
30132      * Note: the field should not have been rendered yet. For a field that has already been
30133      * rendered, use {@link #addElement}.
30134      * @param {Roo.form.Field} field
30135      * @return {Roo.ToolbarItem}
30136      */
30137      
30138       
30139     addField : function(field) {
30140         if (!this.fields) {
30141             var autoId = 0;
30142             this.fields = new Roo.util.MixedCollection(false, function(o){
30143                 return o.id || ("item" + (++autoId));
30144             });
30145
30146         }
30147         
30148         var td = this.nextBlock();
30149         field.render(td);
30150         var ti = new Roo.Toolbar.Item(td.firstChild);
30151         ti.render(td);
30152         this.items.add(ti);
30153         this.fields.add(field);
30154         return ti;
30155     },
30156     /**
30157      * Hide the toolbar
30158      * @method hide
30159      */
30160      
30161       
30162     hide : function()
30163     {
30164         this.el.child('div').setVisibilityMode(Roo.Element.DISPLAY);
30165         this.el.child('div').hide();
30166     },
30167     /**
30168      * Show the toolbar
30169      * @method show
30170      */
30171     show : function()
30172     {
30173         this.el.child('div').show();
30174     },
30175       
30176     // private
30177     nextBlock : function(){
30178         var td = document.createElement("td");
30179         this.tr.appendChild(td);
30180         return td;
30181     },
30182
30183     // private
30184     destroy : function(){
30185         if(this.items){ // rendered?
30186             Roo.destroy.apply(Roo, this.items.items);
30187         }
30188         if(this.fields){ // rendered?
30189             Roo.destroy.apply(Roo, this.fields.items);
30190         }
30191         Roo.Element.uncache(this.el, this.tr);
30192     }
30193 };
30194
30195 /**
30196  * @class Roo.Toolbar.Item
30197  * The base class that other classes should extend in order to get some basic common toolbar item functionality.
30198  * @constructor
30199  * Creates a new Item
30200  * @param {HTMLElement} el 
30201  */
30202 Roo.Toolbar.Item = function(el){
30203     var cfg = {};
30204     if (typeof (el.xtype) != 'undefined') {
30205         cfg = el;
30206         el = cfg.el;
30207     }
30208     
30209     this.el = Roo.getDom(el);
30210     this.id = Roo.id(this.el);
30211     this.hidden = false;
30212     
30213     this.addEvents({
30214          /**
30215              * @event render
30216              * Fires when the button is rendered
30217              * @param {Button} this
30218              */
30219         'render': true
30220     });
30221     Roo.Toolbar.Item.superclass.constructor.call(this,cfg);
30222 };
30223 Roo.extend(Roo.Toolbar.Item, Roo.util.Observable, {
30224 //Roo.Toolbar.Item.prototype = {
30225     
30226     /**
30227      * Get this item's HTML Element
30228      * @return {HTMLElement}
30229      */
30230     getEl : function(){
30231        return this.el;  
30232     },
30233
30234     // private
30235     render : function(td){
30236         
30237          this.td = td;
30238         td.appendChild(this.el);
30239         
30240         this.fireEvent('render', this);
30241     },
30242     
30243     /**
30244      * Removes and destroys this item.
30245      */
30246     destroy : function(){
30247         this.td.parentNode.removeChild(this.td);
30248     },
30249     
30250     /**
30251      * Shows this item.
30252      */
30253     show: function(){
30254         this.hidden = false;
30255         this.td.style.display = "";
30256     },
30257     
30258     /**
30259      * Hides this item.
30260      */
30261     hide: function(){
30262         this.hidden = true;
30263         this.td.style.display = "none";
30264     },
30265     
30266     /**
30267      * Convenience function for boolean show/hide.
30268      * @param {Boolean} visible true to show/false to hide
30269      */
30270     setVisible: function(visible){
30271         if(visible) {
30272             this.show();
30273         }else{
30274             this.hide();
30275         }
30276     },
30277     
30278     /**
30279      * Try to focus this item.
30280      */
30281     focus : function(){
30282         Roo.fly(this.el).focus();
30283     },
30284     
30285     /**
30286      * Disables this item.
30287      */
30288     disable : function(){
30289         Roo.fly(this.td).addClass("x-item-disabled");
30290         this.disabled = true;
30291         this.el.disabled = true;
30292     },
30293     
30294     /**
30295      * Enables this item.
30296      */
30297     enable : function(){
30298         Roo.fly(this.td).removeClass("x-item-disabled");
30299         this.disabled = false;
30300         this.el.disabled = false;
30301     }
30302 });
30303
30304
30305 /**
30306  * @class Roo.Toolbar.Separator
30307  * @extends Roo.Toolbar.Item
30308  * A simple toolbar separator class
30309  * @constructor
30310  * Creates a new Separator
30311  */
30312 Roo.Toolbar.Separator = function(cfg){
30313     
30314     var s = document.createElement("span");
30315     s.className = "ytb-sep";
30316     if (cfg) {
30317         cfg.el = s;
30318     }
30319     
30320     Roo.Toolbar.Separator.superclass.constructor.call(this, cfg || s);
30321 };
30322 Roo.extend(Roo.Toolbar.Separator, Roo.Toolbar.Item, {
30323     enable:Roo.emptyFn,
30324     disable:Roo.emptyFn,
30325     focus:Roo.emptyFn
30326 });
30327
30328 /**
30329  * @class Roo.Toolbar.Spacer
30330  * @extends Roo.Toolbar.Item
30331  * A simple element that adds extra horizontal space to a toolbar.
30332  * @constructor
30333  * Creates a new Spacer
30334  */
30335 Roo.Toolbar.Spacer = function(cfg){
30336     var s = document.createElement("div");
30337     s.className = "ytb-spacer";
30338     if (cfg) {
30339         cfg.el = s;
30340     }
30341     Roo.Toolbar.Spacer.superclass.constructor.call(this, cfg || s);
30342 };
30343 Roo.extend(Roo.Toolbar.Spacer, Roo.Toolbar.Item, {
30344     enable:Roo.emptyFn,
30345     disable:Roo.emptyFn,
30346     focus:Roo.emptyFn
30347 });
30348
30349 /**
30350  * @class Roo.Toolbar.Fill
30351  * @extends Roo.Toolbar.Spacer
30352  * A simple element that adds a greedy (100% width) horizontal space to a toolbar.
30353  * @constructor
30354  * Creates a new Spacer
30355  */
30356 Roo.Toolbar.Fill = Roo.extend(Roo.Toolbar.Spacer, {
30357     // private
30358     render : function(td){
30359         td.style.width = '100%';
30360         Roo.Toolbar.Fill.superclass.render.call(this, td);
30361     }
30362 });
30363
30364 /**
30365  * @class Roo.Toolbar.TextItem
30366  * @extends Roo.Toolbar.Item
30367  * A simple class that renders text directly into a toolbar.
30368  * @constructor
30369  * Creates a new TextItem
30370  * @param {String} text
30371  */
30372 Roo.Toolbar.TextItem = function(cfg){
30373     var  text = cfg || "";
30374     if (typeof(cfg) == 'object') {
30375         text = cfg.text || "";
30376     }  else {
30377         cfg = null;
30378     }
30379     var s = document.createElement("span");
30380     s.className = "ytb-text";
30381     s.innerHTML = text;
30382     if (cfg) {
30383         cfg.el  = s;
30384     }
30385     
30386     Roo.Toolbar.TextItem.superclass.constructor.call(this, cfg ||  s);
30387 };
30388 Roo.extend(Roo.Toolbar.TextItem, Roo.Toolbar.Item, {
30389     
30390      
30391     enable:Roo.emptyFn,
30392     disable:Roo.emptyFn,
30393     focus:Roo.emptyFn
30394 });
30395
30396 /**
30397  * @class Roo.Toolbar.Button
30398  * @extends Roo.Button
30399  * A button that renders into a toolbar.
30400  * @constructor
30401  * Creates a new Button
30402  * @param {Object} config A standard {@link Roo.Button} config object
30403  */
30404 Roo.Toolbar.Button = function(config){
30405     Roo.Toolbar.Button.superclass.constructor.call(this, null, config);
30406 };
30407 Roo.extend(Roo.Toolbar.Button, Roo.Button, {
30408     render : function(td){
30409         this.td = td;
30410         Roo.Toolbar.Button.superclass.render.call(this, td);
30411     },
30412     
30413     /**
30414      * Removes and destroys this button
30415      */
30416     destroy : function(){
30417         Roo.Toolbar.Button.superclass.destroy.call(this);
30418         this.td.parentNode.removeChild(this.td);
30419     },
30420     
30421     /**
30422      * Shows this button
30423      */
30424     show: function(){
30425         this.hidden = false;
30426         this.td.style.display = "";
30427     },
30428     
30429     /**
30430      * Hides this button
30431      */
30432     hide: function(){
30433         this.hidden = true;
30434         this.td.style.display = "none";
30435     },
30436
30437     /**
30438      * Disables this item
30439      */
30440     disable : function(){
30441         Roo.fly(this.td).addClass("x-item-disabled");
30442         this.disabled = true;
30443     },
30444
30445     /**
30446      * Enables this item
30447      */
30448     enable : function(){
30449         Roo.fly(this.td).removeClass("x-item-disabled");
30450         this.disabled = false;
30451     }
30452 });
30453 // backwards compat
30454 Roo.ToolbarButton = Roo.Toolbar.Button;
30455
30456 /**
30457  * @class Roo.Toolbar.SplitButton
30458  * @extends Roo.SplitButton
30459  * A menu button that renders into a toolbar.
30460  * @constructor
30461  * Creates a new SplitButton
30462  * @param {Object} config A standard {@link Roo.SplitButton} config object
30463  */
30464 Roo.Toolbar.SplitButton = function(config){
30465     Roo.Toolbar.SplitButton.superclass.constructor.call(this, null, config);
30466 };
30467 Roo.extend(Roo.Toolbar.SplitButton, Roo.SplitButton, {
30468     render : function(td){
30469         this.td = td;
30470         Roo.Toolbar.SplitButton.superclass.render.call(this, td);
30471     },
30472     
30473     /**
30474      * Removes and destroys this button
30475      */
30476     destroy : function(){
30477         Roo.Toolbar.SplitButton.superclass.destroy.call(this);
30478         this.td.parentNode.removeChild(this.td);
30479     },
30480     
30481     /**
30482      * Shows this button
30483      */
30484     show: function(){
30485         this.hidden = false;
30486         this.td.style.display = "";
30487     },
30488     
30489     /**
30490      * Hides this button
30491      */
30492     hide: function(){
30493         this.hidden = true;
30494         this.td.style.display = "none";
30495     }
30496 });
30497
30498 // backwards compat
30499 Roo.Toolbar.MenuButton = Roo.Toolbar.SplitButton;/*
30500  * Based on:
30501  * Ext JS Library 1.1.1
30502  * Copyright(c) 2006-2007, Ext JS, LLC.
30503  *
30504  * Originally Released Under LGPL - original licence link has changed is not relivant.
30505  *
30506  * Fork - LGPL
30507  * <script type="text/javascript">
30508  */
30509  
30510 /**
30511  * @class Roo.PagingToolbar
30512  * @extends Roo.Toolbar
30513  * A specialized toolbar that is bound to a {@link Roo.data.Store} and provides automatic paging controls.
30514  * @constructor
30515  * Create a new PagingToolbar
30516  * @param {Object} config The config object
30517  */
30518 Roo.PagingToolbar = function(el, ds, config)
30519 {
30520     // old args format still supported... - xtype is prefered..
30521     if (typeof(el) == 'object' && el.xtype) {
30522         // created from xtype...
30523         config = el;
30524         ds = el.dataSource;
30525         el = config.container;
30526     }
30527     var items = [];
30528     if (config.items) {
30529         items = config.items;
30530         config.items = [];
30531     }
30532     
30533     Roo.PagingToolbar.superclass.constructor.call(this, el, null, config);
30534     this.ds = ds;
30535     this.cursor = 0;
30536     this.renderButtons(this.el);
30537     this.bind(ds);
30538     
30539     // supprot items array.
30540    
30541     Roo.each(items, function(e) {
30542         this.add(Roo.factory(e));
30543     },this);
30544     
30545 };
30546
30547 Roo.extend(Roo.PagingToolbar, Roo.Toolbar, {
30548     /**
30549      * @cfg {Roo.data.Store} dataSource
30550      * The underlying data store providing the paged data
30551      */
30552     /**
30553      * @cfg {String/HTMLElement/Element} container
30554      * container The id or element that will contain the toolbar
30555      */
30556     /**
30557      * @cfg {Boolean} displayInfo
30558      * True to display the displayMsg (defaults to false)
30559      */
30560     /**
30561      * @cfg {Number} pageSize
30562      * The number of records to display per page (defaults to 20)
30563      */
30564     pageSize: 20,
30565     /**
30566      * @cfg {String} displayMsg
30567      * The paging status message to display (defaults to "Displaying {start} - {end} of {total}")
30568      */
30569     displayMsg : 'Displaying {0} - {1} of {2}',
30570     /**
30571      * @cfg {String} emptyMsg
30572      * The message to display when no records are found (defaults to "No data to display")
30573      */
30574     emptyMsg : 'No data to display',
30575     /**
30576      * Customizable piece of the default paging text (defaults to "Page")
30577      * @type String
30578      */
30579     beforePageText : "Page",
30580     /**
30581      * Customizable piece of the default paging text (defaults to "of %0")
30582      * @type String
30583      */
30584     afterPageText : "of {0}",
30585     /**
30586      * Customizable piece of the default paging text (defaults to "First Page")
30587      * @type String
30588      */
30589     firstText : "First Page",
30590     /**
30591      * Customizable piece of the default paging text (defaults to "Previous Page")
30592      * @type String
30593      */
30594     prevText : "Previous Page",
30595     /**
30596      * Customizable piece of the default paging text (defaults to "Next Page")
30597      * @type String
30598      */
30599     nextText : "Next Page",
30600     /**
30601      * Customizable piece of the default paging text (defaults to "Last Page")
30602      * @type String
30603      */
30604     lastText : "Last Page",
30605     /**
30606      * Customizable piece of the default paging text (defaults to "Refresh")
30607      * @type String
30608      */
30609     refreshText : "Refresh",
30610
30611     // private
30612     renderButtons : function(el){
30613         Roo.PagingToolbar.superclass.render.call(this, el);
30614         this.first = this.addButton({
30615             tooltip: this.firstText,
30616             cls: "x-btn-icon x-grid-page-first",
30617             disabled: true,
30618             handler: this.onClick.createDelegate(this, ["first"])
30619         });
30620         this.prev = this.addButton({
30621             tooltip: this.prevText,
30622             cls: "x-btn-icon x-grid-page-prev",
30623             disabled: true,
30624             handler: this.onClick.createDelegate(this, ["prev"])
30625         });
30626         //this.addSeparator();
30627         this.add(this.beforePageText);
30628         this.field = Roo.get(this.addDom({
30629            tag: "input",
30630            type: "text",
30631            size: "3",
30632            value: "1",
30633            cls: "x-grid-page-number"
30634         }).el);
30635         this.field.on("keydown", this.onPagingKeydown, this);
30636         this.field.on("focus", function(){this.dom.select();});
30637         this.afterTextEl = this.addText(String.format(this.afterPageText, 1));
30638         this.field.setHeight(18);
30639         //this.addSeparator();
30640         this.next = this.addButton({
30641             tooltip: this.nextText,
30642             cls: "x-btn-icon x-grid-page-next",
30643             disabled: true,
30644             handler: this.onClick.createDelegate(this, ["next"])
30645         });
30646         this.last = this.addButton({
30647             tooltip: this.lastText,
30648             cls: "x-btn-icon x-grid-page-last",
30649             disabled: true,
30650             handler: this.onClick.createDelegate(this, ["last"])
30651         });
30652         //this.addSeparator();
30653         this.loading = this.addButton({
30654             tooltip: this.refreshText,
30655             cls: "x-btn-icon x-grid-loading",
30656             handler: this.onClick.createDelegate(this, ["refresh"])
30657         });
30658
30659         if(this.displayInfo){
30660             this.displayEl = Roo.fly(this.el.dom.firstChild).createChild({cls:'x-paging-info'});
30661         }
30662     },
30663
30664     // private
30665     updateInfo : function(){
30666         if(this.displayEl){
30667             var count = this.ds.getCount();
30668             var msg = count == 0 ?
30669                 this.emptyMsg :
30670                 String.format(
30671                     this.displayMsg,
30672                     this.cursor+1, this.cursor+count, this.ds.getTotalCount()    
30673                 );
30674             this.displayEl.update(msg);
30675         }
30676     },
30677
30678     // private
30679     onLoad : function(ds, r, o){
30680        this.cursor = o.params ? o.params.start : 0;
30681        var d = this.getPageData(), ap = d.activePage, ps = d.pages;
30682
30683        this.afterTextEl.el.innerHTML = String.format(this.afterPageText, d.pages);
30684        this.field.dom.value = ap;
30685        this.first.setDisabled(ap == 1);
30686        this.prev.setDisabled(ap == 1);
30687        this.next.setDisabled(ap == ps);
30688        this.last.setDisabled(ap == ps);
30689        this.loading.enable();
30690        this.updateInfo();
30691     },
30692
30693     // private
30694     getPageData : function(){
30695         var total = this.ds.getTotalCount();
30696         return {
30697             total : total,
30698             activePage : Math.ceil((this.cursor+this.pageSize)/this.pageSize),
30699             pages :  total < this.pageSize ? 1 : Math.ceil(total/this.pageSize)
30700         };
30701     },
30702
30703     // private
30704     onLoadError : function(){
30705         this.loading.enable();
30706     },
30707
30708     // private
30709     onPagingKeydown : function(e){
30710         var k = e.getKey();
30711         var d = this.getPageData();
30712         if(k == e.RETURN){
30713             var v = this.field.dom.value, pageNum;
30714             if(!v || isNaN(pageNum = parseInt(v, 10))){
30715                 this.field.dom.value = d.activePage;
30716                 return;
30717             }
30718             pageNum = Math.min(Math.max(1, pageNum), d.pages) - 1;
30719             this.ds.load({params:{start: pageNum * this.pageSize, limit: this.pageSize}});
30720             e.stopEvent();
30721         }
30722         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))
30723         {
30724           var pageNum = (k == e.HOME || (k == e.DOWN && e.ctrlKey) || (k == e.LEFT && e.ctrlKey) || (k == e.PAGEDOWN && e.ctrlKey)) ? 1 : d.pages;
30725           this.field.dom.value = pageNum;
30726           this.ds.load({params:{start: (pageNum - 1) * this.pageSize, limit: this.pageSize}});
30727           e.stopEvent();
30728         }
30729         else if(k == e.UP || k == e.RIGHT || k == e.PAGEUP || k == e.DOWN || k == e.LEFT || k == e.PAGEDOWN)
30730         {
30731           var v = this.field.dom.value, pageNum; 
30732           var increment = (e.shiftKey) ? 10 : 1;
30733           if(k == e.DOWN || k == e.LEFT || k == e.PAGEDOWN) {
30734             increment *= -1;
30735           }
30736           if(!v || isNaN(pageNum = parseInt(v, 10))) {
30737             this.field.dom.value = d.activePage;
30738             return;
30739           }
30740           else if(parseInt(v, 10) + increment >= 1 & parseInt(v, 10) + increment <= d.pages)
30741           {
30742             this.field.dom.value = parseInt(v, 10) + increment;
30743             pageNum = Math.min(Math.max(1, pageNum + increment), d.pages) - 1;
30744             this.ds.load({params:{start: pageNum * this.pageSize, limit: this.pageSize}});
30745           }
30746           e.stopEvent();
30747         }
30748     },
30749
30750     // private
30751     beforeLoad : function(){
30752         if(this.loading){
30753             this.loading.disable();
30754         }
30755     },
30756
30757     // private
30758     onClick : function(which){
30759         var ds = this.ds;
30760         switch(which){
30761             case "first":
30762                 ds.load({params:{start: 0, limit: this.pageSize}});
30763             break;
30764             case "prev":
30765                 ds.load({params:{start: Math.max(0, this.cursor-this.pageSize), limit: this.pageSize}});
30766             break;
30767             case "next":
30768                 ds.load({params:{start: this.cursor+this.pageSize, limit: this.pageSize}});
30769             break;
30770             case "last":
30771                 var total = ds.getTotalCount();
30772                 var extra = total % this.pageSize;
30773                 var lastStart = extra ? (total - extra) : total-this.pageSize;
30774                 ds.load({params:{start: lastStart, limit: this.pageSize}});
30775             break;
30776             case "refresh":
30777                 ds.load({params:{start: this.cursor, limit: this.pageSize}});
30778             break;
30779         }
30780     },
30781
30782     /**
30783      * Unbinds the paging toolbar from the specified {@link Roo.data.Store}
30784      * @param {Roo.data.Store} store The data store to unbind
30785      */
30786     unbind : function(ds){
30787         ds.un("beforeload", this.beforeLoad, this);
30788         ds.un("load", this.onLoad, this);
30789         ds.un("loadexception", this.onLoadError, this);
30790         ds.un("remove", this.updateInfo, this);
30791         ds.un("add", this.updateInfo, this);
30792         this.ds = undefined;
30793     },
30794
30795     /**
30796      * Binds the paging toolbar to the specified {@link Roo.data.Store}
30797      * @param {Roo.data.Store} store The data store to bind
30798      */
30799     bind : function(ds){
30800         ds.on("beforeload", this.beforeLoad, this);
30801         ds.on("load", this.onLoad, this);
30802         ds.on("loadexception", this.onLoadError, this);
30803         ds.on("remove", this.updateInfo, this);
30804         ds.on("add", this.updateInfo, this);
30805         this.ds = ds;
30806     }
30807 });/*
30808  * Based on:
30809  * Ext JS Library 1.1.1
30810  * Copyright(c) 2006-2007, Ext JS, LLC.
30811  *
30812  * Originally Released Under LGPL - original licence link has changed is not relivant.
30813  *
30814  * Fork - LGPL
30815  * <script type="text/javascript">
30816  */
30817
30818 /**
30819  * @class Roo.Resizable
30820  * @extends Roo.util.Observable
30821  * <p>Applies drag handles to an element to make it resizable. The drag handles are inserted into the element
30822  * and positioned absolute. Some elements, such as a textarea or image, don't support this. To overcome that, you can wrap
30823  * 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
30824  * the element will be wrapped for you automatically.</p>
30825  * <p>Here is the list of valid resize handles:</p>
30826  * <pre>
30827 Value   Description
30828 ------  -------------------
30829  'n'     north
30830  's'     south
30831  'e'     east
30832  'w'     west
30833  'nw'    northwest
30834  'sw'    southwest
30835  'se'    southeast
30836  'ne'    northeast
30837  'hd'    horizontal drag
30838  'all'   all
30839 </pre>
30840  * <p>Here's an example showing the creation of a typical Resizable:</p>
30841  * <pre><code>
30842 var resizer = new Roo.Resizable("element-id", {
30843     handles: 'all',
30844     minWidth: 200,
30845     minHeight: 100,
30846     maxWidth: 500,
30847     maxHeight: 400,
30848     pinned: true
30849 });
30850 resizer.on("resize", myHandler);
30851 </code></pre>
30852  * <p>To hide a particular handle, set its display to none in CSS, or through script:<br>
30853  * resizer.east.setDisplayed(false);</p>
30854  * @cfg {Boolean/String/Element} resizeChild True to resize the first child, or id/element to resize (defaults to false)
30855  * @cfg {Array/String} adjustments String "auto" or an array [width, height] with values to be <b>added</b> to the
30856  * resize operation's new size (defaults to [0, 0])
30857  * @cfg {Number} minWidth The minimum width for the element (defaults to 5)
30858  * @cfg {Number} minHeight The minimum height for the element (defaults to 5)
30859  * @cfg {Number} maxWidth The maximum width for the element (defaults to 10000)
30860  * @cfg {Number} maxHeight The maximum height for the element (defaults to 10000)
30861  * @cfg {Boolean} enabled False to disable resizing (defaults to true)
30862  * @cfg {Boolean} wrap True to wrap an element with a div if needed (required for textareas and images, defaults to false)
30863  * @cfg {Number} width The width of the element in pixels (defaults to null)
30864  * @cfg {Number} height The height of the element in pixels (defaults to null)
30865  * @cfg {Boolean} animate True to animate the resize (not compatible with dynamic sizing, defaults to false)
30866  * @cfg {Number} duration Animation duration if animate = true (defaults to .35)
30867  * @cfg {Boolean} dynamic True to resize the element while dragging instead of using a proxy (defaults to false)
30868  * @cfg {String} handles String consisting of the resize handles to display (defaults to undefined)
30869  * @cfg {Boolean} multiDirectional <b>Deprecated</b>.  The old style of adding multi-direction resize handles, deprecated
30870  * in favor of the handles config option (defaults to false)
30871  * @cfg {Boolean} disableTrackOver True to disable mouse tracking. This is only applied at config time. (defaults to false)
30872  * @cfg {String} easing Animation easing if animate = true (defaults to 'easingOutStrong')
30873  * @cfg {Number} widthIncrement The increment to snap the width resize in pixels (dynamic must be true, defaults to 0)
30874  * @cfg {Number} heightIncrement The increment to snap the height resize in pixels (dynamic must be true, defaults to 0)
30875  * @cfg {Boolean} pinned True to ensure that the resize handles are always visible, false to display them only when the
30876  * user mouses over the resizable borders. This is only applied at config time. (defaults to false)
30877  * @cfg {Boolean} preserveRatio True to preserve the original ratio between height and width during resize (defaults to false)
30878  * @cfg {Boolean} transparent True for transparent handles. This is only applied at config time. (defaults to false)
30879  * @cfg {Number} minX The minimum allowed page X for the element (only used for west resizing, defaults to 0)
30880  * @cfg {Number} minY The minimum allowed page Y for the element (only used for north resizing, defaults to 0)
30881  * @cfg {Boolean} draggable Convenience to initialize drag drop (defaults to false)
30882  * @constructor
30883  * Create a new resizable component
30884  * @param {String/HTMLElement/Roo.Element} el The id or element to resize
30885  * @param {Object} config configuration options
30886   */
30887 Roo.Resizable = function(el, config)
30888 {
30889     this.el = Roo.get(el);
30890
30891     if(config && config.wrap){
30892         config.resizeChild = this.el;
30893         this.el = this.el.wrap(typeof config.wrap == "object" ? config.wrap : {cls:"xresizable-wrap"});
30894         this.el.id = this.el.dom.id = config.resizeChild.id + "-rzwrap";
30895         this.el.setStyle("overflow", "hidden");
30896         this.el.setPositioning(config.resizeChild.getPositioning());
30897         config.resizeChild.clearPositioning();
30898         if(!config.width || !config.height){
30899             var csize = config.resizeChild.getSize();
30900             this.el.setSize(csize.width, csize.height);
30901         }
30902         if(config.pinned && !config.adjustments){
30903             config.adjustments = "auto";
30904         }
30905     }
30906
30907     this.proxy = this.el.createProxy({tag: "div", cls: "x-resizable-proxy", id: this.el.id + "-rzproxy"});
30908     this.proxy.unselectable();
30909     this.proxy.enableDisplayMode('block');
30910
30911     Roo.apply(this, config);
30912
30913     if(this.pinned){
30914         this.disableTrackOver = true;
30915         this.el.addClass("x-resizable-pinned");
30916     }
30917     // if the element isn't positioned, make it relative
30918     var position = this.el.getStyle("position");
30919     if(position != "absolute" && position != "fixed"){
30920         this.el.setStyle("position", "relative");
30921     }
30922     if(!this.handles){ // no handles passed, must be legacy style
30923         this.handles = 's,e,se';
30924         if(this.multiDirectional){
30925             this.handles += ',n,w';
30926         }
30927     }
30928     if(this.handles == "all"){
30929         this.handles = "n s e w ne nw se sw";
30930     }
30931     var hs = this.handles.split(/\s*?[,;]\s*?| /);
30932     var ps = Roo.Resizable.positions;
30933     for(var i = 0, len = hs.length; i < len; i++){
30934         if(hs[i] && ps[hs[i]]){
30935             var pos = ps[hs[i]];
30936             this[pos] = new Roo.Resizable.Handle(this, pos, this.disableTrackOver, this.transparent);
30937         }
30938     }
30939     // legacy
30940     this.corner = this.southeast;
30941     
30942     // updateBox = the box can move..
30943     if(this.handles.indexOf("n") != -1 || this.handles.indexOf("w") != -1 || this.handles.indexOf("hd") != -1) {
30944         this.updateBox = true;
30945     }
30946
30947     this.activeHandle = null;
30948
30949     if(this.resizeChild){
30950         if(typeof this.resizeChild == "boolean"){
30951             this.resizeChild = Roo.get(this.el.dom.firstChild, true);
30952         }else{
30953             this.resizeChild = Roo.get(this.resizeChild, true);
30954         }
30955     }
30956     
30957     if(this.adjustments == "auto"){
30958         var rc = this.resizeChild;
30959         var hw = this.west, he = this.east, hn = this.north, hs = this.south;
30960         if(rc && (hw || hn)){
30961             rc.position("relative");
30962             rc.setLeft(hw ? hw.el.getWidth() : 0);
30963             rc.setTop(hn ? hn.el.getHeight() : 0);
30964         }
30965         this.adjustments = [
30966             (he ? -he.el.getWidth() : 0) + (hw ? -hw.el.getWidth() : 0),
30967             (hn ? -hn.el.getHeight() : 0) + (hs ? -hs.el.getHeight() : 0) -1
30968         ];
30969     }
30970
30971     if(this.draggable){
30972         this.dd = this.dynamic ?
30973             this.el.initDD(null) : this.el.initDDProxy(null, {dragElId: this.proxy.id});
30974         this.dd.setHandleElId(this.resizeChild ? this.resizeChild.id : this.el.id);
30975     }
30976
30977     // public events
30978     this.addEvents({
30979         /**
30980          * @event beforeresize
30981          * Fired before resize is allowed. Set enabled to false to cancel resize.
30982          * @param {Roo.Resizable} this
30983          * @param {Roo.EventObject} e The mousedown event
30984          */
30985         "beforeresize" : true,
30986         /**
30987          * @event resizing
30988          * Fired a resizing.
30989          * @param {Roo.Resizable} this
30990          * @param {Number} x The new x position
30991          * @param {Number} y The new y position
30992          * @param {Number} w The new w width
30993          * @param {Number} h The new h hight
30994          * @param {Roo.EventObject} e The mouseup event
30995          */
30996         "resizing" : true,
30997         /**
30998          * @event resize
30999          * Fired after a resize.
31000          * @param {Roo.Resizable} this
31001          * @param {Number} width The new width
31002          * @param {Number} height The new height
31003          * @param {Roo.EventObject} e The mouseup event
31004          */
31005         "resize" : true
31006     });
31007
31008     if(this.width !== null && this.height !== null){
31009         this.resizeTo(this.width, this.height);
31010     }else{
31011         this.updateChildSize();
31012     }
31013     if(Roo.isIE){
31014         this.el.dom.style.zoom = 1;
31015     }
31016     Roo.Resizable.superclass.constructor.call(this);
31017 };
31018
31019 Roo.extend(Roo.Resizable, Roo.util.Observable, {
31020         resizeChild : false,
31021         adjustments : [0, 0],
31022         minWidth : 5,
31023         minHeight : 5,
31024         maxWidth : 10000,
31025         maxHeight : 10000,
31026         enabled : true,
31027         animate : false,
31028         duration : .35,
31029         dynamic : false,
31030         handles : false,
31031         multiDirectional : false,
31032         disableTrackOver : false,
31033         easing : 'easeOutStrong',
31034         widthIncrement : 0,
31035         heightIncrement : 0,
31036         pinned : false,
31037         width : null,
31038         height : null,
31039         preserveRatio : false,
31040         transparent: false,
31041         minX: 0,
31042         minY: 0,
31043         draggable: false,
31044
31045         /**
31046          * @cfg {String/HTMLElement/Element} constrainTo Constrain the resize to a particular element
31047          */
31048         constrainTo: undefined,
31049         /**
31050          * @cfg {Roo.lib.Region} resizeRegion Constrain the resize to a particular region
31051          */
31052         resizeRegion: undefined,
31053
31054
31055     /**
31056      * Perform a manual resize
31057      * @param {Number} width
31058      * @param {Number} height
31059      */
31060     resizeTo : function(width, height){
31061         this.el.setSize(width, height);
31062         this.updateChildSize();
31063         this.fireEvent("resize", this, width, height, null);
31064     },
31065
31066     // private
31067     startSizing : function(e, handle){
31068         this.fireEvent("beforeresize", this, e);
31069         if(this.enabled){ // 2nd enabled check in case disabled before beforeresize handler
31070
31071             if(!this.overlay){
31072                 this.overlay = this.el.createProxy({tag: "div", cls: "x-resizable-overlay", html: "&#160;"});
31073                 this.overlay.unselectable();
31074                 this.overlay.enableDisplayMode("block");
31075                 this.overlay.on("mousemove", this.onMouseMove, this);
31076                 this.overlay.on("mouseup", this.onMouseUp, this);
31077             }
31078             this.overlay.setStyle("cursor", handle.el.getStyle("cursor"));
31079
31080             this.resizing = true;
31081             this.startBox = this.el.getBox();
31082             this.startPoint = e.getXY();
31083             this.offsets = [(this.startBox.x + this.startBox.width) - this.startPoint[0],
31084                             (this.startBox.y + this.startBox.height) - this.startPoint[1]];
31085
31086             this.overlay.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
31087             this.overlay.show();
31088
31089             if(this.constrainTo) {
31090                 var ct = Roo.get(this.constrainTo);
31091                 this.resizeRegion = ct.getRegion().adjust(
31092                     ct.getFrameWidth('t'),
31093                     ct.getFrameWidth('l'),
31094                     -ct.getFrameWidth('b'),
31095                     -ct.getFrameWidth('r')
31096                 );
31097             }
31098
31099             this.proxy.setStyle('visibility', 'hidden'); // workaround display none
31100             this.proxy.show();
31101             this.proxy.setBox(this.startBox);
31102             if(!this.dynamic){
31103                 this.proxy.setStyle('visibility', 'visible');
31104             }
31105         }
31106     },
31107
31108     // private
31109     onMouseDown : function(handle, e){
31110         if(this.enabled){
31111             e.stopEvent();
31112             this.activeHandle = handle;
31113             this.startSizing(e, handle);
31114         }
31115     },
31116
31117     // private
31118     onMouseUp : function(e){
31119         var size = this.resizeElement();
31120         this.resizing = false;
31121         this.handleOut();
31122         this.overlay.hide();
31123         this.proxy.hide();
31124         this.fireEvent("resize", this, size.width, size.height, e);
31125     },
31126
31127     // private
31128     updateChildSize : function(){
31129         
31130         if(this.resizeChild){
31131             var el = this.el;
31132             var child = this.resizeChild;
31133             var adj = this.adjustments;
31134             if(el.dom.offsetWidth){
31135                 var b = el.getSize(true);
31136                 child.setSize(b.width+adj[0], b.height+adj[1]);
31137             }
31138             // Second call here for IE
31139             // The first call enables instant resizing and
31140             // the second call corrects scroll bars if they
31141             // exist
31142             if(Roo.isIE){
31143                 setTimeout(function(){
31144                     if(el.dom.offsetWidth){
31145                         var b = el.getSize(true);
31146                         child.setSize(b.width+adj[0], b.height+adj[1]);
31147                     }
31148                 }, 10);
31149             }
31150         }
31151     },
31152
31153     // private
31154     snap : function(value, inc, min){
31155         if(!inc || !value) {
31156             return value;
31157         }
31158         var newValue = value;
31159         var m = value % inc;
31160         if(m > 0){
31161             if(m > (inc/2)){
31162                 newValue = value + (inc-m);
31163             }else{
31164                 newValue = value - m;
31165             }
31166         }
31167         return Math.max(min, newValue);
31168     },
31169
31170     // private
31171     resizeElement : function(){
31172         var box = this.proxy.getBox();
31173         if(this.updateBox){
31174             this.el.setBox(box, false, this.animate, this.duration, null, this.easing);
31175         }else{
31176             this.el.setSize(box.width, box.height, this.animate, this.duration, null, this.easing);
31177         }
31178         this.updateChildSize();
31179         if(!this.dynamic){
31180             this.proxy.hide();
31181         }
31182         return box;
31183     },
31184
31185     // private
31186     constrain : function(v, diff, m, mx){
31187         if(v - diff < m){
31188             diff = v - m;
31189         }else if(v - diff > mx){
31190             diff = mx - v;
31191         }
31192         return diff;
31193     },
31194
31195     // private
31196     onMouseMove : function(e){
31197         
31198         if(this.enabled){
31199             try{// try catch so if something goes wrong the user doesn't get hung
31200
31201             if(this.resizeRegion && !this.resizeRegion.contains(e.getPoint())) {
31202                 return;
31203             }
31204
31205             //var curXY = this.startPoint;
31206             var curSize = this.curSize || this.startBox;
31207             var x = this.startBox.x, y = this.startBox.y;
31208             var ox = x, oy = y;
31209             var w = curSize.width, h = curSize.height;
31210             var ow = w, oh = h;
31211             var mw = this.minWidth, mh = this.minHeight;
31212             var mxw = this.maxWidth, mxh = this.maxHeight;
31213             var wi = this.widthIncrement;
31214             var hi = this.heightIncrement;
31215
31216             var eventXY = e.getXY();
31217             var diffX = -(this.startPoint[0] - Math.max(this.minX, eventXY[0]));
31218             var diffY = -(this.startPoint[1] - Math.max(this.minY, eventXY[1]));
31219
31220             var pos = this.activeHandle.position;
31221
31222             switch(pos){
31223                 case "east":
31224                     w += diffX;
31225                     w = Math.min(Math.max(mw, w), mxw);
31226                     break;
31227              
31228                 case "south":
31229                     h += diffY;
31230                     h = Math.min(Math.max(mh, h), mxh);
31231                     break;
31232                 case "southeast":
31233                     w += diffX;
31234                     h += diffY;
31235                     w = Math.min(Math.max(mw, w), mxw);
31236                     h = Math.min(Math.max(mh, h), mxh);
31237                     break;
31238                 case "north":
31239                     diffY = this.constrain(h, diffY, mh, mxh);
31240                     y += diffY;
31241                     h -= diffY;
31242                     break;
31243                 case "hdrag":
31244                     
31245                     if (wi) {
31246                         var adiffX = Math.abs(diffX);
31247                         var sub = (adiffX % wi); // how much 
31248                         if (sub > (wi/2)) { // far enough to snap
31249                             diffX = (diffX > 0) ? diffX-sub + wi : diffX+sub - wi;
31250                         } else {
31251                             // remove difference.. 
31252                             diffX = (diffX > 0) ? diffX-sub : diffX+sub;
31253                         }
31254                     }
31255                     x += diffX;
31256                     x = Math.max(this.minX, x);
31257                     break;
31258                 case "west":
31259                     diffX = this.constrain(w, diffX, mw, mxw);
31260                     x += diffX;
31261                     w -= diffX;
31262                     break;
31263                 case "northeast":
31264                     w += diffX;
31265                     w = Math.min(Math.max(mw, w), mxw);
31266                     diffY = this.constrain(h, diffY, mh, mxh);
31267                     y += diffY;
31268                     h -= diffY;
31269                     break;
31270                 case "northwest":
31271                     diffX = this.constrain(w, diffX, mw, mxw);
31272                     diffY = this.constrain(h, diffY, mh, mxh);
31273                     y += diffY;
31274                     h -= diffY;
31275                     x += diffX;
31276                     w -= diffX;
31277                     break;
31278                case "southwest":
31279                     diffX = this.constrain(w, diffX, mw, mxw);
31280                     h += diffY;
31281                     h = Math.min(Math.max(mh, h), mxh);
31282                     x += diffX;
31283                     w -= diffX;
31284                     break;
31285             }
31286
31287             var sw = this.snap(w, wi, mw);
31288             var sh = this.snap(h, hi, mh);
31289             if(sw != w || sh != h){
31290                 switch(pos){
31291                     case "northeast":
31292                         y -= sh - h;
31293                     break;
31294                     case "north":
31295                         y -= sh - h;
31296                         break;
31297                     case "southwest":
31298                         x -= sw - w;
31299                     break;
31300                     case "west":
31301                         x -= sw - w;
31302                         break;
31303                     case "northwest":
31304                         x -= sw - w;
31305                         y -= sh - h;
31306                     break;
31307                 }
31308                 w = sw;
31309                 h = sh;
31310             }
31311
31312             if(this.preserveRatio){
31313                 switch(pos){
31314                     case "southeast":
31315                     case "east":
31316                         h = oh * (w/ow);
31317                         h = Math.min(Math.max(mh, h), mxh);
31318                         w = ow * (h/oh);
31319                        break;
31320                     case "south":
31321                         w = ow * (h/oh);
31322                         w = Math.min(Math.max(mw, w), mxw);
31323                         h = oh * (w/ow);
31324                         break;
31325                     case "northeast":
31326                         w = ow * (h/oh);
31327                         w = Math.min(Math.max(mw, w), mxw);
31328                         h = oh * (w/ow);
31329                     break;
31330                     case "north":
31331                         var tw = w;
31332                         w = ow * (h/oh);
31333                         w = Math.min(Math.max(mw, w), mxw);
31334                         h = oh * (w/ow);
31335                         x += (tw - w) / 2;
31336                         break;
31337                     case "southwest":
31338                         h = oh * (w/ow);
31339                         h = Math.min(Math.max(mh, h), mxh);
31340                         var tw = w;
31341                         w = ow * (h/oh);
31342                         x += tw - w;
31343                         break;
31344                     case "west":
31345                         var th = h;
31346                         h = oh * (w/ow);
31347                         h = Math.min(Math.max(mh, h), mxh);
31348                         y += (th - h) / 2;
31349                         var tw = w;
31350                         w = ow * (h/oh);
31351                         x += tw - w;
31352                        break;
31353                     case "northwest":
31354                         var tw = w;
31355                         var th = h;
31356                         h = oh * (w/ow);
31357                         h = Math.min(Math.max(mh, h), mxh);
31358                         w = ow * (h/oh);
31359                         y += th - h;
31360                         x += tw - w;
31361                        break;
31362
31363                 }
31364             }
31365             if (pos == 'hdrag') {
31366                 w = ow;
31367             }
31368             this.proxy.setBounds(x, y, w, h);
31369             if(this.dynamic){
31370                 this.resizeElement();
31371             }
31372             }catch(e){}
31373         }
31374         this.fireEvent("resizing", this, x, y, w, h, e);
31375     },
31376
31377     // private
31378     handleOver : function(){
31379         if(this.enabled){
31380             this.el.addClass("x-resizable-over");
31381         }
31382     },
31383
31384     // private
31385     handleOut : function(){
31386         if(!this.resizing){
31387             this.el.removeClass("x-resizable-over");
31388         }
31389     },
31390
31391     /**
31392      * Returns the element this component is bound to.
31393      * @return {Roo.Element}
31394      */
31395     getEl : function(){
31396         return this.el;
31397     },
31398
31399     /**
31400      * Returns the resizeChild element (or null).
31401      * @return {Roo.Element}
31402      */
31403     getResizeChild : function(){
31404         return this.resizeChild;
31405     },
31406     groupHandler : function()
31407     {
31408         
31409     },
31410     /**
31411      * Destroys this resizable. If the element was wrapped and
31412      * removeEl is not true then the element remains.
31413      * @param {Boolean} removeEl (optional) true to remove the element from the DOM
31414      */
31415     destroy : function(removeEl){
31416         this.proxy.remove();
31417         if(this.overlay){
31418             this.overlay.removeAllListeners();
31419             this.overlay.remove();
31420         }
31421         var ps = Roo.Resizable.positions;
31422         for(var k in ps){
31423             if(typeof ps[k] != "function" && this[ps[k]]){
31424                 var h = this[ps[k]];
31425                 h.el.removeAllListeners();
31426                 h.el.remove();
31427             }
31428         }
31429         if(removeEl){
31430             this.el.update("");
31431             this.el.remove();
31432         }
31433     }
31434 });
31435
31436 // private
31437 // hash to map config positions to true positions
31438 Roo.Resizable.positions = {
31439     n: "north", s: "south", e: "east", w: "west", se: "southeast", sw: "southwest", nw: "northwest", ne: "northeast", 
31440     hd: "hdrag"
31441 };
31442
31443 // private
31444 Roo.Resizable.Handle = function(rz, pos, disableTrackOver, transparent){
31445     if(!this.tpl){
31446         // only initialize the template if resizable is used
31447         var tpl = Roo.DomHelper.createTemplate(
31448             {tag: "div", cls: "x-resizable-handle x-resizable-handle-{0}"}
31449         );
31450         tpl.compile();
31451         Roo.Resizable.Handle.prototype.tpl = tpl;
31452     }
31453     this.position = pos;
31454     this.rz = rz;
31455     // show north drag fro topdra
31456     var handlepos = pos == 'hdrag' ? 'north' : pos;
31457     
31458     this.el = this.tpl.append(rz.el.dom, [handlepos], true);
31459     if (pos == 'hdrag') {
31460         this.el.setStyle('cursor', 'pointer');
31461     }
31462     this.el.unselectable();
31463     if(transparent){
31464         this.el.setOpacity(0);
31465     }
31466     this.el.on("mousedown", this.onMouseDown, this);
31467     if(!disableTrackOver){
31468         this.el.on("mouseover", this.onMouseOver, this);
31469         this.el.on("mouseout", this.onMouseOut, this);
31470     }
31471 };
31472
31473 // private
31474 Roo.Resizable.Handle.prototype = {
31475     afterResize : function(rz){
31476         Roo.log('after?');
31477         // do nothing
31478     },
31479     // private
31480     onMouseDown : function(e){
31481         this.rz.onMouseDown(this, e);
31482     },
31483     // private
31484     onMouseOver : function(e){
31485         this.rz.handleOver(this, e);
31486     },
31487     // private
31488     onMouseOut : function(e){
31489         this.rz.handleOut(this, e);
31490     }
31491 };/*
31492  * Based on:
31493  * Ext JS Library 1.1.1
31494  * Copyright(c) 2006-2007, Ext JS, LLC.
31495  *
31496  * Originally Released Under LGPL - original licence link has changed is not relivant.
31497  *
31498  * Fork - LGPL
31499  * <script type="text/javascript">
31500  */
31501
31502 /**
31503  * @class Roo.Editor
31504  * @extends Roo.Component
31505  * A base editor field that handles displaying/hiding on demand and has some built-in sizing and event handling logic.
31506  * @constructor
31507  * Create a new Editor
31508  * @param {Roo.form.Field} field The Field object (or descendant)
31509  * @param {Object} config The config object
31510  */
31511 Roo.Editor = function(field, config){
31512     Roo.Editor.superclass.constructor.call(this, config);
31513     this.field = field;
31514     this.addEvents({
31515         /**
31516              * @event beforestartedit
31517              * Fires when editing is initiated, but before the value changes.  Editing can be canceled by returning
31518              * false from the handler of this event.
31519              * @param {Editor} this
31520              * @param {Roo.Element} boundEl The underlying element bound to this editor
31521              * @param {Mixed} value The field value being set
31522              */
31523         "beforestartedit" : true,
31524         /**
31525              * @event startedit
31526              * Fires when this editor is displayed
31527              * @param {Roo.Element} boundEl The underlying element bound to this editor
31528              * @param {Mixed} value The starting field value
31529              */
31530         "startedit" : true,
31531         /**
31532              * @event beforecomplete
31533              * Fires after a change has been made to the field, but before the change is reflected in the underlying
31534              * field.  Saving the change to the field can be canceled by returning false from the handler of this event.
31535              * Note that if the value has not changed and ignoreNoChange = true, the editing will still end but this
31536              * event will not fire since no edit actually occurred.
31537              * @param {Editor} this
31538              * @param {Mixed} value The current field value
31539              * @param {Mixed} startValue The original field value
31540              */
31541         "beforecomplete" : true,
31542         /**
31543              * @event complete
31544              * Fires after editing is complete and any changed value has been written to the underlying field.
31545              * @param {Editor} this
31546              * @param {Mixed} value The current field value
31547              * @param {Mixed} startValue The original field value
31548              */
31549         "complete" : true,
31550         /**
31551          * @event specialkey
31552          * Fires when any key related to navigation (arrows, tab, enter, esc, etc.) is pressed.  You can check
31553          * {@link Roo.EventObject#getKey} to determine which key was pressed.
31554          * @param {Roo.form.Field} this
31555          * @param {Roo.EventObject} e The event object
31556          */
31557         "specialkey" : true
31558     });
31559 };
31560
31561 Roo.extend(Roo.Editor, Roo.Component, {
31562     /**
31563      * @cfg {Boolean/String} autosize
31564      * True for the editor to automatically adopt the size of the underlying field, "width" to adopt the width only,
31565      * or "height" to adopt the height only (defaults to false)
31566      */
31567     /**
31568      * @cfg {Boolean} revertInvalid
31569      * True to automatically revert the field value and cancel the edit when the user completes an edit and the field
31570      * validation fails (defaults to true)
31571      */
31572     /**
31573      * @cfg {Boolean} ignoreNoChange
31574      * True to skip the the edit completion process (no save, no events fired) if the user completes an edit and
31575      * the value has not changed (defaults to false).  Applies only to string values - edits for other data types
31576      * will never be ignored.
31577      */
31578     /**
31579      * @cfg {Boolean} hideEl
31580      * False to keep the bound element visible while the editor is displayed (defaults to true)
31581      */
31582     /**
31583      * @cfg {Mixed} value
31584      * The data value of the underlying field (defaults to "")
31585      */
31586     value : "",
31587     /**
31588      * @cfg {String} alignment
31589      * The position to align to (see {@link Roo.Element#alignTo} for more details, defaults to "c-c?").
31590      */
31591     alignment: "c-c?",
31592     /**
31593      * @cfg {Boolean/String} shadow "sides" for sides/bottom only, "frame" for 4-way shadow, and "drop"
31594      * for bottom-right shadow (defaults to "frame")
31595      */
31596     shadow : "frame",
31597     /**
31598      * @cfg {Boolean} constrain True to constrain the editor to the viewport
31599      */
31600     constrain : false,
31601     /**
31602      * @cfg {Boolean} completeOnEnter True to complete the edit when the enter key is pressed (defaults to false)
31603      */
31604     completeOnEnter : false,
31605     /**
31606      * @cfg {Boolean} cancelOnEsc True to cancel the edit when the escape key is pressed (defaults to false)
31607      */
31608     cancelOnEsc : false,
31609     /**
31610      * @cfg {Boolean} updateEl True to update the innerHTML of the bound element when the update completes (defaults to false)
31611      */
31612     updateEl : false,
31613
31614     // private
31615     onRender : function(ct, position){
31616         this.el = new Roo.Layer({
31617             shadow: this.shadow,
31618             cls: "x-editor",
31619             parentEl : ct,
31620             shim : this.shim,
31621             shadowOffset:4,
31622             id: this.id,
31623             constrain: this.constrain
31624         });
31625         this.el.setStyle("overflow", Roo.isGecko ? "auto" : "hidden");
31626         if(this.field.msgTarget != 'title'){
31627             this.field.msgTarget = 'qtip';
31628         }
31629         this.field.render(this.el);
31630         if(Roo.isGecko){
31631             this.field.el.dom.setAttribute('autocomplete', 'off');
31632         }
31633         this.field.on("specialkey", this.onSpecialKey, this);
31634         if(this.swallowKeys){
31635             this.field.el.swallowEvent(['keydown','keypress']);
31636         }
31637         this.field.show();
31638         this.field.on("blur", this.onBlur, this);
31639         if(this.field.grow){
31640             this.field.on("autosize", this.el.sync,  this.el, {delay:1});
31641         }
31642     },
31643
31644     onSpecialKey : function(field, e)
31645     {
31646         //Roo.log('editor onSpecialKey');
31647         if(this.completeOnEnter && e.getKey() == e.ENTER){
31648             e.stopEvent();
31649             this.completeEdit();
31650             return;
31651         }
31652         // do not fire special key otherwise it might hide close the editor...
31653         if(e.getKey() == e.ENTER){    
31654             return;
31655         }
31656         if(this.cancelOnEsc && e.getKey() == e.ESC){
31657             this.cancelEdit();
31658             return;
31659         } 
31660         this.fireEvent('specialkey', field, e);
31661     
31662     },
31663
31664     /**
31665      * Starts the editing process and shows the editor.
31666      * @param {String/HTMLElement/Element} el The element to edit
31667      * @param {String} value (optional) A value to initialize the editor with. If a value is not provided, it defaults
31668       * to the innerHTML of el.
31669      */
31670     startEdit : function(el, value){
31671         if(this.editing){
31672             this.completeEdit();
31673         }
31674         this.boundEl = Roo.get(el);
31675         var v = value !== undefined ? value : this.boundEl.dom.innerHTML;
31676         if(!this.rendered){
31677             this.render(this.parentEl || document.body);
31678         }
31679         if(this.fireEvent("beforestartedit", this, this.boundEl, v) === false){
31680             return;
31681         }
31682         this.startValue = v;
31683         this.field.setValue(v);
31684         if(this.autoSize){
31685             var sz = this.boundEl.getSize();
31686             switch(this.autoSize){
31687                 case "width":
31688                 this.setSize(sz.width,  "");
31689                 break;
31690                 case "height":
31691                 this.setSize("",  sz.height);
31692                 break;
31693                 default:
31694                 this.setSize(sz.width,  sz.height);
31695             }
31696         }
31697         this.el.alignTo(this.boundEl, this.alignment);
31698         this.editing = true;
31699         if(Roo.QuickTips){
31700             Roo.QuickTips.disable();
31701         }
31702         this.show();
31703     },
31704
31705     /**
31706      * Sets the height and width of this editor.
31707      * @param {Number} width The new width
31708      * @param {Number} height The new height
31709      */
31710     setSize : function(w, h){
31711         this.field.setSize(w, h);
31712         if(this.el){
31713             this.el.sync();
31714         }
31715     },
31716
31717     /**
31718      * Realigns the editor to the bound field based on the current alignment config value.
31719      */
31720     realign : function(){
31721         this.el.alignTo(this.boundEl, this.alignment);
31722     },
31723
31724     /**
31725      * Ends the editing process, persists the changed value to the underlying field, and hides the editor.
31726      * @param {Boolean} remainVisible Override the default behavior and keep the editor visible after edit (defaults to false)
31727      */
31728     completeEdit : function(remainVisible){
31729         if(!this.editing){
31730             return;
31731         }
31732         var v = this.getValue();
31733         if(this.revertInvalid !== false && !this.field.isValid()){
31734             v = this.startValue;
31735             this.cancelEdit(true);
31736         }
31737         if(String(v) === String(this.startValue) && this.ignoreNoChange){
31738             this.editing = false;
31739             this.hide();
31740             return;
31741         }
31742         if(this.fireEvent("beforecomplete", this, v, this.startValue) !== false){
31743             this.editing = false;
31744             if(this.updateEl && this.boundEl){
31745                 this.boundEl.update(v);
31746             }
31747             if(remainVisible !== true){
31748                 this.hide();
31749             }
31750             this.fireEvent("complete", this, v, this.startValue);
31751         }
31752     },
31753
31754     // private
31755     onShow : function(){
31756         this.el.show();
31757         if(this.hideEl !== false){
31758             this.boundEl.hide();
31759         }
31760         this.field.show();
31761         if(Roo.isIE && !this.fixIEFocus){ // IE has problems with focusing the first time
31762             this.fixIEFocus = true;
31763             this.deferredFocus.defer(50, this);
31764         }else{
31765             this.field.focus();
31766         }
31767         this.fireEvent("startedit", this.boundEl, this.startValue);
31768     },
31769
31770     deferredFocus : function(){
31771         if(this.editing){
31772             this.field.focus();
31773         }
31774     },
31775
31776     /**
31777      * Cancels the editing process and hides the editor without persisting any changes.  The field value will be
31778      * reverted to the original starting value.
31779      * @param {Boolean} remainVisible Override the default behavior and keep the editor visible after
31780      * cancel (defaults to false)
31781      */
31782     cancelEdit : function(remainVisible){
31783         if(this.editing){
31784             this.setValue(this.startValue);
31785             if(remainVisible !== true){
31786                 this.hide();
31787             }
31788         }
31789     },
31790
31791     // private
31792     onBlur : function(){
31793         if(this.allowBlur !== true && this.editing){
31794             this.completeEdit();
31795         }
31796     },
31797
31798     // private
31799     onHide : function(){
31800         if(this.editing){
31801             this.completeEdit();
31802             return;
31803         }
31804         this.field.blur();
31805         if(this.field.collapse){
31806             this.field.collapse();
31807         }
31808         this.el.hide();
31809         if(this.hideEl !== false){
31810             this.boundEl.show();
31811         }
31812         if(Roo.QuickTips){
31813             Roo.QuickTips.enable();
31814         }
31815     },
31816
31817     /**
31818      * Sets the data value of the editor
31819      * @param {Mixed} value Any valid value supported by the underlying field
31820      */
31821     setValue : function(v){
31822         this.field.setValue(v);
31823     },
31824
31825     /**
31826      * Gets the data value of the editor
31827      * @return {Mixed} The data value
31828      */
31829     getValue : function(){
31830         return this.field.getValue();
31831     }
31832 });/*
31833  * Based on:
31834  * Ext JS Library 1.1.1
31835  * Copyright(c) 2006-2007, Ext JS, LLC.
31836  *
31837  * Originally Released Under LGPL - original licence link has changed is not relivant.
31838  *
31839  * Fork - LGPL
31840  * <script type="text/javascript">
31841  */
31842  
31843 /**
31844  * @class Roo.BasicDialog
31845  * @extends Roo.util.Observable
31846  * Lightweight Dialog Class.  The code below shows the creation of a typical dialog using existing HTML markup:
31847  * <pre><code>
31848 var dlg = new Roo.BasicDialog("my-dlg", {
31849     height: 200,
31850     width: 300,
31851     minHeight: 100,
31852     minWidth: 150,
31853     modal: true,
31854     proxyDrag: true,
31855     shadow: true
31856 });
31857 dlg.addKeyListener(27, dlg.hide, dlg); // ESC can also close the dialog
31858 dlg.addButton('OK', dlg.hide, dlg);    // Could call a save function instead of hiding
31859 dlg.addButton('Cancel', dlg.hide, dlg);
31860 dlg.show();
31861 </code></pre>
31862   <b>A Dialog should always be a direct child of the body element.</b>
31863  * @cfg {Boolean/DomHelper} autoCreate True to auto create from scratch, or using a DomHelper Object (defaults to false)
31864  * @cfg {String} title Default text to display in the title bar (defaults to null)
31865  * @cfg {Number} width Width of the dialog in pixels (can also be set via CSS).  Determined by browser if unspecified.
31866  * @cfg {Number} height Height of the dialog in pixels (can also be set via CSS).  Determined by browser if unspecified.
31867  * @cfg {Number} x The default left page coordinate of the dialog (defaults to center screen)
31868  * @cfg {Number} y The default top page coordinate of the dialog (defaults to center screen)
31869  * @cfg {String/Element} animateTarget Id or element from which the dialog should animate while opening
31870  * (defaults to null with no animation)
31871  * @cfg {Boolean} resizable False to disable manual dialog resizing (defaults to true)
31872  * @cfg {String} resizeHandles Which resize handles to display - see the {@link Roo.Resizable} handles config
31873  * property for valid values (defaults to 'all')
31874  * @cfg {Number} minHeight The minimum allowable height for a resizable dialog (defaults to 80)
31875  * @cfg {Number} minWidth The minimum allowable width for a resizable dialog (defaults to 200)
31876  * @cfg {Boolean} modal True to show the dialog modally, preventing user interaction with the rest of the page (defaults to false)
31877  * @cfg {Boolean} autoScroll True to allow the dialog body contents to overflow and display scrollbars (defaults to false)
31878  * @cfg {Boolean} closable False to remove the built-in top-right corner close button (defaults to true)
31879  * @cfg {Boolean} collapsible False to remove the built-in top-right corner collapse button (defaults to true)
31880  * @cfg {Boolean} constraintoviewport True to keep the dialog constrained within the visible viewport boundaries (defaults to true)
31881  * @cfg {Boolean} syncHeightBeforeShow True to cause the dimensions to be recalculated before the dialog is shown (defaults to false)
31882  * @cfg {Boolean} draggable False to disable dragging of the dialog within the viewport (defaults to true)
31883  * @cfg {Boolean} autoTabs If true, all elements with class 'x-dlg-tab' will get automatically converted to tabs (defaults to false)
31884  * @cfg {String} tabTag The tag name of tab elements, used when autoTabs = true (defaults to 'div')
31885  * @cfg {Boolean} proxyDrag True to drag a lightweight proxy element rather than the dialog itself, used when
31886  * draggable = true (defaults to false)
31887  * @cfg {Boolean} fixedcenter True to ensure that anytime the dialog is shown or resized it gets centered (defaults to false)
31888  * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
31889  * shadow (defaults to false)
31890  * @cfg {Number} shadowOffset The number of pixels to offset the shadow if displayed (defaults to 5)
31891  * @cfg {String} buttonAlign Valid values are "left," "center" and "right" (defaults to "right")
31892  * @cfg {Number} minButtonWidth Minimum width of all dialog buttons (defaults to 75)
31893  * @cfg {Array} buttons Array of buttons
31894  * @cfg {Boolean} shim True to create an iframe shim that prevents selects from showing through (defaults to false)
31895  * @constructor
31896  * Create a new BasicDialog.
31897  * @param {String/HTMLElement/Roo.Element} el The container element or DOM node, or its id
31898  * @param {Object} config Configuration options
31899  */
31900 Roo.BasicDialog = function(el, config){
31901     this.el = Roo.get(el);
31902     var dh = Roo.DomHelper;
31903     if(!this.el && config && config.autoCreate){
31904         if(typeof config.autoCreate == "object"){
31905             if(!config.autoCreate.id){
31906                 config.autoCreate.id = el;
31907             }
31908             this.el = dh.append(document.body,
31909                         config.autoCreate, true);
31910         }else{
31911             this.el = dh.append(document.body,
31912                         {tag: "div", id: el, style:'visibility:hidden;'}, true);
31913         }
31914     }
31915     el = this.el;
31916     el.setDisplayed(true);
31917     el.hide = this.hideAction;
31918     this.id = el.id;
31919     el.addClass("x-dlg");
31920
31921     Roo.apply(this, config);
31922
31923     this.proxy = el.createProxy("x-dlg-proxy");
31924     this.proxy.hide = this.hideAction;
31925     this.proxy.setOpacity(.5);
31926     this.proxy.hide();
31927
31928     if(config.width){
31929         el.setWidth(config.width);
31930     }
31931     if(config.height){
31932         el.setHeight(config.height);
31933     }
31934     this.size = el.getSize();
31935     if(typeof config.x != "undefined" && typeof config.y != "undefined"){
31936         this.xy = [config.x,config.y];
31937     }else{
31938         this.xy = el.getCenterXY(true);
31939     }
31940     /** The header element @type Roo.Element */
31941     this.header = el.child("> .x-dlg-hd");
31942     /** The body element @type Roo.Element */
31943     this.body = el.child("> .x-dlg-bd");
31944     /** The footer element @type Roo.Element */
31945     this.footer = el.child("> .x-dlg-ft");
31946
31947     if(!this.header){
31948         this.header = el.createChild({tag: "div", cls:"x-dlg-hd", html: "&#160;"}, this.body ? this.body.dom : null);
31949     }
31950     if(!this.body){
31951         this.body = el.createChild({tag: "div", cls:"x-dlg-bd"});
31952     }
31953
31954     this.header.unselectable();
31955     if(this.title){
31956         this.header.update(this.title);
31957     }
31958     // this element allows the dialog to be focused for keyboard event
31959     this.focusEl = el.createChild({tag: "a", href:"#", cls:"x-dlg-focus", tabIndex:"-1"});
31960     this.focusEl.swallowEvent("click", true);
31961
31962     this.header.wrap({cls:"x-dlg-hd-right"}).wrap({cls:"x-dlg-hd-left"}, true);
31963
31964     // wrap the body and footer for special rendering
31965     this.bwrap = this.body.wrap({tag: "div", cls:"x-dlg-dlg-body"});
31966     if(this.footer){
31967         this.bwrap.dom.appendChild(this.footer.dom);
31968     }
31969
31970     this.bg = this.el.createChild({
31971         tag: "div", cls:"x-dlg-bg",
31972         html: '<div class="x-dlg-bg-left"><div class="x-dlg-bg-right"><div class="x-dlg-bg-center">&#160;</div></div></div>'
31973     });
31974     this.centerBg = this.bg.child("div.x-dlg-bg-center");
31975
31976
31977     if(this.autoScroll !== false && !this.autoTabs){
31978         this.body.setStyle("overflow", "auto");
31979     }
31980
31981     this.toolbox = this.el.createChild({cls: "x-dlg-toolbox"});
31982
31983     if(this.closable !== false){
31984         this.el.addClass("x-dlg-closable");
31985         this.close = this.toolbox.createChild({cls:"x-dlg-close"});
31986         this.close.on("click", this.closeClick, this);
31987         this.close.addClassOnOver("x-dlg-close-over");
31988     }
31989     if(this.collapsible !== false){
31990         this.collapseBtn = this.toolbox.createChild({cls:"x-dlg-collapse"});
31991         this.collapseBtn.on("click", this.collapseClick, this);
31992         this.collapseBtn.addClassOnOver("x-dlg-collapse-over");
31993         this.header.on("dblclick", this.collapseClick, this);
31994     }
31995     if(this.resizable !== false){
31996         this.el.addClass("x-dlg-resizable");
31997         this.resizer = new Roo.Resizable(el, {
31998             minWidth: this.minWidth || 80,
31999             minHeight:this.minHeight || 80,
32000             handles: this.resizeHandles || "all",
32001             pinned: true
32002         });
32003         this.resizer.on("beforeresize", this.beforeResize, this);
32004         this.resizer.on("resize", this.onResize, this);
32005     }
32006     if(this.draggable !== false){
32007         el.addClass("x-dlg-draggable");
32008         if (!this.proxyDrag) {
32009             var dd = new Roo.dd.DD(el.dom.id, "WindowDrag");
32010         }
32011         else {
32012             var dd = new Roo.dd.DDProxy(el.dom.id, "WindowDrag", {dragElId: this.proxy.id});
32013         }
32014         dd.setHandleElId(this.header.id);
32015         dd.endDrag = this.endMove.createDelegate(this);
32016         dd.startDrag = this.startMove.createDelegate(this);
32017         dd.onDrag = this.onDrag.createDelegate(this);
32018         dd.scroll = false;
32019         this.dd = dd;
32020     }
32021     if(this.modal){
32022         this.mask = dh.append(document.body, {tag: "div", cls:"x-dlg-mask"}, true);
32023         this.mask.enableDisplayMode("block");
32024         this.mask.hide();
32025         this.el.addClass("x-dlg-modal");
32026     }
32027     if(this.shadow){
32028         this.shadow = new Roo.Shadow({
32029             mode : typeof this.shadow == "string" ? this.shadow : "sides",
32030             offset : this.shadowOffset
32031         });
32032     }else{
32033         this.shadowOffset = 0;
32034     }
32035     if(Roo.useShims && this.shim !== false){
32036         this.shim = this.el.createShim();
32037         this.shim.hide = this.hideAction;
32038         this.shim.hide();
32039     }else{
32040         this.shim = false;
32041     }
32042     if(this.autoTabs){
32043         this.initTabs();
32044     }
32045     if (this.buttons) { 
32046         var bts= this.buttons;
32047         this.buttons = [];
32048         Roo.each(bts, function(b) {
32049             this.addButton(b);
32050         }, this);
32051     }
32052     
32053     
32054     this.addEvents({
32055         /**
32056          * @event keydown
32057          * Fires when a key is pressed
32058          * @param {Roo.BasicDialog} this
32059          * @param {Roo.EventObject} e
32060          */
32061         "keydown" : true,
32062         /**
32063          * @event move
32064          * Fires when this dialog is moved by the user.
32065          * @param {Roo.BasicDialog} this
32066          * @param {Number} x The new page X
32067          * @param {Number} y The new page Y
32068          */
32069         "move" : true,
32070         /**
32071          * @event resize
32072          * Fires when this dialog is resized by the user.
32073          * @param {Roo.BasicDialog} this
32074          * @param {Number} width The new width
32075          * @param {Number} height The new height
32076          */
32077         "resize" : true,
32078         /**
32079          * @event beforehide
32080          * Fires before this dialog is hidden.
32081          * @param {Roo.BasicDialog} this
32082          */
32083         "beforehide" : true,
32084         /**
32085          * @event hide
32086          * Fires when this dialog is hidden.
32087          * @param {Roo.BasicDialog} this
32088          */
32089         "hide" : true,
32090         /**
32091          * @event beforeshow
32092          * Fires before this dialog is shown.
32093          * @param {Roo.BasicDialog} this
32094          */
32095         "beforeshow" : true,
32096         /**
32097          * @event show
32098          * Fires when this dialog is shown.
32099          * @param {Roo.BasicDialog} this
32100          */
32101         "show" : true
32102     });
32103     el.on("keydown", this.onKeyDown, this);
32104     el.on("mousedown", this.toFront, this);
32105     Roo.EventManager.onWindowResize(this.adjustViewport, this, true);
32106     this.el.hide();
32107     Roo.DialogManager.register(this);
32108     Roo.BasicDialog.superclass.constructor.call(this);
32109 };
32110
32111 Roo.extend(Roo.BasicDialog, Roo.util.Observable, {
32112     shadowOffset: Roo.isIE ? 6 : 5,
32113     minHeight: 80,
32114     minWidth: 200,
32115     minButtonWidth: 75,
32116     defaultButton: null,
32117     buttonAlign: "right",
32118     tabTag: 'div',
32119     firstShow: true,
32120
32121     /**
32122      * Sets the dialog title text
32123      * @param {String} text The title text to display
32124      * @return {Roo.BasicDialog} this
32125      */
32126     setTitle : function(text){
32127         this.header.update(text);
32128         return this;
32129     },
32130
32131     // private
32132     closeClick : function(){
32133         this.hide();
32134     },
32135
32136     // private
32137     collapseClick : function(){
32138         this[this.collapsed ? "expand" : "collapse"]();
32139     },
32140
32141     /**
32142      * Collapses the dialog to its minimized state (only the title bar is visible).
32143      * Equivalent to the user clicking the collapse dialog button.
32144      */
32145     collapse : function(){
32146         if(!this.collapsed){
32147             this.collapsed = true;
32148             this.el.addClass("x-dlg-collapsed");
32149             this.restoreHeight = this.el.getHeight();
32150             this.resizeTo(this.el.getWidth(), this.header.getHeight());
32151         }
32152     },
32153
32154     /**
32155      * Expands a collapsed dialog back to its normal state.  Equivalent to the user
32156      * clicking the expand dialog button.
32157      */
32158     expand : function(){
32159         if(this.collapsed){
32160             this.collapsed = false;
32161             this.el.removeClass("x-dlg-collapsed");
32162             this.resizeTo(this.el.getWidth(), this.restoreHeight);
32163         }
32164     },
32165
32166     /**
32167      * Reinitializes the tabs component, clearing out old tabs and finding new ones.
32168      * @return {Roo.TabPanel} The tabs component
32169      */
32170     initTabs : function(){
32171         var tabs = this.getTabs();
32172         while(tabs.getTab(0)){
32173             tabs.removeTab(0);
32174         }
32175         this.el.select(this.tabTag+'.x-dlg-tab').each(function(el){
32176             var dom = el.dom;
32177             tabs.addTab(Roo.id(dom), dom.title);
32178             dom.title = "";
32179         });
32180         tabs.activate(0);
32181         return tabs;
32182     },
32183
32184     // private
32185     beforeResize : function(){
32186         this.resizer.minHeight = Math.max(this.minHeight, this.getHeaderFooterHeight(true)+40);
32187     },
32188
32189     // private
32190     onResize : function(){
32191         this.refreshSize();
32192         this.syncBodyHeight();
32193         this.adjustAssets();
32194         this.focus();
32195         this.fireEvent("resize", this, this.size.width, this.size.height);
32196     },
32197
32198     // private
32199     onKeyDown : function(e){
32200         if(this.isVisible()){
32201             this.fireEvent("keydown", this, e);
32202         }
32203     },
32204
32205     /**
32206      * Resizes the dialog.
32207      * @param {Number} width
32208      * @param {Number} height
32209      * @return {Roo.BasicDialog} this
32210      */
32211     resizeTo : function(width, height){
32212         this.el.setSize(width, height);
32213         this.size = {width: width, height: height};
32214         this.syncBodyHeight();
32215         if(this.fixedcenter){
32216             this.center();
32217         }
32218         if(this.isVisible()){
32219             this.constrainXY();
32220             this.adjustAssets();
32221         }
32222         this.fireEvent("resize", this, width, height);
32223         return this;
32224     },
32225
32226
32227     /**
32228      * Resizes the dialog to fit the specified content size.
32229      * @param {Number} width
32230      * @param {Number} height
32231      * @return {Roo.BasicDialog} this
32232      */
32233     setContentSize : function(w, h){
32234         h += this.getHeaderFooterHeight() + this.body.getMargins("tb");
32235         w += this.body.getMargins("lr") + this.bwrap.getMargins("lr") + this.centerBg.getPadding("lr");
32236         //if(!this.el.isBorderBox()){
32237             h +=  this.body.getPadding("tb") + this.bwrap.getBorderWidth("tb") + this.body.getBorderWidth("tb") + this.el.getBorderWidth("tb");
32238             w += this.body.getPadding("lr") + this.bwrap.getBorderWidth("lr") + this.body.getBorderWidth("lr") + this.bwrap.getPadding("lr") + this.el.getBorderWidth("lr");
32239         //}
32240         if(this.tabs){
32241             h += this.tabs.stripWrap.getHeight() + this.tabs.bodyEl.getMargins("tb") + this.tabs.bodyEl.getPadding("tb");
32242             w += this.tabs.bodyEl.getMargins("lr") + this.tabs.bodyEl.getPadding("lr");
32243         }
32244         this.resizeTo(w, h);
32245         return this;
32246     },
32247
32248     /**
32249      * Adds a key listener for when this dialog is displayed.  This allows you to hook in a function that will be
32250      * executed in response to a particular key being pressed while the dialog is active.
32251      * @param {Number/Array/Object} key Either the numeric key code, array of key codes or an object with the following options:
32252      *                                  {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
32253      * @param {Function} fn The function to call
32254      * @param {Object} scope (optional) The scope of the function
32255      * @return {Roo.BasicDialog} this
32256      */
32257     addKeyListener : function(key, fn, scope){
32258         var keyCode, shift, ctrl, alt;
32259         if(typeof key == "object" && !(key instanceof Array)){
32260             keyCode = key["key"];
32261             shift = key["shift"];
32262             ctrl = key["ctrl"];
32263             alt = key["alt"];
32264         }else{
32265             keyCode = key;
32266         }
32267         var handler = function(dlg, e){
32268             if((!shift || e.shiftKey) && (!ctrl || e.ctrlKey) &&  (!alt || e.altKey)){
32269                 var k = e.getKey();
32270                 if(keyCode instanceof Array){
32271                     for(var i = 0, len = keyCode.length; i < len; i++){
32272                         if(keyCode[i] == k){
32273                           fn.call(scope || window, dlg, k, e);
32274                           return;
32275                         }
32276                     }
32277                 }else{
32278                     if(k == keyCode){
32279                         fn.call(scope || window, dlg, k, e);
32280                     }
32281                 }
32282             }
32283         };
32284         this.on("keydown", handler);
32285         return this;
32286     },
32287
32288     /**
32289      * Returns the TabPanel component (creates it if it doesn't exist).
32290      * Note: If you wish to simply check for the existence of tabs without creating them,
32291      * check for a null 'tabs' property.
32292      * @return {Roo.TabPanel} The tabs component
32293      */
32294     getTabs : function(){
32295         if(!this.tabs){
32296             this.el.addClass("x-dlg-auto-tabs");
32297             this.body.addClass(this.tabPosition == "bottom" ? "x-tabs-bottom" : "x-tabs-top");
32298             this.tabs = new Roo.TabPanel(this.body.dom, this.tabPosition == "bottom");
32299         }
32300         return this.tabs;
32301     },
32302
32303     /**
32304      * Adds a button to the footer section of the dialog.
32305      * @param {String/Object} config A string becomes the button text, an object can either be a Button config
32306      * object or a valid Roo.DomHelper element config
32307      * @param {Function} handler The function called when the button is clicked
32308      * @param {Object} scope (optional) The scope of the handler function (accepts position as a property)
32309      * @return {Roo.Button} The new button
32310      */
32311     addButton : function(config, handler, scope){
32312         var dh = Roo.DomHelper;
32313         if(!this.footer){
32314             this.footer = dh.append(this.bwrap, {tag: "div", cls:"x-dlg-ft"}, true);
32315         }
32316         if(!this.btnContainer){
32317             var tb = this.footer.createChild({
32318
32319                 cls:"x-dlg-btns x-dlg-btns-"+this.buttonAlign,
32320                 html:'<table cellspacing="0"><tbody><tr></tr></tbody></table><div class="x-clear"></div>'
32321             }, null, true);
32322             this.btnContainer = tb.firstChild.firstChild.firstChild;
32323         }
32324         var bconfig = {
32325             handler: handler,
32326             scope: scope,
32327             minWidth: this.minButtonWidth,
32328             hideParent:true
32329         };
32330         if(typeof config == "string"){
32331             bconfig.text = config;
32332         }else{
32333             if(config.tag){
32334                 bconfig.dhconfig = config;
32335             }else{
32336                 Roo.apply(bconfig, config);
32337             }
32338         }
32339         var fc = false;
32340         if ((typeof(bconfig.position) != 'undefined') && bconfig.position < this.btnContainer.childNodes.length-1) {
32341             bconfig.position = Math.max(0, bconfig.position);
32342             fc = this.btnContainer.childNodes[bconfig.position];
32343         }
32344          
32345         var btn = new Roo.Button(
32346             fc ? 
32347                 this.btnContainer.insertBefore(document.createElement("td"),fc)
32348                 : this.btnContainer.appendChild(document.createElement("td")),
32349             //Roo.get(this.btnContainer).createChild( { tag: 'td'},  fc ),
32350             bconfig
32351         );
32352         this.syncBodyHeight();
32353         if(!this.buttons){
32354             /**
32355              * Array of all the buttons that have been added to this dialog via addButton
32356              * @type Array
32357              */
32358             this.buttons = [];
32359         }
32360         this.buttons.push(btn);
32361         return btn;
32362     },
32363
32364     /**
32365      * Sets the default button to be focused when the dialog is displayed.
32366      * @param {Roo.BasicDialog.Button} btn The button object returned by {@link #addButton}
32367      * @return {Roo.BasicDialog} this
32368      */
32369     setDefaultButton : function(btn){
32370         this.defaultButton = btn;
32371         return this;
32372     },
32373
32374     // private
32375     getHeaderFooterHeight : function(safe){
32376         var height = 0;
32377         if(this.header){
32378            height += this.header.getHeight();
32379         }
32380         if(this.footer){
32381            var fm = this.footer.getMargins();
32382             height += (this.footer.getHeight()+fm.top+fm.bottom);
32383         }
32384         height += this.bwrap.getPadding("tb")+this.bwrap.getBorderWidth("tb");
32385         height += this.centerBg.getPadding("tb");
32386         return height;
32387     },
32388
32389     // private
32390     syncBodyHeight : function()
32391     {
32392         var bd = this.body, // the text
32393             cb = this.centerBg, // wrapper around bottom.. but does not seem to be used..
32394             bw = this.bwrap;
32395         var height = this.size.height - this.getHeaderFooterHeight(false);
32396         bd.setHeight(height-bd.getMargins("tb"));
32397         var hh = this.header.getHeight();
32398         var h = this.size.height-hh;
32399         cb.setHeight(h);
32400         
32401         bw.setLeftTop(cb.getPadding("l"), hh+cb.getPadding("t"));
32402         bw.setHeight(h-cb.getPadding("tb"));
32403         
32404         bw.setWidth(this.el.getWidth(true)-cb.getPadding("lr"));
32405         bd.setWidth(bw.getWidth(true));
32406         if(this.tabs){
32407             this.tabs.syncHeight();
32408             if(Roo.isIE){
32409                 this.tabs.el.repaint();
32410             }
32411         }
32412     },
32413
32414     /**
32415      * Restores the previous state of the dialog if Roo.state is configured.
32416      * @return {Roo.BasicDialog} this
32417      */
32418     restoreState : function(){
32419         var box = Roo.state.Manager.get(this.stateId || (this.el.id + "-state"));
32420         if(box && box.width){
32421             this.xy = [box.x, box.y];
32422             this.resizeTo(box.width, box.height);
32423         }
32424         return this;
32425     },
32426
32427     // private
32428     beforeShow : function(){
32429         this.expand();
32430         if(this.fixedcenter){
32431             this.xy = this.el.getCenterXY(true);
32432         }
32433         if(this.modal){
32434             Roo.get(document.body).addClass("x-body-masked");
32435             this.mask.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
32436             this.mask.show();
32437         }
32438         this.constrainXY();
32439     },
32440
32441     // private
32442     animShow : function(){
32443         var b = Roo.get(this.animateTarget).getBox();
32444         this.proxy.setSize(b.width, b.height);
32445         this.proxy.setLocation(b.x, b.y);
32446         this.proxy.show();
32447         this.proxy.setBounds(this.xy[0], this.xy[1], this.size.width, this.size.height,
32448                     true, .35, this.showEl.createDelegate(this));
32449     },
32450
32451     /**
32452      * Shows the dialog.
32453      * @param {String/HTMLElement/Roo.Element} animateTarget (optional) Reset the animation target
32454      * @return {Roo.BasicDialog} this
32455      */
32456     show : function(animateTarget){
32457         if (this.fireEvent("beforeshow", this) === false){
32458             return;
32459         }
32460         if(this.syncHeightBeforeShow){
32461             this.syncBodyHeight();
32462         }else if(this.firstShow){
32463             this.firstShow = false;
32464             this.syncBodyHeight(); // sync the height on the first show instead of in the constructor
32465         }
32466         this.animateTarget = animateTarget || this.animateTarget;
32467         if(!this.el.isVisible()){
32468             this.beforeShow();
32469             if(this.animateTarget && Roo.get(this.animateTarget)){
32470                 this.animShow();
32471             }else{
32472                 this.showEl();
32473             }
32474         }
32475         return this;
32476     },
32477
32478     // private
32479     showEl : function(){
32480         this.proxy.hide();
32481         this.el.setXY(this.xy);
32482         this.el.show();
32483         this.adjustAssets(true);
32484         this.toFront();
32485         this.focus();
32486         // IE peekaboo bug - fix found by Dave Fenwick
32487         if(Roo.isIE){
32488             this.el.repaint();
32489         }
32490         this.fireEvent("show", this);
32491     },
32492
32493     /**
32494      * Focuses the dialog.  If a defaultButton is set, it will receive focus, otherwise the
32495      * dialog itself will receive focus.
32496      */
32497     focus : function(){
32498         if(this.defaultButton){
32499             this.defaultButton.focus();
32500         }else{
32501             this.focusEl.focus();
32502         }
32503     },
32504
32505     // private
32506     constrainXY : function(){
32507         if(this.constraintoviewport !== false){
32508             if(!this.viewSize){
32509                 if(this.container){
32510                     var s = this.container.getSize();
32511                     this.viewSize = [s.width, s.height];
32512                 }else{
32513                     this.viewSize = [Roo.lib.Dom.getViewWidth(),Roo.lib.Dom.getViewHeight()];
32514                 }
32515             }
32516             var s = Roo.get(this.container||document).getScroll();
32517
32518             var x = this.xy[0], y = this.xy[1];
32519             var w = this.size.width, h = this.size.height;
32520             var vw = this.viewSize[0], vh = this.viewSize[1];
32521             // only move it if it needs it
32522             var moved = false;
32523             // first validate right/bottom
32524             if(x + w > vw+s.left){
32525                 x = vw - w;
32526                 moved = true;
32527             }
32528             if(y + h > vh+s.top){
32529                 y = vh - h;
32530                 moved = true;
32531             }
32532             // then make sure top/left isn't negative
32533             if(x < s.left){
32534                 x = s.left;
32535                 moved = true;
32536             }
32537             if(y < s.top){
32538                 y = s.top;
32539                 moved = true;
32540             }
32541             if(moved){
32542                 // cache xy
32543                 this.xy = [x, y];
32544                 if(this.isVisible()){
32545                     this.el.setLocation(x, y);
32546                     this.adjustAssets();
32547                 }
32548             }
32549         }
32550     },
32551
32552     // private
32553     onDrag : function(){
32554         if(!this.proxyDrag){
32555             this.xy = this.el.getXY();
32556             this.adjustAssets();
32557         }
32558     },
32559
32560     // private
32561     adjustAssets : function(doShow){
32562         var x = this.xy[0], y = this.xy[1];
32563         var w = this.size.width, h = this.size.height;
32564         if(doShow === true){
32565             if(this.shadow){
32566                 this.shadow.show(this.el);
32567             }
32568             if(this.shim){
32569                 this.shim.show();
32570             }
32571         }
32572         if(this.shadow && this.shadow.isVisible()){
32573             this.shadow.show(this.el);
32574         }
32575         if(this.shim && this.shim.isVisible()){
32576             this.shim.setBounds(x, y, w, h);
32577         }
32578     },
32579
32580     // private
32581     adjustViewport : function(w, h){
32582         if(!w || !h){
32583             w = Roo.lib.Dom.getViewWidth();
32584             h = Roo.lib.Dom.getViewHeight();
32585         }
32586         // cache the size
32587         this.viewSize = [w, h];
32588         if(this.modal && this.mask.isVisible()){
32589             this.mask.setSize(w, h); // first make sure the mask isn't causing overflow
32590             this.mask.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
32591         }
32592         if(this.isVisible()){
32593             this.constrainXY();
32594         }
32595     },
32596
32597     /**
32598      * Destroys this dialog and all its supporting elements (including any tabs, shim,
32599      * shadow, proxy, mask, etc.)  Also removes all event listeners.
32600      * @param {Boolean} removeEl (optional) true to remove the element from the DOM
32601      */
32602     destroy : function(removeEl){
32603         if(this.isVisible()){
32604             this.animateTarget = null;
32605             this.hide();
32606         }
32607         Roo.EventManager.removeResizeListener(this.adjustViewport, this);
32608         if(this.tabs){
32609             this.tabs.destroy(removeEl);
32610         }
32611         Roo.destroy(
32612              this.shim,
32613              this.proxy,
32614              this.resizer,
32615              this.close,
32616              this.mask
32617         );
32618         if(this.dd){
32619             this.dd.unreg();
32620         }
32621         if(this.buttons){
32622            for(var i = 0, len = this.buttons.length; i < len; i++){
32623                this.buttons[i].destroy();
32624            }
32625         }
32626         this.el.removeAllListeners();
32627         if(removeEl === true){
32628             this.el.update("");
32629             this.el.remove();
32630         }
32631         Roo.DialogManager.unregister(this);
32632     },
32633
32634     // private
32635     startMove : function(){
32636         if(this.proxyDrag){
32637             this.proxy.show();
32638         }
32639         if(this.constraintoviewport !== false){
32640             this.dd.constrainTo(document.body, {right: this.shadowOffset, bottom: this.shadowOffset});
32641         }
32642     },
32643
32644     // private
32645     endMove : function(){
32646         if(!this.proxyDrag){
32647             Roo.dd.DD.prototype.endDrag.apply(this.dd, arguments);
32648         }else{
32649             Roo.dd.DDProxy.prototype.endDrag.apply(this.dd, arguments);
32650             this.proxy.hide();
32651         }
32652         this.refreshSize();
32653         this.adjustAssets();
32654         this.focus();
32655         this.fireEvent("move", this, this.xy[0], this.xy[1]);
32656     },
32657
32658     /**
32659      * Brings this dialog to the front of any other visible dialogs
32660      * @return {Roo.BasicDialog} this
32661      */
32662     toFront : function(){
32663         Roo.DialogManager.bringToFront(this);
32664         return this;
32665     },
32666
32667     /**
32668      * Sends this dialog to the back (under) of any other visible dialogs
32669      * @return {Roo.BasicDialog} this
32670      */
32671     toBack : function(){
32672         Roo.DialogManager.sendToBack(this);
32673         return this;
32674     },
32675
32676     /**
32677      * Centers this dialog in the viewport
32678      * @return {Roo.BasicDialog} this
32679      */
32680     center : function(){
32681         var xy = this.el.getCenterXY(true);
32682         this.moveTo(xy[0], xy[1]);
32683         return this;
32684     },
32685
32686     /**
32687      * Moves the dialog's top-left corner to the specified point
32688      * @param {Number} x
32689      * @param {Number} y
32690      * @return {Roo.BasicDialog} this
32691      */
32692     moveTo : function(x, y){
32693         this.xy = [x,y];
32694         if(this.isVisible()){
32695             this.el.setXY(this.xy);
32696             this.adjustAssets();
32697         }
32698         return this;
32699     },
32700
32701     /**
32702      * Aligns the dialog to the specified element
32703      * @param {String/HTMLElement/Roo.Element} element The element to align to.
32704      * @param {String} position The position to align to (see {@link Roo.Element#alignTo} for more details).
32705      * @param {Array} offsets (optional) Offset the positioning by [x, y]
32706      * @return {Roo.BasicDialog} this
32707      */
32708     alignTo : function(element, position, offsets){
32709         this.xy = this.el.getAlignToXY(element, position, offsets);
32710         if(this.isVisible()){
32711             this.el.setXY(this.xy);
32712             this.adjustAssets();
32713         }
32714         return this;
32715     },
32716
32717     /**
32718      * Anchors an element to another element and realigns it when the window is resized.
32719      * @param {String/HTMLElement/Roo.Element} element The element to align to.
32720      * @param {String} position The position to align to (see {@link Roo.Element#alignTo} for more details)
32721      * @param {Array} offsets (optional) Offset the positioning by [x, y]
32722      * @param {Boolean/Number} monitorScroll (optional) true to monitor body scroll and reposition. If this parameter
32723      * is a number, it is used as the buffer delay (defaults to 50ms).
32724      * @return {Roo.BasicDialog} this
32725      */
32726     anchorTo : function(el, alignment, offsets, monitorScroll){
32727         var action = function(){
32728             this.alignTo(el, alignment, offsets);
32729         };
32730         Roo.EventManager.onWindowResize(action, this);
32731         var tm = typeof monitorScroll;
32732         if(tm != 'undefined'){
32733             Roo.EventManager.on(window, 'scroll', action, this,
32734                 {buffer: tm == 'number' ? monitorScroll : 50});
32735         }
32736         action.call(this);
32737         return this;
32738     },
32739
32740     /**
32741      * Returns true if the dialog is visible
32742      * @return {Boolean}
32743      */
32744     isVisible : function(){
32745         return this.el.isVisible();
32746     },
32747
32748     // private
32749     animHide : function(callback){
32750         var b = Roo.get(this.animateTarget).getBox();
32751         this.proxy.show();
32752         this.proxy.setBounds(this.xy[0], this.xy[1], this.size.width, this.size.height);
32753         this.el.hide();
32754         this.proxy.setBounds(b.x, b.y, b.width, b.height, true, .35,
32755                     this.hideEl.createDelegate(this, [callback]));
32756     },
32757
32758     /**
32759      * Hides the dialog.
32760      * @param {Function} callback (optional) Function to call when the dialog is hidden
32761      * @return {Roo.BasicDialog} this
32762      */
32763     hide : function(callback){
32764         if (this.fireEvent("beforehide", this) === false){
32765             return;
32766         }
32767         if(this.shadow){
32768             this.shadow.hide();
32769         }
32770         if(this.shim) {
32771           this.shim.hide();
32772         }
32773         // sometimes animateTarget seems to get set.. causing problems...
32774         // this just double checks..
32775         if(this.animateTarget && Roo.get(this.animateTarget)) {
32776            this.animHide(callback);
32777         }else{
32778             this.el.hide();
32779             this.hideEl(callback);
32780         }
32781         return this;
32782     },
32783
32784     // private
32785     hideEl : function(callback){
32786         this.proxy.hide();
32787         if(this.modal){
32788             this.mask.hide();
32789             Roo.get(document.body).removeClass("x-body-masked");
32790         }
32791         this.fireEvent("hide", this);
32792         if(typeof callback == "function"){
32793             callback();
32794         }
32795     },
32796
32797     // private
32798     hideAction : function(){
32799         this.setLeft("-10000px");
32800         this.setTop("-10000px");
32801         this.setStyle("visibility", "hidden");
32802     },
32803
32804     // private
32805     refreshSize : function(){
32806         this.size = this.el.getSize();
32807         this.xy = this.el.getXY();
32808         Roo.state.Manager.set(this.stateId || this.el.id + "-state", this.el.getBox());
32809     },
32810
32811     // private
32812     // z-index is managed by the DialogManager and may be overwritten at any time
32813     setZIndex : function(index){
32814         if(this.modal){
32815             this.mask.setStyle("z-index", index);
32816         }
32817         if(this.shim){
32818             this.shim.setStyle("z-index", ++index);
32819         }
32820         if(this.shadow){
32821             this.shadow.setZIndex(++index);
32822         }
32823         this.el.setStyle("z-index", ++index);
32824         if(this.proxy){
32825             this.proxy.setStyle("z-index", ++index);
32826         }
32827         if(this.resizer){
32828             this.resizer.proxy.setStyle("z-index", ++index);
32829         }
32830
32831         this.lastZIndex = index;
32832     },
32833
32834     /**
32835      * Returns the element for this dialog
32836      * @return {Roo.Element} The underlying dialog Element
32837      */
32838     getEl : function(){
32839         return this.el;
32840     }
32841 });
32842
32843 /**
32844  * @class Roo.DialogManager
32845  * Provides global access to BasicDialogs that have been created and
32846  * support for z-indexing (layering) multiple open dialogs.
32847  */
32848 Roo.DialogManager = function(){
32849     var list = {};
32850     var accessList = [];
32851     var front = null;
32852
32853     // private
32854     var sortDialogs = function(d1, d2){
32855         return (!d1._lastAccess || d1._lastAccess < d2._lastAccess) ? -1 : 1;
32856     };
32857
32858     // private
32859     var orderDialogs = function(){
32860         accessList.sort(sortDialogs);
32861         var seed = Roo.DialogManager.zseed;
32862         for(var i = 0, len = accessList.length; i < len; i++){
32863             var dlg = accessList[i];
32864             if(dlg){
32865                 dlg.setZIndex(seed + (i*10));
32866             }
32867         }
32868     };
32869
32870     return {
32871         /**
32872          * The starting z-index for BasicDialogs (defaults to 9000)
32873          * @type Number The z-index value
32874          */
32875         zseed : 9000,
32876
32877         // private
32878         register : function(dlg){
32879             list[dlg.id] = dlg;
32880             accessList.push(dlg);
32881         },
32882
32883         // private
32884         unregister : function(dlg){
32885             delete list[dlg.id];
32886             var i=0;
32887             var len=0;
32888             if(!accessList.indexOf){
32889                 for(  i = 0, len = accessList.length; i < len; i++){
32890                     if(accessList[i] == dlg){
32891                         accessList.splice(i, 1);
32892                         return;
32893                     }
32894                 }
32895             }else{
32896                  i = accessList.indexOf(dlg);
32897                 if(i != -1){
32898                     accessList.splice(i, 1);
32899                 }
32900             }
32901         },
32902
32903         /**
32904          * Gets a registered dialog by id
32905          * @param {String/Object} id The id of the dialog or a dialog
32906          * @return {Roo.BasicDialog} this
32907          */
32908         get : function(id){
32909             return typeof id == "object" ? id : list[id];
32910         },
32911
32912         /**
32913          * Brings the specified dialog to the front
32914          * @param {String/Object} dlg The id of the dialog or a dialog
32915          * @return {Roo.BasicDialog} this
32916          */
32917         bringToFront : function(dlg){
32918             dlg = this.get(dlg);
32919             if(dlg != front){
32920                 front = dlg;
32921                 dlg._lastAccess = new Date().getTime();
32922                 orderDialogs();
32923             }
32924             return dlg;
32925         },
32926
32927         /**
32928          * Sends the specified dialog to the back
32929          * @param {String/Object} dlg The id of the dialog or a dialog
32930          * @return {Roo.BasicDialog} this
32931          */
32932         sendToBack : function(dlg){
32933             dlg = this.get(dlg);
32934             dlg._lastAccess = -(new Date().getTime());
32935             orderDialogs();
32936             return dlg;
32937         },
32938
32939         /**
32940          * Hides all dialogs
32941          */
32942         hideAll : function(){
32943             for(var id in list){
32944                 if(list[id] && typeof list[id] != "function" && list[id].isVisible()){
32945                     list[id].hide();
32946                 }
32947             }
32948         }
32949     };
32950 }();
32951
32952 /**
32953  * @class Roo.LayoutDialog
32954  * @extends Roo.BasicDialog
32955  * Dialog which provides adjustments for working with a layout in a Dialog.
32956  * Add your necessary layout config options to the dialog's config.<br>
32957  * Example usage (including a nested layout):
32958  * <pre><code>
32959 if(!dialog){
32960     dialog = new Roo.LayoutDialog("download-dlg", {
32961         modal: true,
32962         width:600,
32963         height:450,
32964         shadow:true,
32965         minWidth:500,
32966         minHeight:350,
32967         autoTabs:true,
32968         proxyDrag:true,
32969         // layout config merges with the dialog config
32970         center:{
32971             tabPosition: "top",
32972             alwaysShowTabs: true
32973         }
32974     });
32975     dialog.addKeyListener(27, dialog.hide, dialog);
32976     dialog.setDefaultButton(dialog.addButton("Close", dialog.hide, dialog));
32977     dialog.addButton("Build It!", this.getDownload, this);
32978
32979     // we can even add nested layouts
32980     var innerLayout = new Roo.BorderLayout("dl-inner", {
32981         east: {
32982             initialSize: 200,
32983             autoScroll:true,
32984             split:true
32985         },
32986         center: {
32987             autoScroll:true
32988         }
32989     });
32990     innerLayout.beginUpdate();
32991     innerLayout.add("east", new Roo.ContentPanel("dl-details"));
32992     innerLayout.add("center", new Roo.ContentPanel("selection-panel"));
32993     innerLayout.endUpdate(true);
32994
32995     var layout = dialog.getLayout();
32996     layout.beginUpdate();
32997     layout.add("center", new Roo.ContentPanel("standard-panel",
32998                         {title: "Download the Source", fitToFrame:true}));
32999     layout.add("center", new Roo.NestedLayoutPanel(innerLayout,
33000                {title: "Build your own roo.js"}));
33001     layout.getRegion("center").showPanel(sp);
33002     layout.endUpdate();
33003 }
33004 </code></pre>
33005     * @constructor
33006     * @param {String/HTMLElement/Roo.Element} el The id of or container element, or config
33007     * @param {Object} config configuration options
33008   */
33009 Roo.LayoutDialog = function(el, cfg){
33010     
33011     var config=  cfg;
33012     if (typeof(cfg) == 'undefined') {
33013         config = Roo.apply({}, el);
33014         // not sure why we use documentElement here.. - it should always be body.
33015         // IE7 borks horribly if we use documentElement.
33016         // webkit also does not like documentElement - it creates a body element...
33017         el = Roo.get( document.body || document.documentElement ).createChild();
33018         //config.autoCreate = true;
33019     }
33020     
33021     
33022     config.autoTabs = false;
33023     Roo.LayoutDialog.superclass.constructor.call(this, el, config);
33024     this.body.setStyle({overflow:"hidden", position:"relative"});
33025     this.layout = new Roo.BorderLayout(this.body.dom, config);
33026     this.layout.monitorWindowResize = false;
33027     this.el.addClass("x-dlg-auto-layout");
33028     // fix case when center region overwrites center function
33029     this.center = Roo.BasicDialog.prototype.center;
33030     this.on("show", this.layout.layout, this.layout, true);
33031     if (config.items) {
33032         var xitems = config.items;
33033         delete config.items;
33034         Roo.each(xitems, this.addxtype, this);
33035     }
33036     
33037     
33038 };
33039 Roo.extend(Roo.LayoutDialog, Roo.BasicDialog, {
33040     /**
33041      * Ends update of the layout <strike>and resets display to none</strike>. Use standard beginUpdate/endUpdate on the layout.
33042      * @deprecated
33043      */
33044     endUpdate : function(){
33045         this.layout.endUpdate();
33046     },
33047
33048     /**
33049      * Begins an update of the layout <strike>and sets display to block and visibility to hidden</strike>. Use standard beginUpdate/endUpdate on the layout.
33050      *  @deprecated
33051      */
33052     beginUpdate : function(){
33053         this.layout.beginUpdate();
33054     },
33055
33056     /**
33057      * Get the BorderLayout for this dialog
33058      * @return {Roo.BorderLayout}
33059      */
33060     getLayout : function(){
33061         return this.layout;
33062     },
33063
33064     showEl : function(){
33065         Roo.LayoutDialog.superclass.showEl.apply(this, arguments);
33066         if(Roo.isIE7){
33067             this.layout.layout();
33068         }
33069     },
33070
33071     // private
33072     // Use the syncHeightBeforeShow config option to control this automatically
33073     syncBodyHeight : function(){
33074         Roo.LayoutDialog.superclass.syncBodyHeight.call(this);
33075         if(this.layout){this.layout.layout();}
33076     },
33077     
33078       /**
33079      * Add an xtype element (actually adds to the layout.)
33080      * @return {Object} xdata xtype object data.
33081      */
33082     
33083     addxtype : function(c) {
33084         return this.layout.addxtype(c);
33085     }
33086 });/*
33087  * Based on:
33088  * Ext JS Library 1.1.1
33089  * Copyright(c) 2006-2007, Ext JS, LLC.
33090  *
33091  * Originally Released Under LGPL - original licence link has changed is not relivant.
33092  *
33093  * Fork - LGPL
33094  * <script type="text/javascript">
33095  */
33096  
33097 /**
33098  * @class Roo.MessageBox
33099  * Utility class for generating different styles of message boxes.  The alias Roo.Msg can also be used.
33100  * Example usage:
33101  *<pre><code>
33102 // Basic alert:
33103 Roo.Msg.alert('Status', 'Changes saved successfully.');
33104
33105 // Prompt for user data:
33106 Roo.Msg.prompt('Name', 'Please enter your name:', function(btn, text){
33107     if (btn == 'ok'){
33108         // process text value...
33109     }
33110 });
33111
33112 // Show a dialog using config options:
33113 Roo.Msg.show({
33114    title:'Save Changes?',
33115    msg: 'Your are closing a tab that has unsaved changes. Would you like to save your changes?',
33116    buttons: Roo.Msg.YESNOCANCEL,
33117    fn: processResult,
33118    animEl: 'elId'
33119 });
33120 </code></pre>
33121  * @singleton
33122  */
33123 Roo.MessageBox = function(){
33124     var dlg, opt, mask, waitTimer;
33125     var bodyEl, msgEl, textboxEl, textareaEl, progressEl, pp;
33126     var buttons, activeTextEl, bwidth;
33127
33128     // private
33129     var handleButton = function(button){
33130         dlg.hide();
33131         Roo.callback(opt.fn, opt.scope||window, [button, activeTextEl.dom.value], 1);
33132     };
33133
33134     // private
33135     var handleHide = function(){
33136         if(opt && opt.cls){
33137             dlg.el.removeClass(opt.cls);
33138         }
33139         if(waitTimer){
33140             Roo.TaskMgr.stop(waitTimer);
33141             waitTimer = null;
33142         }
33143     };
33144
33145     // private
33146     var updateButtons = function(b){
33147         var width = 0;
33148         if(!b){
33149             buttons["ok"].hide();
33150             buttons["cancel"].hide();
33151             buttons["yes"].hide();
33152             buttons["no"].hide();
33153             dlg.footer.dom.style.display = 'none';
33154             return width;
33155         }
33156         dlg.footer.dom.style.display = '';
33157         for(var k in buttons){
33158             if(typeof buttons[k] != "function"){
33159                 if(b[k]){
33160                     buttons[k].show();
33161                     buttons[k].setText(typeof b[k] == "string" ? b[k] : Roo.MessageBox.buttonText[k]);
33162                     width += buttons[k].el.getWidth()+15;
33163                 }else{
33164                     buttons[k].hide();
33165                 }
33166             }
33167         }
33168         return width;
33169     };
33170
33171     // private
33172     var handleEsc = function(d, k, e){
33173         if(opt && opt.closable !== false){
33174             dlg.hide();
33175         }
33176         if(e){
33177             e.stopEvent();
33178         }
33179     };
33180
33181     return {
33182         /**
33183          * Returns a reference to the underlying {@link Roo.BasicDialog} element
33184          * @return {Roo.BasicDialog} The BasicDialog element
33185          */
33186         getDialog : function(){
33187            if(!dlg){
33188                 dlg = new Roo.BasicDialog("x-msg-box", {
33189                     autoCreate : true,
33190                     shadow: true,
33191                     draggable: true,
33192                     resizable:false,
33193                     constraintoviewport:false,
33194                     fixedcenter:true,
33195                     collapsible : false,
33196                     shim:true,
33197                     modal: true,
33198                     width:400, height:100,
33199                     buttonAlign:"center",
33200                     closeClick : function(){
33201                         if(opt && opt.buttons && opt.buttons.no && !opt.buttons.cancel){
33202                             handleButton("no");
33203                         }else{
33204                             handleButton("cancel");
33205                         }
33206                     }
33207                 });
33208                 dlg.on("hide", handleHide);
33209                 mask = dlg.mask;
33210                 dlg.addKeyListener(27, handleEsc);
33211                 buttons = {};
33212                 var bt = this.buttonText;
33213                 buttons["ok"] = dlg.addButton(bt["ok"], handleButton.createCallback("ok"));
33214                 buttons["yes"] = dlg.addButton(bt["yes"], handleButton.createCallback("yes"));
33215                 buttons["no"] = dlg.addButton(bt["no"], handleButton.createCallback("no"));
33216                 buttons["cancel"] = dlg.addButton(bt["cancel"], handleButton.createCallback("cancel"));
33217                 bodyEl = dlg.body.createChild({
33218
33219                     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>'
33220                 });
33221                 msgEl = bodyEl.dom.firstChild;
33222                 textboxEl = Roo.get(bodyEl.dom.childNodes[2]);
33223                 textboxEl.enableDisplayMode();
33224                 textboxEl.addKeyListener([10,13], function(){
33225                     if(dlg.isVisible() && opt && opt.buttons){
33226                         if(opt.buttons.ok){
33227                             handleButton("ok");
33228                         }else if(opt.buttons.yes){
33229                             handleButton("yes");
33230                         }
33231                     }
33232                 });
33233                 textareaEl = Roo.get(bodyEl.dom.childNodes[3]);
33234                 textareaEl.enableDisplayMode();
33235                 progressEl = Roo.get(bodyEl.dom.childNodes[4]);
33236                 progressEl.enableDisplayMode();
33237                 var pf = progressEl.dom.firstChild;
33238                 if (pf) {
33239                     pp = Roo.get(pf.firstChild);
33240                     pp.setHeight(pf.offsetHeight);
33241                 }
33242                 
33243             }
33244             return dlg;
33245         },
33246
33247         /**
33248          * Updates the message box body text
33249          * @param {String} text (optional) Replaces the message box element's innerHTML with the specified string (defaults to
33250          * the XHTML-compliant non-breaking space character '&amp;#160;')
33251          * @return {Roo.MessageBox} This message box
33252          */
33253         updateText : function(text){
33254             if(!dlg.isVisible() && !opt.width){
33255                 dlg.resizeTo(this.maxWidth, 100); // resize first so content is never clipped from previous shows
33256             }
33257             msgEl.innerHTML = text || '&#160;';
33258       
33259             var cw =  Math.max(msgEl.offsetWidth, msgEl.parentNode.scrollWidth);
33260             //Roo.log("guesed size: " + JSON.stringify([cw,msgEl.offsetWidth, msgEl.parentNode.scrollWidth]));
33261             var w = Math.max(
33262                     Math.min(opt.width || cw , this.maxWidth), 
33263                     Math.max(opt.minWidth || this.minWidth, bwidth)
33264             );
33265             if(opt.prompt){
33266                 activeTextEl.setWidth(w);
33267             }
33268             if(dlg.isVisible()){
33269                 dlg.fixedcenter = false;
33270             }
33271             // to big, make it scroll. = But as usual stupid IE does not support
33272             // !important..
33273             
33274             if ( bodyEl.getHeight() > (Roo.lib.Dom.getViewHeight() - 100)) {
33275                 bodyEl.setHeight ( Roo.lib.Dom.getViewHeight() - 100 );
33276                 bodyEl.dom.style.overflowY = 'auto' + ( Roo.isIE ? '' : ' !important');
33277             } else {
33278                 bodyEl.dom.style.height = '';
33279                 bodyEl.dom.style.overflowY = '';
33280             }
33281             if (cw > w) {
33282                 bodyEl.dom.style.get = 'auto' + ( Roo.isIE ? '' : ' !important');
33283             } else {
33284                 bodyEl.dom.style.overflowX = '';
33285             }
33286             
33287             dlg.setContentSize(w, bodyEl.getHeight());
33288             if(dlg.isVisible()){
33289                 dlg.fixedcenter = true;
33290             }
33291             return this;
33292         },
33293
33294         /**
33295          * Updates a progress-style message box's text and progress bar.  Only relevant on message boxes
33296          * initiated via {@link Roo.MessageBox#progress} or by calling {@link Roo.MessageBox#show} with progress: true.
33297          * @param {Number} value Any number between 0 and 1 (e.g., .5)
33298          * @param {String} text (optional) If defined, the message box's body text is replaced with the specified string (defaults to undefined)
33299          * @return {Roo.MessageBox} This message box
33300          */
33301         updateProgress : function(value, text){
33302             if(text){
33303                 this.updateText(text);
33304             }
33305             if (pp) { // weird bug on my firefox - for some reason this is not defined
33306                 pp.setWidth(Math.floor(value*progressEl.dom.firstChild.offsetWidth));
33307             }
33308             return this;
33309         },        
33310
33311         /**
33312          * Returns true if the message box is currently displayed
33313          * @return {Boolean} True if the message box is visible, else false
33314          */
33315         isVisible : function(){
33316             return dlg && dlg.isVisible();  
33317         },
33318
33319         /**
33320          * Hides the message box if it is displayed
33321          */
33322         hide : function(){
33323             if(this.isVisible()){
33324                 dlg.hide();
33325             }  
33326         },
33327
33328         /**
33329          * Displays a new message box, or reinitializes an existing message box, based on the config options
33330          * passed in. All functions (e.g. prompt, alert, etc) on MessageBox call this function internally.
33331          * The following config object properties are supported:
33332          * <pre>
33333 Property    Type             Description
33334 ----------  ---------------  ------------------------------------------------------------------------------------
33335 animEl            String/Element   An id or Element from which the message box should animate as it opens and
33336                                    closes (defaults to undefined)
33337 buttons           Object/Boolean   A button config object (e.g., Roo.MessageBox.OKCANCEL or {ok:'Foo',
33338                                    cancel:'Bar'}), or false to not show any buttons (defaults to false)
33339 closable          Boolean          False to hide the top-right close button (defaults to true).  Note that
33340                                    progress and wait dialogs will ignore this property and always hide the
33341                                    close button as they can only be closed programmatically.
33342 cls               String           A custom CSS class to apply to the message box element
33343 defaultTextHeight Number           The default height in pixels of the message box's multiline textarea if
33344                                    displayed (defaults to 75)
33345 fn                Function         A callback function to execute after closing the dialog.  The arguments to the
33346                                    function will be btn (the name of the button that was clicked, if applicable,
33347                                    e.g. "ok"), and text (the value of the active text field, if applicable).
33348                                    Progress and wait dialogs will ignore this option since they do not respond to
33349                                    user actions and can only be closed programmatically, so any required function
33350                                    should be called by the same code after it closes the dialog.
33351 icon              String           A CSS class that provides a background image to be used as an icon for
33352                                    the dialog (e.g., Roo.MessageBox.WARNING or 'custom-class', defaults to '')
33353 maxWidth          Number           The maximum width in pixels of the message box (defaults to 600)
33354 minWidth          Number           The minimum width in pixels of the message box (defaults to 100)
33355 modal             Boolean          False to allow user interaction with the page while the message box is
33356                                    displayed (defaults to true)
33357 msg               String           A string that will replace the existing message box body text (defaults
33358                                    to the XHTML-compliant non-breaking space character '&#160;')
33359 multiline         Boolean          True to prompt the user to enter multi-line text (defaults to false)
33360 progress          Boolean          True to display a progress bar (defaults to false)
33361 progressText      String           The text to display inside the progress bar if progress = true (defaults to '')
33362 prompt            Boolean          True to prompt the user to enter single-line text (defaults to false)
33363 proxyDrag         Boolean          True to display a lightweight proxy while dragging (defaults to false)
33364 title             String           The title text
33365 value             String           The string value to set into the active textbox element if displayed
33366 wait              Boolean          True to display a progress bar (defaults to false)
33367 width             Number           The width of the dialog in pixels
33368 </pre>
33369          *
33370          * Example usage:
33371          * <pre><code>
33372 Roo.Msg.show({
33373    title: 'Address',
33374    msg: 'Please enter your address:',
33375    width: 300,
33376    buttons: Roo.MessageBox.OKCANCEL,
33377    multiline: true,
33378    fn: saveAddress,
33379    animEl: 'addAddressBtn'
33380 });
33381 </code></pre>
33382          * @param {Object} config Configuration options
33383          * @return {Roo.MessageBox} This message box
33384          */
33385         show : function(options)
33386         {
33387             
33388             // this causes nightmares if you show one dialog after another
33389             // especially on callbacks..
33390              
33391             if(this.isVisible()){
33392                 
33393                 this.hide();
33394                 Roo.log("[Roo.Messagebox] Show called while message displayed:" );
33395                 Roo.log("Old Dialog Message:" +  msgEl.innerHTML );
33396                 Roo.log("New Dialog Message:" +  options.msg )
33397                 //this.alert("ERROR", "Multiple dialogs where displayed at the same time");
33398                 //throw "Roo.MessageBox ERROR : Multiple dialogs where displayed at the same time";
33399                 
33400             }
33401             var d = this.getDialog();
33402             opt = options;
33403             d.setTitle(opt.title || "&#160;");
33404             d.close.setDisplayed(opt.closable !== false);
33405             activeTextEl = textboxEl;
33406             opt.prompt = opt.prompt || (opt.multiline ? true : false);
33407             if(opt.prompt){
33408                 if(opt.multiline){
33409                     textboxEl.hide();
33410                     textareaEl.show();
33411                     textareaEl.setHeight(typeof opt.multiline == "number" ?
33412                         opt.multiline : this.defaultTextHeight);
33413                     activeTextEl = textareaEl;
33414                 }else{
33415                     textboxEl.show();
33416                     textareaEl.hide();
33417                 }
33418             }else{
33419                 textboxEl.hide();
33420                 textareaEl.hide();
33421             }
33422             progressEl.setDisplayed(opt.progress === true);
33423             this.updateProgress(0);
33424             activeTextEl.dom.value = opt.value || "";
33425             if(opt.prompt){
33426                 dlg.setDefaultButton(activeTextEl);
33427             }else{
33428                 var bs = opt.buttons;
33429                 var db = null;
33430                 if(bs && bs.ok){
33431                     db = buttons["ok"];
33432                 }else if(bs && bs.yes){
33433                     db = buttons["yes"];
33434                 }
33435                 dlg.setDefaultButton(db);
33436             }
33437             bwidth = updateButtons(opt.buttons);
33438             this.updateText(opt.msg);
33439             if(opt.cls){
33440                 d.el.addClass(opt.cls);
33441             }
33442             d.proxyDrag = opt.proxyDrag === true;
33443             d.modal = opt.modal !== false;
33444             d.mask = opt.modal !== false ? mask : false;
33445             if(!d.isVisible()){
33446                 // force it to the end of the z-index stack so it gets a cursor in FF
33447                 document.body.appendChild(dlg.el.dom);
33448                 d.animateTarget = null;
33449                 d.show(options.animEl);
33450             }
33451             return this;
33452         },
33453
33454         /**
33455          * Displays a message box with a progress bar.  This message box has no buttons and is not closeable by
33456          * the user.  You are responsible for updating the progress bar as needed via {@link Roo.MessageBox#updateProgress}
33457          * and closing the message box when the process is complete.
33458          * @param {String} title The title bar text
33459          * @param {String} msg The message box body text
33460          * @return {Roo.MessageBox} This message box
33461          */
33462         progress : function(title, msg){
33463             this.show({
33464                 title : title,
33465                 msg : msg,
33466                 buttons: false,
33467                 progress:true,
33468                 closable:false,
33469                 minWidth: this.minProgressWidth,
33470                 modal : true
33471             });
33472             return this;
33473         },
33474
33475         /**
33476          * Displays a standard read-only message box with an OK button (comparable to the basic JavaScript Window.alert).
33477          * If a callback function is passed it will be called after the user clicks the button, and the
33478          * id of the button that was clicked will be passed as the only parameter to the callback
33479          * (could also be the top-right close button).
33480          * @param {String} title The title bar text
33481          * @param {String} msg The message box body text
33482          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33483          * @param {Object} scope (optional) The scope of the callback function
33484          * @return {Roo.MessageBox} This message box
33485          */
33486         alert : function(title, msg, fn, scope){
33487             this.show({
33488                 title : title,
33489                 msg : msg,
33490                 buttons: this.OK,
33491                 fn: fn,
33492                 scope : scope,
33493                 modal : true
33494             });
33495             return this;
33496         },
33497
33498         /**
33499          * Displays a message box with an infinitely auto-updating progress bar.  This can be used to block user
33500          * interaction while waiting for a long-running process to complete that does not have defined intervals.
33501          * You are responsible for closing the message box when the process is complete.
33502          * @param {String} msg The message box body text
33503          * @param {String} title (optional) The title bar text
33504          * @return {Roo.MessageBox} This message box
33505          */
33506         wait : function(msg, title){
33507             this.show({
33508                 title : title,
33509                 msg : msg,
33510                 buttons: false,
33511                 closable:false,
33512                 progress:true,
33513                 modal:true,
33514                 width:300,
33515                 wait:true
33516             });
33517             waitTimer = Roo.TaskMgr.start({
33518                 run: function(i){
33519                     Roo.MessageBox.updateProgress(((((i+20)%20)+1)*5)*.01);
33520                 },
33521                 interval: 1000
33522             });
33523             return this;
33524         },
33525
33526         /**
33527          * Displays a confirmation message box with Yes and No buttons (comparable to JavaScript's Window.confirm).
33528          * If a callback function is passed it will be called after the user clicks either button, and the id of the
33529          * button that was clicked will be passed as the only parameter to the callback (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         confirm : function(title, msg, fn, scope){
33537             this.show({
33538                 title : title,
33539                 msg : msg,
33540                 buttons: this.YESNO,
33541                 fn: fn,
33542                 scope : scope,
33543                 modal : true
33544             });
33545             return this;
33546         },
33547
33548         /**
33549          * Displays a message box with OK and Cancel buttons prompting the user to enter some text (comparable to
33550          * JavaScript's Window.prompt).  The prompt can be a single-line or multi-line textbox.  If a callback function
33551          * is passed it will be called after the user clicks either button, and the id of the button that was clicked
33552          * (could also be the top-right close button) and the text that was entered will be passed as the two
33553          * parameters to the callback.
33554          * @param {String} title The title bar text
33555          * @param {String} msg The message box body text
33556          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33557          * @param {Object} scope (optional) The scope of the callback function
33558          * @param {Boolean/Number} multiline (optional) True to create a multiline textbox using the defaultTextHeight
33559          * property, or the height in pixels to create the textbox (defaults to false / single-line)
33560          * @return {Roo.MessageBox} This message box
33561          */
33562         prompt : function(title, msg, fn, scope, multiline){
33563             this.show({
33564                 title : title,
33565                 msg : msg,
33566                 buttons: this.OKCANCEL,
33567                 fn: fn,
33568                 minWidth:250,
33569                 scope : scope,
33570                 prompt:true,
33571                 multiline: multiline,
33572                 modal : true
33573             });
33574             return this;
33575         },
33576
33577         /**
33578          * Button config that displays a single OK button
33579          * @type Object
33580          */
33581         OK : {ok:true},
33582         /**
33583          * Button config that displays Yes and No buttons
33584          * @type Object
33585          */
33586         YESNO : {yes:true, no:true},
33587         /**
33588          * Button config that displays OK and Cancel buttons
33589          * @type Object
33590          */
33591         OKCANCEL : {ok:true, cancel:true},
33592         /**
33593          * Button config that displays Yes, No and Cancel buttons
33594          * @type Object
33595          */
33596         YESNOCANCEL : {yes:true, no:true, cancel:true},
33597
33598         /**
33599          * The default height in pixels of the message box's multiline textarea if displayed (defaults to 75)
33600          * @type Number
33601          */
33602         defaultTextHeight : 75,
33603         /**
33604          * The maximum width in pixels of the message box (defaults to 600)
33605          * @type Number
33606          */
33607         maxWidth : 600,
33608         /**
33609          * The minimum width in pixels of the message box (defaults to 100)
33610          * @type Number
33611          */
33612         minWidth : 100,
33613         /**
33614          * The minimum width in pixels of the message box if it is a progress-style dialog.  This is useful
33615          * for setting a different minimum width than text-only dialogs may need (defaults to 250)
33616          * @type Number
33617          */
33618         minProgressWidth : 250,
33619         /**
33620          * An object containing the default button text strings that can be overriden for localized language support.
33621          * Supported properties are: ok, cancel, yes and no.
33622          * Customize the default text like so: Roo.MessageBox.buttonText.yes = "S?";
33623          * @type Object
33624          */
33625         buttonText : {
33626             ok : "OK",
33627             cancel : "Cancel",
33628             yes : "Yes",
33629             no : "No"
33630         }
33631     };
33632 }();
33633
33634 /**
33635  * Shorthand for {@link Roo.MessageBox}
33636  */
33637 Roo.Msg = Roo.MessageBox;/*
33638  * Based on:
33639  * Ext JS Library 1.1.1
33640  * Copyright(c) 2006-2007, Ext JS, LLC.
33641  *
33642  * Originally Released Under LGPL - original licence link has changed is not relivant.
33643  *
33644  * Fork - LGPL
33645  * <script type="text/javascript">
33646  */
33647 /**
33648  * @class Roo.QuickTips
33649  * Provides attractive and customizable tooltips for any element.
33650  * @singleton
33651  */
33652 Roo.QuickTips = function(){
33653     var el, tipBody, tipBodyText, tipTitle, tm, cfg, close, tagEls = {}, esc, removeCls = null, bdLeft, bdRight;
33654     var ce, bd, xy, dd;
33655     var visible = false, disabled = true, inited = false;
33656     var showProc = 1, hideProc = 1, dismissProc = 1, locks = [];
33657     
33658     var onOver = function(e){
33659         if(disabled){
33660             return;
33661         }
33662         var t = e.getTarget();
33663         if(!t || t.nodeType !== 1 || t == document || t == document.body){
33664             return;
33665         }
33666         if(ce && t == ce.el){
33667             clearTimeout(hideProc);
33668             return;
33669         }
33670         if(t && tagEls[t.id]){
33671             tagEls[t.id].el = t;
33672             showProc = show.defer(tm.showDelay, tm, [tagEls[t.id]]);
33673             return;
33674         }
33675         var ttp, et = Roo.fly(t);
33676         var ns = cfg.namespace;
33677         if(tm.interceptTitles && t.title){
33678             ttp = t.title;
33679             t.qtip = ttp;
33680             t.removeAttribute("title");
33681             e.preventDefault();
33682         }else{
33683             ttp = t.qtip || et.getAttributeNS(ns, cfg.attribute) || et.getAttributeNS(cfg.alt_namespace, cfg.attribute) ;
33684         }
33685         if(ttp){
33686             showProc = show.defer(tm.showDelay, tm, [{
33687                 el: t, 
33688                 text: ttp.replace(/\\n/g,'<br/>'),
33689                 width: et.getAttributeNS(ns, cfg.width),
33690                 autoHide: et.getAttributeNS(ns, cfg.hide) != "user",
33691                 title: et.getAttributeNS(ns, cfg.title),
33692                     cls: et.getAttributeNS(ns, cfg.cls)
33693             }]);
33694         }
33695     };
33696     
33697     var onOut = function(e){
33698         clearTimeout(showProc);
33699         var t = e.getTarget();
33700         if(t && ce && ce.el == t && (tm.autoHide && ce.autoHide !== false)){
33701             hideProc = setTimeout(hide, tm.hideDelay);
33702         }
33703     };
33704     
33705     var onMove = function(e){
33706         if(disabled){
33707             return;
33708         }
33709         xy = e.getXY();
33710         xy[1] += 18;
33711         if(tm.trackMouse && ce){
33712             el.setXY(xy);
33713         }
33714     };
33715     
33716     var onDown = function(e){
33717         clearTimeout(showProc);
33718         clearTimeout(hideProc);
33719         if(!e.within(el)){
33720             if(tm.hideOnClick){
33721                 hide();
33722                 tm.disable();
33723                 tm.enable.defer(100, tm);
33724             }
33725         }
33726     };
33727     
33728     var getPad = function(){
33729         return 2;//bdLeft.getPadding('l')+bdRight.getPadding('r');
33730     };
33731
33732     var show = function(o){
33733         if(disabled){
33734             return;
33735         }
33736         clearTimeout(dismissProc);
33737         ce = o;
33738         if(removeCls){ // in case manually hidden
33739             el.removeClass(removeCls);
33740             removeCls = null;
33741         }
33742         if(ce.cls){
33743             el.addClass(ce.cls);
33744             removeCls = ce.cls;
33745         }
33746         if(ce.title){
33747             tipTitle.update(ce.title);
33748             tipTitle.show();
33749         }else{
33750             tipTitle.update('');
33751             tipTitle.hide();
33752         }
33753         el.dom.style.width  = tm.maxWidth+'px';
33754         //tipBody.dom.style.width = '';
33755         tipBodyText.update(o.text);
33756         var p = getPad(), w = ce.width;
33757         if(!w){
33758             var td = tipBodyText.dom;
33759             var aw = Math.max(td.offsetWidth, td.clientWidth, td.scrollWidth);
33760             if(aw > tm.maxWidth){
33761                 w = tm.maxWidth;
33762             }else if(aw < tm.minWidth){
33763                 w = tm.minWidth;
33764             }else{
33765                 w = aw;
33766             }
33767         }
33768         //tipBody.setWidth(w);
33769         el.setWidth(parseInt(w, 10) + p);
33770         if(ce.autoHide === false){
33771             close.setDisplayed(true);
33772             if(dd){
33773                 dd.unlock();
33774             }
33775         }else{
33776             close.setDisplayed(false);
33777             if(dd){
33778                 dd.lock();
33779             }
33780         }
33781         if(xy){
33782             el.avoidY = xy[1]-18;
33783             el.setXY(xy);
33784         }
33785         if(tm.animate){
33786             el.setOpacity(.1);
33787             el.setStyle("visibility", "visible");
33788             el.fadeIn({callback: afterShow});
33789         }else{
33790             afterShow();
33791         }
33792     };
33793     
33794     var afterShow = function(){
33795         if(ce){
33796             el.show();
33797             esc.enable();
33798             if(tm.autoDismiss && ce.autoHide !== false){
33799                 dismissProc = setTimeout(hide, tm.autoDismissDelay);
33800             }
33801         }
33802     };
33803     
33804     var hide = function(noanim){
33805         clearTimeout(dismissProc);
33806         clearTimeout(hideProc);
33807         ce = null;
33808         if(el.isVisible()){
33809             esc.disable();
33810             if(noanim !== true && tm.animate){
33811                 el.fadeOut({callback: afterHide});
33812             }else{
33813                 afterHide();
33814             } 
33815         }
33816     };
33817     
33818     var afterHide = function(){
33819         el.hide();
33820         if(removeCls){
33821             el.removeClass(removeCls);
33822             removeCls = null;
33823         }
33824     };
33825     
33826     return {
33827         /**
33828         * @cfg {Number} minWidth
33829         * The minimum width of the quick tip (defaults to 40)
33830         */
33831        minWidth : 40,
33832         /**
33833         * @cfg {Number} maxWidth
33834         * The maximum width of the quick tip (defaults to 300)
33835         */
33836        maxWidth : 300,
33837         /**
33838         * @cfg {Boolean} interceptTitles
33839         * True to automatically use the element's DOM title value if available (defaults to false)
33840         */
33841        interceptTitles : false,
33842         /**
33843         * @cfg {Boolean} trackMouse
33844         * True to have the quick tip follow the mouse as it moves over the target element (defaults to false)
33845         */
33846        trackMouse : false,
33847         /**
33848         * @cfg {Boolean} hideOnClick
33849         * True to hide the quick tip if the user clicks anywhere in the document (defaults to true)
33850         */
33851        hideOnClick : true,
33852         /**
33853         * @cfg {Number} showDelay
33854         * Delay in milliseconds before the quick tip displays after the mouse enters the target element (defaults to 500)
33855         */
33856        showDelay : 500,
33857         /**
33858         * @cfg {Number} hideDelay
33859         * Delay in milliseconds before the quick tip hides when autoHide = true (defaults to 200)
33860         */
33861        hideDelay : 200,
33862         /**
33863         * @cfg {Boolean} autoHide
33864         * True to automatically hide the quick tip after the mouse exits the target element (defaults to true).
33865         * Used in conjunction with hideDelay.
33866         */
33867        autoHide : true,
33868         /**
33869         * @cfg {Boolean}
33870         * True to automatically hide the quick tip after a set period of time, regardless of the user's actions
33871         * (defaults to true).  Used in conjunction with autoDismissDelay.
33872         */
33873        autoDismiss : true,
33874         /**
33875         * @cfg {Number}
33876         * Delay in milliseconds before the quick tip hides when autoDismiss = true (defaults to 5000)
33877         */
33878        autoDismissDelay : 5000,
33879        /**
33880         * @cfg {Boolean} animate
33881         * True to turn on fade animation. Defaults to false (ClearType/scrollbar flicker issues in IE7).
33882         */
33883        animate : false,
33884
33885        /**
33886         * @cfg {String} title
33887         * Title text to display (defaults to '').  This can be any valid HTML markup.
33888         */
33889         title: '',
33890        /**
33891         * @cfg {String} text
33892         * Body text to display (defaults to '').  This can be any valid HTML markup.
33893         */
33894         text : '',
33895        /**
33896         * @cfg {String} cls
33897         * A CSS class to apply to the base quick tip element (defaults to '').
33898         */
33899         cls : '',
33900        /**
33901         * @cfg {Number} width
33902         * Width in pixels of the quick tip (defaults to auto).  Width will be ignored if it exceeds the bounds of
33903         * minWidth or maxWidth.
33904         */
33905         width : null,
33906
33907     /**
33908      * Initialize and enable QuickTips for first use.  This should be called once before the first attempt to access
33909      * or display QuickTips in a page.
33910      */
33911        init : function(){
33912           tm = Roo.QuickTips;
33913           cfg = tm.tagConfig;
33914           if(!inited){
33915               if(!Roo.isReady){ // allow calling of init() before onReady
33916                   Roo.onReady(Roo.QuickTips.init, Roo.QuickTips);
33917                   return;
33918               }
33919               el = new Roo.Layer({cls:"x-tip", shadow:"drop", shim: true, constrain:true, shadowOffset:4});
33920               el.fxDefaults = {stopFx: true};
33921               // maximum custom styling
33922               //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>');
33923               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>');              
33924               tipTitle = el.child('h3');
33925               tipTitle.enableDisplayMode("block");
33926               tipBody = el.child('div.x-tip-bd');
33927               tipBodyText = el.child('div.x-tip-bd-inner');
33928               //bdLeft = el.child('div.x-tip-bd-left');
33929               //bdRight = el.child('div.x-tip-bd-right');
33930               close = el.child('div.x-tip-close');
33931               close.enableDisplayMode("block");
33932               close.on("click", hide);
33933               var d = Roo.get(document);
33934               d.on("mousedown", onDown);
33935               d.on("mouseover", onOver);
33936               d.on("mouseout", onOut);
33937               d.on("mousemove", onMove);
33938               esc = d.addKeyListener(27, hide);
33939               esc.disable();
33940               if(Roo.dd.DD){
33941                   dd = el.initDD("default", null, {
33942                       onDrag : function(){
33943                           el.sync();  
33944                       }
33945                   });
33946                   dd.setHandleElId(tipTitle.id);
33947                   dd.lock();
33948               }
33949               inited = true;
33950           }
33951           this.enable(); 
33952        },
33953
33954     /**
33955      * Configures a new quick tip instance and assigns it to a target element.  The following config options
33956      * are supported:
33957      * <pre>
33958 Property    Type                   Description
33959 ----------  ---------------------  ------------------------------------------------------------------------
33960 target      Element/String/Array   An Element, id or array of ids that this quick tip should be tied to
33961      * </ul>
33962      * @param {Object} config The config object
33963      */
33964        register : function(config){
33965            var cs = config instanceof Array ? config : arguments;
33966            for(var i = 0, len = cs.length; i < len; i++) {
33967                var c = cs[i];
33968                var target = c.target;
33969                if(target){
33970                    if(target instanceof Array){
33971                        for(var j = 0, jlen = target.length; j < jlen; j++){
33972                            tagEls[target[j]] = c;
33973                        }
33974                    }else{
33975                        tagEls[typeof target == 'string' ? target : Roo.id(target)] = c;
33976                    }
33977                }
33978            }
33979        },
33980
33981     /**
33982      * Removes this quick tip from its element and destroys it.
33983      * @param {String/HTMLElement/Element} el The element from which the quick tip is to be removed.
33984      */
33985        unregister : function(el){
33986            delete tagEls[Roo.id(el)];
33987        },
33988
33989     /**
33990      * Enable this quick tip.
33991      */
33992        enable : function(){
33993            if(inited && disabled){
33994                locks.pop();
33995                if(locks.length < 1){
33996                    disabled = false;
33997                }
33998            }
33999        },
34000
34001     /**
34002      * Disable this quick tip.
34003      */
34004        disable : function(){
34005           disabled = true;
34006           clearTimeout(showProc);
34007           clearTimeout(hideProc);
34008           clearTimeout(dismissProc);
34009           if(ce){
34010               hide(true);
34011           }
34012           locks.push(1);
34013        },
34014
34015     /**
34016      * Returns true if the quick tip is enabled, else false.
34017      */
34018        isEnabled : function(){
34019             return !disabled;
34020        },
34021
34022         // private
34023        tagConfig : {
34024            namespace : "roo", // was ext?? this may break..
34025            alt_namespace : "ext",
34026            attribute : "qtip",
34027            width : "width",
34028            target : "target",
34029            title : "qtitle",
34030            hide : "hide",
34031            cls : "qclass"
34032        }
34033    };
34034 }();
34035
34036 // backwards compat
34037 Roo.QuickTips.tips = Roo.QuickTips.register;/*
34038  * Based on:
34039  * Ext JS Library 1.1.1
34040  * Copyright(c) 2006-2007, Ext JS, LLC.
34041  *
34042  * Originally Released Under LGPL - original licence link has changed is not relivant.
34043  *
34044  * Fork - LGPL
34045  * <script type="text/javascript">
34046  */
34047  
34048
34049 /**
34050  * @class Roo.tree.TreePanel
34051  * @extends Roo.data.Tree
34052
34053  * @cfg {Boolean} rootVisible false to hide the root node (defaults to true)
34054  * @cfg {Boolean} lines false to disable tree lines (defaults to true)
34055  * @cfg {Boolean} enableDD true to enable drag and drop
34056  * @cfg {Boolean} enableDrag true to enable just drag
34057  * @cfg {Boolean} enableDrop true to enable just drop
34058  * @cfg {Object} dragConfig Custom config to pass to the {@link Roo.tree.TreeDragZone} instance
34059  * @cfg {Object} dropConfig Custom config to pass to the {@link Roo.tree.TreeDropZone} instance
34060  * @cfg {String} ddGroup The DD group this TreePanel belongs to
34061  * @cfg {String} ddAppendOnly True if the tree should only allow append drops (use for trees which are sorted)
34062  * @cfg {Boolean} ddScroll true to enable YUI body scrolling
34063  * @cfg {Boolean} containerScroll true to register this container with ScrollManager
34064  * @cfg {Boolean} hlDrop false to disable node highlight on drop (defaults to the value of Roo.enableFx)
34065  * @cfg {String} hlColor The color of the node highlight (defaults to C3DAF9)
34066  * @cfg {Boolean} animate true to enable animated expand/collapse (defaults to the value of Roo.enableFx)
34067  * @cfg {Boolean} singleExpand true if only 1 node per branch may be expanded
34068  * @cfg {Boolean} selModel A tree selection model to use with this TreePanel (defaults to a {@link Roo.tree.DefaultSelectionModel})
34069  * @cfg {Boolean} loader A TreeLoader for use with this TreePanel
34070  * @cfg {Object|Roo.tree.TreeEditor} editor The TreeEditor or xtype data to display when clicked.
34071  * @cfg {String} pathSeparator The token used to separate sub-paths in path strings (defaults to '/')
34072  * @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>
34073  * @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>
34074  * 
34075  * @constructor
34076  * @param {String/HTMLElement/Element} el The container element
34077  * @param {Object} config
34078  */
34079 Roo.tree.TreePanel = function(el, config){
34080     var root = false;
34081     var loader = false;
34082     if (config.root) {
34083         root = config.root;
34084         delete config.root;
34085     }
34086     if (config.loader) {
34087         loader = config.loader;
34088         delete config.loader;
34089     }
34090     
34091     Roo.apply(this, config);
34092     Roo.tree.TreePanel.superclass.constructor.call(this);
34093     this.el = Roo.get(el);
34094     this.el.addClass('x-tree');
34095     //console.log(root);
34096     if (root) {
34097         this.setRootNode( Roo.factory(root, Roo.tree));
34098     }
34099     if (loader) {
34100         this.loader = Roo.factory(loader, Roo.tree);
34101     }
34102    /**
34103     * Read-only. The id of the container element becomes this TreePanel's id.
34104     */
34105     this.id = this.el.id;
34106     this.addEvents({
34107         /**
34108         * @event beforeload
34109         * Fires before a node is loaded, return false to cancel
34110         * @param {Node} node The node being loaded
34111         */
34112         "beforeload" : true,
34113         /**
34114         * @event load
34115         * Fires when a node is loaded
34116         * @param {Node} node The node that was loaded
34117         */
34118         "load" : true,
34119         /**
34120         * @event textchange
34121         * Fires when the text for a node is changed
34122         * @param {Node} node The node
34123         * @param {String} text The new text
34124         * @param {String} oldText The old text
34125         */
34126         "textchange" : true,
34127         /**
34128         * @event beforeexpand
34129         * Fires before a node is expanded, return false to cancel.
34130         * @param {Node} node The node
34131         * @param {Boolean} deep
34132         * @param {Boolean} anim
34133         */
34134         "beforeexpand" : true,
34135         /**
34136         * @event beforecollapse
34137         * Fires before a node is collapsed, return false to cancel.
34138         * @param {Node} node The node
34139         * @param {Boolean} deep
34140         * @param {Boolean} anim
34141         */
34142         "beforecollapse" : true,
34143         /**
34144         * @event expand
34145         * Fires when a node is expanded
34146         * @param {Node} node The node
34147         */
34148         "expand" : true,
34149         /**
34150         * @event disabledchange
34151         * Fires when the disabled status of a node changes
34152         * @param {Node} node The node
34153         * @param {Boolean} disabled
34154         */
34155         "disabledchange" : true,
34156         /**
34157         * @event collapse
34158         * Fires when a node is collapsed
34159         * @param {Node} node The node
34160         */
34161         "collapse" : true,
34162         /**
34163         * @event beforeclick
34164         * Fires before click processing on a node. Return false to cancel the default action.
34165         * @param {Node} node The node
34166         * @param {Roo.EventObject} e The event object
34167         */
34168         "beforeclick":true,
34169         /**
34170         * @event checkchange
34171         * Fires when a node with a checkbox's checked property changes
34172         * @param {Node} this This node
34173         * @param {Boolean} checked
34174         */
34175         "checkchange":true,
34176         /**
34177         * @event click
34178         * Fires when a node is clicked
34179         * @param {Node} node The node
34180         * @param {Roo.EventObject} e The event object
34181         */
34182         "click":true,
34183         /**
34184         * @event dblclick
34185         * Fires when a node is double clicked
34186         * @param {Node} node The node
34187         * @param {Roo.EventObject} e The event object
34188         */
34189         "dblclick":true,
34190         /**
34191         * @event contextmenu
34192         * Fires when a node is right clicked
34193         * @param {Node} node The node
34194         * @param {Roo.EventObject} e The event object
34195         */
34196         "contextmenu":true,
34197         /**
34198         * @event beforechildrenrendered
34199         * Fires right before the child nodes for a node are rendered
34200         * @param {Node} node The node
34201         */
34202         "beforechildrenrendered":true,
34203         /**
34204         * @event startdrag
34205         * Fires when a node starts being dragged
34206         * @param {Roo.tree.TreePanel} this
34207         * @param {Roo.tree.TreeNode} node
34208         * @param {event} e The raw browser event
34209         */ 
34210        "startdrag" : true,
34211        /**
34212         * @event enddrag
34213         * Fires when a drag operation is complete
34214         * @param {Roo.tree.TreePanel} this
34215         * @param {Roo.tree.TreeNode} node
34216         * @param {event} e The raw browser event
34217         */
34218        "enddrag" : true,
34219        /**
34220         * @event dragdrop
34221         * Fires when a dragged node is dropped on a valid DD target
34222         * @param {Roo.tree.TreePanel} this
34223         * @param {Roo.tree.TreeNode} node
34224         * @param {DD} dd The dd it was dropped on
34225         * @param {event} e The raw browser event
34226         */
34227        "dragdrop" : true,
34228        /**
34229         * @event beforenodedrop
34230         * Fires when a DD object is dropped on a node in this tree for preprocessing. Return false to cancel the drop. The dropEvent
34231         * passed to handlers has the following properties:<br />
34232         * <ul style="padding:5px;padding-left:16px;">
34233         * <li>tree - The TreePanel</li>
34234         * <li>target - The node being targeted for the drop</li>
34235         * <li>data - The drag data from the drag source</li>
34236         * <li>point - The point of the drop - append, above or below</li>
34237         * <li>source - The drag source</li>
34238         * <li>rawEvent - Raw mouse event</li>
34239         * <li>dropNode - Drop node(s) provided by the source <b>OR</b> you can supply node(s)
34240         * to be inserted by setting them on this object.</li>
34241         * <li>cancel - Set this to true to cancel the drop.</li>
34242         * </ul>
34243         * @param {Object} dropEvent
34244         */
34245        "beforenodedrop" : true,
34246        /**
34247         * @event nodedrop
34248         * Fires after a DD object is dropped on a node in this tree. The dropEvent
34249         * passed to handlers has the following properties:<br />
34250         * <ul style="padding:5px;padding-left:16px;">
34251         * <li>tree - The TreePanel</li>
34252         * <li>target - The node being targeted for the drop</li>
34253         * <li>data - The drag data from the drag source</li>
34254         * <li>point - The point of the drop - append, above or below</li>
34255         * <li>source - The drag source</li>
34256         * <li>rawEvent - Raw mouse event</li>
34257         * <li>dropNode - Dropped node(s).</li>
34258         * </ul>
34259         * @param {Object} dropEvent
34260         */
34261        "nodedrop" : true,
34262         /**
34263         * @event nodedragover
34264         * Fires when a tree node is being targeted for a drag drop, return false to signal drop not allowed. The dragOverEvent
34265         * passed to handlers has the following properties:<br />
34266         * <ul style="padding:5px;padding-left:16px;">
34267         * <li>tree - The TreePanel</li>
34268         * <li>target - The node being targeted for the drop</li>
34269         * <li>data - The drag data from the drag source</li>
34270         * <li>point - The point of the drop - append, above or below</li>
34271         * <li>source - The drag source</li>
34272         * <li>rawEvent - Raw mouse event</li>
34273         * <li>dropNode - Drop node(s) provided by the source.</li>
34274         * <li>cancel - Set this to true to signal drop not allowed.</li>
34275         * </ul>
34276         * @param {Object} dragOverEvent
34277         */
34278        "nodedragover" : true
34279         
34280     });
34281     if(this.singleExpand){
34282        this.on("beforeexpand", this.restrictExpand, this);
34283     }
34284     if (this.editor) {
34285         this.editor.tree = this;
34286         this.editor = Roo.factory(this.editor, Roo.tree);
34287     }
34288     
34289     if (this.selModel) {
34290         this.selModel = Roo.factory(this.selModel, Roo.tree);
34291     }
34292    
34293 };
34294 Roo.extend(Roo.tree.TreePanel, Roo.data.Tree, {
34295     rootVisible : true,
34296     animate: Roo.enableFx,
34297     lines : true,
34298     enableDD : false,
34299     hlDrop : Roo.enableFx,
34300   
34301     renderer: false,
34302     
34303     rendererTip: false,
34304     // private
34305     restrictExpand : function(node){
34306         var p = node.parentNode;
34307         if(p){
34308             if(p.expandedChild && p.expandedChild.parentNode == p){
34309                 p.expandedChild.collapse();
34310             }
34311             p.expandedChild = node;
34312         }
34313     },
34314
34315     // private override
34316     setRootNode : function(node){
34317         Roo.tree.TreePanel.superclass.setRootNode.call(this, node);
34318         if(!this.rootVisible){
34319             node.ui = new Roo.tree.RootTreeNodeUI(node);
34320         }
34321         return node;
34322     },
34323
34324     /**
34325      * Returns the container element for this TreePanel
34326      */
34327     getEl : function(){
34328         return this.el;
34329     },
34330
34331     /**
34332      * Returns the default TreeLoader for this TreePanel
34333      */
34334     getLoader : function(){
34335         return this.loader;
34336     },
34337
34338     /**
34339      * Expand all nodes
34340      */
34341     expandAll : function(){
34342         this.root.expand(true);
34343     },
34344
34345     /**
34346      * Collapse all nodes
34347      */
34348     collapseAll : function(){
34349         this.root.collapse(true);
34350     },
34351
34352     /**
34353      * Returns the selection model used by this TreePanel
34354      */
34355     getSelectionModel : function(){
34356         if(!this.selModel){
34357             this.selModel = new Roo.tree.DefaultSelectionModel();
34358         }
34359         return this.selModel;
34360     },
34361
34362     /**
34363      * Retrieve an array of checked nodes, or an array of a specific attribute of checked nodes (e.g. "id")
34364      * @param {String} attribute (optional) Defaults to null (return the actual nodes)
34365      * @param {TreeNode} startNode (optional) The node to start from, defaults to the root
34366      * @return {Array}
34367      */
34368     getChecked : function(a, startNode){
34369         startNode = startNode || this.root;
34370         var r = [];
34371         var f = function(){
34372             if(this.attributes.checked){
34373                 r.push(!a ? this : (a == 'id' ? this.id : this.attributes[a]));
34374             }
34375         }
34376         startNode.cascade(f);
34377         return r;
34378     },
34379
34380     /**
34381      * Expands a specified path in this TreePanel. A path can be retrieved from a node with {@link Roo.data.Node#getPath}
34382      * @param {String} path
34383      * @param {String} attr (optional) The attribute used in the path (see {@link Roo.data.Node#getPath} for more info)
34384      * @param {Function} callback (optional) The callback to call when the expand is complete. The callback will be called with
34385      * (bSuccess, oLastNode) where bSuccess is if the expand was successful and oLastNode is the last node that was expanded.
34386      */
34387     expandPath : function(path, attr, callback){
34388         attr = attr || "id";
34389         var keys = path.split(this.pathSeparator);
34390         var curNode = this.root;
34391         if(curNode.attributes[attr] != keys[1]){ // invalid root
34392             if(callback){
34393                 callback(false, null);
34394             }
34395             return;
34396         }
34397         var index = 1;
34398         var f = function(){
34399             if(++index == keys.length){
34400                 if(callback){
34401                     callback(true, curNode);
34402                 }
34403                 return;
34404             }
34405             var c = curNode.findChild(attr, keys[index]);
34406             if(!c){
34407                 if(callback){
34408                     callback(false, curNode);
34409                 }
34410                 return;
34411             }
34412             curNode = c;
34413             c.expand(false, false, f);
34414         };
34415         curNode.expand(false, false, f);
34416     },
34417
34418     /**
34419      * Selects the node in this tree at the specified path. A path can be retrieved from a node with {@link Roo.data.Node#getPath}
34420      * @param {String} path
34421      * @param {String} attr (optional) The attribute used in the path (see {@link Roo.data.Node#getPath} for more info)
34422      * @param {Function} callback (optional) The callback to call when the selection is complete. The callback will be called with
34423      * (bSuccess, oSelNode) where bSuccess is if the selection was successful and oSelNode is the selected node.
34424      */
34425     selectPath : function(path, attr, callback){
34426         attr = attr || "id";
34427         var keys = path.split(this.pathSeparator);
34428         var v = keys.pop();
34429         if(keys.length > 0){
34430             var f = function(success, node){
34431                 if(success && node){
34432                     var n = node.findChild(attr, v);
34433                     if(n){
34434                         n.select();
34435                         if(callback){
34436                             callback(true, n);
34437                         }
34438                     }else if(callback){
34439                         callback(false, n);
34440                     }
34441                 }else{
34442                     if(callback){
34443                         callback(false, n);
34444                     }
34445                 }
34446             };
34447             this.expandPath(keys.join(this.pathSeparator), attr, f);
34448         }else{
34449             this.root.select();
34450             if(callback){
34451                 callback(true, this.root);
34452             }
34453         }
34454     },
34455
34456     getTreeEl : function(){
34457         return this.el;
34458     },
34459
34460     /**
34461      * Trigger rendering of this TreePanel
34462      */
34463     render : function(){
34464         if (this.innerCt) {
34465             return this; // stop it rendering more than once!!
34466         }
34467         
34468         this.innerCt = this.el.createChild({tag:"ul",
34469                cls:"x-tree-root-ct " +
34470                (this.lines ? "x-tree-lines" : "x-tree-no-lines")});
34471
34472         if(this.containerScroll){
34473             Roo.dd.ScrollManager.register(this.el);
34474         }
34475         if((this.enableDD || this.enableDrop) && !this.dropZone){
34476            /**
34477             * The dropZone used by this tree if drop is enabled
34478             * @type Roo.tree.TreeDropZone
34479             */
34480              this.dropZone = new Roo.tree.TreeDropZone(this, this.dropConfig || {
34481                ddGroup: this.ddGroup || "TreeDD", appendOnly: this.ddAppendOnly === true
34482            });
34483         }
34484         if((this.enableDD || this.enableDrag) && !this.dragZone){
34485            /**
34486             * The dragZone used by this tree if drag is enabled
34487             * @type Roo.tree.TreeDragZone
34488             */
34489             this.dragZone = new Roo.tree.TreeDragZone(this, this.dragConfig || {
34490                ddGroup: this.ddGroup || "TreeDD",
34491                scroll: this.ddScroll
34492            });
34493         }
34494         this.getSelectionModel().init(this);
34495         if (!this.root) {
34496             Roo.log("ROOT not set in tree");
34497             return this;
34498         }
34499         this.root.render();
34500         if(!this.rootVisible){
34501             this.root.renderChildren();
34502         }
34503         return this;
34504     }
34505 });/*
34506  * Based on:
34507  * Ext JS Library 1.1.1
34508  * Copyright(c) 2006-2007, Ext JS, LLC.
34509  *
34510  * Originally Released Under LGPL - original licence link has changed is not relivant.
34511  *
34512  * Fork - LGPL
34513  * <script type="text/javascript">
34514  */
34515  
34516
34517 /**
34518  * @class Roo.tree.DefaultSelectionModel
34519  * @extends Roo.util.Observable
34520  * The default single selection for a TreePanel.
34521  * @param {Object} cfg Configuration
34522  */
34523 Roo.tree.DefaultSelectionModel = function(cfg){
34524    this.selNode = null;
34525    
34526    
34527    
34528    this.addEvents({
34529        /**
34530         * @event selectionchange
34531         * Fires when the selected node changes
34532         * @param {DefaultSelectionModel} this
34533         * @param {TreeNode} node the new selection
34534         */
34535        "selectionchange" : true,
34536
34537        /**
34538         * @event beforeselect
34539         * Fires before the selected node changes, return false to cancel the change
34540         * @param {DefaultSelectionModel} this
34541         * @param {TreeNode} node the new selection
34542         * @param {TreeNode} node the old selection
34543         */
34544        "beforeselect" : true
34545    });
34546    
34547     Roo.tree.DefaultSelectionModel.superclass.constructor.call(this,cfg);
34548 };
34549
34550 Roo.extend(Roo.tree.DefaultSelectionModel, Roo.util.Observable, {
34551     init : function(tree){
34552         this.tree = tree;
34553         tree.getTreeEl().on("keydown", this.onKeyDown, this);
34554         tree.on("click", this.onNodeClick, this);
34555     },
34556     
34557     onNodeClick : function(node, e){
34558         if (e.ctrlKey && this.selNode == node)  {
34559             this.unselect(node);
34560             return;
34561         }
34562         this.select(node);
34563     },
34564     
34565     /**
34566      * Select a node.
34567      * @param {TreeNode} node The node to select
34568      * @return {TreeNode} The selected node
34569      */
34570     select : function(node){
34571         var last = this.selNode;
34572         if(last != node && this.fireEvent('beforeselect', this, node, last) !== false){
34573             if(last){
34574                 last.ui.onSelectedChange(false);
34575             }
34576             this.selNode = node;
34577             node.ui.onSelectedChange(true);
34578             this.fireEvent("selectionchange", this, node, last);
34579         }
34580         return node;
34581     },
34582     
34583     /**
34584      * Deselect a node.
34585      * @param {TreeNode} node The node to unselect
34586      */
34587     unselect : function(node){
34588         if(this.selNode == node){
34589             this.clearSelections();
34590         }    
34591     },
34592     
34593     /**
34594      * Clear all selections
34595      */
34596     clearSelections : function(){
34597         var n = this.selNode;
34598         if(n){
34599             n.ui.onSelectedChange(false);
34600             this.selNode = null;
34601             this.fireEvent("selectionchange", this, null);
34602         }
34603         return n;
34604     },
34605     
34606     /**
34607      * Get the selected node
34608      * @return {TreeNode} The selected node
34609      */
34610     getSelectedNode : function(){
34611         return this.selNode;    
34612     },
34613     
34614     /**
34615      * Returns true if the node is selected
34616      * @param {TreeNode} node The node to check
34617      * @return {Boolean}
34618      */
34619     isSelected : function(node){
34620         return this.selNode == node;  
34621     },
34622
34623     /**
34624      * Selects the node above the selected node in the tree, intelligently walking the nodes
34625      * @return TreeNode The new selection
34626      */
34627     selectPrevious : function(){
34628         var s = this.selNode || this.lastSelNode;
34629         if(!s){
34630             return null;
34631         }
34632         var ps = s.previousSibling;
34633         if(ps){
34634             if(!ps.isExpanded() || ps.childNodes.length < 1){
34635                 return this.select(ps);
34636             } else{
34637                 var lc = ps.lastChild;
34638                 while(lc && lc.isExpanded() && lc.childNodes.length > 0){
34639                     lc = lc.lastChild;
34640                 }
34641                 return this.select(lc);
34642             }
34643         } else if(s.parentNode && (this.tree.rootVisible || !s.parentNode.isRoot)){
34644             return this.select(s.parentNode);
34645         }
34646         return null;
34647     },
34648
34649     /**
34650      * Selects the node above the selected node in the tree, intelligently walking the nodes
34651      * @return TreeNode The new selection
34652      */
34653     selectNext : function(){
34654         var s = this.selNode || this.lastSelNode;
34655         if(!s){
34656             return null;
34657         }
34658         if(s.firstChild && s.isExpanded()){
34659              return this.select(s.firstChild);
34660          }else if(s.nextSibling){
34661              return this.select(s.nextSibling);
34662          }else if(s.parentNode){
34663             var newS = null;
34664             s.parentNode.bubble(function(){
34665                 if(this.nextSibling){
34666                     newS = this.getOwnerTree().selModel.select(this.nextSibling);
34667                     return false;
34668                 }
34669             });
34670             return newS;
34671          }
34672         return null;
34673     },
34674
34675     onKeyDown : function(e){
34676         var s = this.selNode || this.lastSelNode;
34677         // undesirable, but required
34678         var sm = this;
34679         if(!s){
34680             return;
34681         }
34682         var k = e.getKey();
34683         switch(k){
34684              case e.DOWN:
34685                  e.stopEvent();
34686                  this.selectNext();
34687              break;
34688              case e.UP:
34689                  e.stopEvent();
34690                  this.selectPrevious();
34691              break;
34692              case e.RIGHT:
34693                  e.preventDefault();
34694                  if(s.hasChildNodes()){
34695                      if(!s.isExpanded()){
34696                          s.expand();
34697                      }else if(s.firstChild){
34698                          this.select(s.firstChild, e);
34699                      }
34700                  }
34701              break;
34702              case e.LEFT:
34703                  e.preventDefault();
34704                  if(s.hasChildNodes() && s.isExpanded()){
34705                      s.collapse();
34706                  }else if(s.parentNode && (this.tree.rootVisible || s.parentNode != this.tree.getRootNode())){
34707                      this.select(s.parentNode, e);
34708                  }
34709              break;
34710         };
34711     }
34712 });
34713
34714 /**
34715  * @class Roo.tree.MultiSelectionModel
34716  * @extends Roo.util.Observable
34717  * Multi selection for a TreePanel.
34718  * @param {Object} cfg Configuration
34719  */
34720 Roo.tree.MultiSelectionModel = function(){
34721    this.selNodes = [];
34722    this.selMap = {};
34723    this.addEvents({
34724        /**
34725         * @event selectionchange
34726         * Fires when the selected nodes change
34727         * @param {MultiSelectionModel} this
34728         * @param {Array} nodes Array of the selected nodes
34729         */
34730        "selectionchange" : true
34731    });
34732    Roo.tree.MultiSelectionModel.superclass.constructor.call(this,cfg);
34733    
34734 };
34735
34736 Roo.extend(Roo.tree.MultiSelectionModel, Roo.util.Observable, {
34737     init : function(tree){
34738         this.tree = tree;
34739         tree.getTreeEl().on("keydown", this.onKeyDown, this);
34740         tree.on("click", this.onNodeClick, this);
34741     },
34742     
34743     onNodeClick : function(node, e){
34744         this.select(node, e, e.ctrlKey);
34745     },
34746     
34747     /**
34748      * Select a node.
34749      * @param {TreeNode} node The node to select
34750      * @param {EventObject} e (optional) An event associated with the selection
34751      * @param {Boolean} keepExisting True to retain existing selections
34752      * @return {TreeNode} The selected node
34753      */
34754     select : function(node, e, keepExisting){
34755         if(keepExisting !== true){
34756             this.clearSelections(true);
34757         }
34758         if(this.isSelected(node)){
34759             this.lastSelNode = node;
34760             return node;
34761         }
34762         this.selNodes.push(node);
34763         this.selMap[node.id] = node;
34764         this.lastSelNode = node;
34765         node.ui.onSelectedChange(true);
34766         this.fireEvent("selectionchange", this, this.selNodes);
34767         return node;
34768     },
34769     
34770     /**
34771      * Deselect a node.
34772      * @param {TreeNode} node The node to unselect
34773      */
34774     unselect : function(node){
34775         if(this.selMap[node.id]){
34776             node.ui.onSelectedChange(false);
34777             var sn = this.selNodes;
34778             var index = -1;
34779             if(sn.indexOf){
34780                 index = sn.indexOf(node);
34781             }else{
34782                 for(var i = 0, len = sn.length; i < len; i++){
34783                     if(sn[i] == node){
34784                         index = i;
34785                         break;
34786                     }
34787                 }
34788             }
34789             if(index != -1){
34790                 this.selNodes.splice(index, 1);
34791             }
34792             delete this.selMap[node.id];
34793             this.fireEvent("selectionchange", this, this.selNodes);
34794         }
34795     },
34796     
34797     /**
34798      * Clear all selections
34799      */
34800     clearSelections : function(suppressEvent){
34801         var sn = this.selNodes;
34802         if(sn.length > 0){
34803             for(var i = 0, len = sn.length; i < len; i++){
34804                 sn[i].ui.onSelectedChange(false);
34805             }
34806             this.selNodes = [];
34807             this.selMap = {};
34808             if(suppressEvent !== true){
34809                 this.fireEvent("selectionchange", this, this.selNodes);
34810             }
34811         }
34812     },
34813     
34814     /**
34815      * Returns true if the node is selected
34816      * @param {TreeNode} node The node to check
34817      * @return {Boolean}
34818      */
34819     isSelected : function(node){
34820         return this.selMap[node.id] ? true : false;  
34821     },
34822     
34823     /**
34824      * Returns an array of the selected nodes
34825      * @return {Array}
34826      */
34827     getSelectedNodes : function(){
34828         return this.selNodes;    
34829     },
34830
34831     onKeyDown : Roo.tree.DefaultSelectionModel.prototype.onKeyDown,
34832
34833     selectNext : Roo.tree.DefaultSelectionModel.prototype.selectNext,
34834
34835     selectPrevious : Roo.tree.DefaultSelectionModel.prototype.selectPrevious
34836 });/*
34837  * Based on:
34838  * Ext JS Library 1.1.1
34839  * Copyright(c) 2006-2007, Ext JS, LLC.
34840  *
34841  * Originally Released Under LGPL - original licence link has changed is not relivant.
34842  *
34843  * Fork - LGPL
34844  * <script type="text/javascript">
34845  */
34846  
34847 /**
34848  * @class Roo.tree.TreeNode
34849  * @extends Roo.data.Node
34850  * @cfg {String} text The text for this node
34851  * @cfg {Boolean} expanded true to start the node expanded
34852  * @cfg {Boolean} allowDrag false to make this node undraggable if DD is on (defaults to true)
34853  * @cfg {Boolean} allowDrop false if this node cannot be drop on
34854  * @cfg {Boolean} disabled true to start the node disabled
34855  * @cfg {String} icon The path to an icon for the node. The preferred way to do this
34856  *    is to use the cls or iconCls attributes and add the icon via a CSS background image.
34857  * @cfg {String} cls A css class to be added to the node
34858  * @cfg {String} iconCls A css class to be added to the nodes icon element for applying css background images
34859  * @cfg {String} href URL of the link used for the node (defaults to #)
34860  * @cfg {String} hrefTarget target frame for the link
34861  * @cfg {String} qtip An Ext QuickTip for the node
34862  * @cfg {String} qtipCfg An Ext QuickTip config for the node (used instead of qtip)
34863  * @cfg {Boolean} singleClickExpand True for single click expand on this node
34864  * @cfg {Function} uiProvider A UI <b>class</b> to use for this node (defaults to Roo.tree.TreeNodeUI)
34865  * @cfg {Boolean} checked True to render a checked checkbox for this node, false to render an unchecked checkbox
34866  * (defaults to undefined with no checkbox rendered)
34867  * @constructor
34868  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node
34869  */
34870 Roo.tree.TreeNode = function(attributes){
34871     attributes = attributes || {};
34872     if(typeof attributes == "string"){
34873         attributes = {text: attributes};
34874     }
34875     this.childrenRendered = false;
34876     this.rendered = false;
34877     Roo.tree.TreeNode.superclass.constructor.call(this, attributes);
34878     this.expanded = attributes.expanded === true;
34879     this.isTarget = attributes.isTarget !== false;
34880     this.draggable = attributes.draggable !== false && attributes.allowDrag !== false;
34881     this.allowChildren = attributes.allowChildren !== false && attributes.allowDrop !== false;
34882
34883     /**
34884      * Read-only. The text for this node. To change it use setText().
34885      * @type String
34886      */
34887     this.text = attributes.text;
34888     /**
34889      * True if this node is disabled.
34890      * @type Boolean
34891      */
34892     this.disabled = attributes.disabled === true;
34893
34894     this.addEvents({
34895         /**
34896         * @event textchange
34897         * Fires when the text for this node is changed
34898         * @param {Node} this This node
34899         * @param {String} text The new text
34900         * @param {String} oldText The old text
34901         */
34902         "textchange" : true,
34903         /**
34904         * @event beforeexpand
34905         * Fires before this node is expanded, return false to cancel.
34906         * @param {Node} this This node
34907         * @param {Boolean} deep
34908         * @param {Boolean} anim
34909         */
34910         "beforeexpand" : true,
34911         /**
34912         * @event beforecollapse
34913         * Fires before this node is collapsed, return false to cancel.
34914         * @param {Node} this This node
34915         * @param {Boolean} deep
34916         * @param {Boolean} anim
34917         */
34918         "beforecollapse" : true,
34919         /**
34920         * @event expand
34921         * Fires when this node is expanded
34922         * @param {Node} this This node
34923         */
34924         "expand" : true,
34925         /**
34926         * @event disabledchange
34927         * Fires when the disabled status of this node changes
34928         * @param {Node} this This node
34929         * @param {Boolean} disabled
34930         */
34931         "disabledchange" : true,
34932         /**
34933         * @event collapse
34934         * Fires when this node is collapsed
34935         * @param {Node} this This node
34936         */
34937         "collapse" : true,
34938         /**
34939         * @event beforeclick
34940         * Fires before click processing. Return false to cancel the default action.
34941         * @param {Node} this This node
34942         * @param {Roo.EventObject} e The event object
34943         */
34944         "beforeclick":true,
34945         /**
34946         * @event checkchange
34947         * Fires when a node with a checkbox's checked property changes
34948         * @param {Node} this This node
34949         * @param {Boolean} checked
34950         */
34951         "checkchange":true,
34952         /**
34953         * @event click
34954         * Fires when this node is clicked
34955         * @param {Node} this This node
34956         * @param {Roo.EventObject} e The event object
34957         */
34958         "click":true,
34959         /**
34960         * @event dblclick
34961         * Fires when this node is double clicked
34962         * @param {Node} this This node
34963         * @param {Roo.EventObject} e The event object
34964         */
34965         "dblclick":true,
34966         /**
34967         * @event contextmenu
34968         * Fires when this node is right clicked
34969         * @param {Node} this This node
34970         * @param {Roo.EventObject} e The event object
34971         */
34972         "contextmenu":true,
34973         /**
34974         * @event beforechildrenrendered
34975         * Fires right before the child nodes for this node are rendered
34976         * @param {Node} this This node
34977         */
34978         "beforechildrenrendered":true
34979     });
34980
34981     var uiClass = this.attributes.uiProvider || Roo.tree.TreeNodeUI;
34982
34983     /**
34984      * Read-only. The UI for this node
34985      * @type TreeNodeUI
34986      */
34987     this.ui = new uiClass(this);
34988     
34989     // finally support items[]
34990     if (typeof(this.attributes.items) == 'undefined' || !this.attributes.items) {
34991         return;
34992     }
34993     
34994     
34995     Roo.each(this.attributes.items, function(c) {
34996         this.appendChild(Roo.factory(c,Roo.Tree));
34997     }, this);
34998     delete this.attributes.items;
34999     
35000     
35001     
35002 };
35003 Roo.extend(Roo.tree.TreeNode, Roo.data.Node, {
35004     preventHScroll: true,
35005     /**
35006      * Returns true if this node is expanded
35007      * @return {Boolean}
35008      */
35009     isExpanded : function(){
35010         return this.expanded;
35011     },
35012
35013     /**
35014      * Returns the UI object for this node
35015      * @return {TreeNodeUI}
35016      */
35017     getUI : function(){
35018         return this.ui;
35019     },
35020
35021     // private override
35022     setFirstChild : function(node){
35023         var of = this.firstChild;
35024         Roo.tree.TreeNode.superclass.setFirstChild.call(this, node);
35025         if(this.childrenRendered && of && node != of){
35026             of.renderIndent(true, true);
35027         }
35028         if(this.rendered){
35029             this.renderIndent(true, true);
35030         }
35031     },
35032
35033     // private override
35034     setLastChild : function(node){
35035         var ol = this.lastChild;
35036         Roo.tree.TreeNode.superclass.setLastChild.call(this, node);
35037         if(this.childrenRendered && ol && node != ol){
35038             ol.renderIndent(true, true);
35039         }
35040         if(this.rendered){
35041             this.renderIndent(true, true);
35042         }
35043     },
35044
35045     // these methods are overridden to provide lazy rendering support
35046     // private override
35047     appendChild : function()
35048     {
35049         var node = Roo.tree.TreeNode.superclass.appendChild.apply(this, arguments);
35050         if(node && this.childrenRendered){
35051             node.render();
35052         }
35053         this.ui.updateExpandIcon();
35054         return node;
35055     },
35056
35057     // private override
35058     removeChild : function(node){
35059         this.ownerTree.getSelectionModel().unselect(node);
35060         Roo.tree.TreeNode.superclass.removeChild.apply(this, arguments);
35061         // if it's been rendered remove dom node
35062         if(this.childrenRendered){
35063             node.ui.remove();
35064         }
35065         if(this.childNodes.length < 1){
35066             this.collapse(false, false);
35067         }else{
35068             this.ui.updateExpandIcon();
35069         }
35070         if(!this.firstChild) {
35071             this.childrenRendered = false;
35072         }
35073         return node;
35074     },
35075
35076     // private override
35077     insertBefore : function(node, refNode){
35078         var newNode = Roo.tree.TreeNode.superclass.insertBefore.apply(this, arguments);
35079         if(newNode && refNode && this.childrenRendered){
35080             node.render();
35081         }
35082         this.ui.updateExpandIcon();
35083         return newNode;
35084     },
35085
35086     /**
35087      * Sets the text for this node
35088      * @param {String} text
35089      */
35090     setText : function(text){
35091         var oldText = this.text;
35092         this.text = text;
35093         this.attributes.text = text;
35094         if(this.rendered){ // event without subscribing
35095             this.ui.onTextChange(this, text, oldText);
35096         }
35097         this.fireEvent("textchange", this, text, oldText);
35098     },
35099
35100     /**
35101      * Triggers selection of this node
35102      */
35103     select : function(){
35104         this.getOwnerTree().getSelectionModel().select(this);
35105     },
35106
35107     /**
35108      * Triggers deselection of this node
35109      */
35110     unselect : function(){
35111         this.getOwnerTree().getSelectionModel().unselect(this);
35112     },
35113
35114     /**
35115      * Returns true if this node is selected
35116      * @return {Boolean}
35117      */
35118     isSelected : function(){
35119         return this.getOwnerTree().getSelectionModel().isSelected(this);
35120     },
35121
35122     /**
35123      * Expand this node.
35124      * @param {Boolean} deep (optional) True to expand all children as well
35125      * @param {Boolean} anim (optional) false to cancel the default animation
35126      * @param {Function} callback (optional) A callback to be called when
35127      * expanding this node completes (does not wait for deep expand to complete).
35128      * Called with 1 parameter, this node.
35129      */
35130     expand : function(deep, anim, callback){
35131         if(!this.expanded){
35132             if(this.fireEvent("beforeexpand", this, deep, anim) === false){
35133                 return;
35134             }
35135             if(!this.childrenRendered){
35136                 this.renderChildren();
35137             }
35138             this.expanded = true;
35139             
35140             if(!this.isHiddenRoot() && (this.getOwnerTree() && this.getOwnerTree().animate && anim !== false) || anim){
35141                 this.ui.animExpand(function(){
35142                     this.fireEvent("expand", this);
35143                     if(typeof callback == "function"){
35144                         callback(this);
35145                     }
35146                     if(deep === true){
35147                         this.expandChildNodes(true);
35148                     }
35149                 }.createDelegate(this));
35150                 return;
35151             }else{
35152                 this.ui.expand();
35153                 this.fireEvent("expand", this);
35154                 if(typeof callback == "function"){
35155                     callback(this);
35156                 }
35157             }
35158         }else{
35159            if(typeof callback == "function"){
35160                callback(this);
35161            }
35162         }
35163         if(deep === true){
35164             this.expandChildNodes(true);
35165         }
35166     },
35167
35168     isHiddenRoot : function(){
35169         return this.isRoot && !this.getOwnerTree().rootVisible;
35170     },
35171
35172     /**
35173      * Collapse this node.
35174      * @param {Boolean} deep (optional) True to collapse all children as well
35175      * @param {Boolean} anim (optional) false to cancel the default animation
35176      */
35177     collapse : function(deep, anim){
35178         if(this.expanded && !this.isHiddenRoot()){
35179             if(this.fireEvent("beforecollapse", this, deep, anim) === false){
35180                 return;
35181             }
35182             this.expanded = false;
35183             if((this.getOwnerTree().animate && anim !== false) || anim){
35184                 this.ui.animCollapse(function(){
35185                     this.fireEvent("collapse", this);
35186                     if(deep === true){
35187                         this.collapseChildNodes(true);
35188                     }
35189                 }.createDelegate(this));
35190                 return;
35191             }else{
35192                 this.ui.collapse();
35193                 this.fireEvent("collapse", this);
35194             }
35195         }
35196         if(deep === true){
35197             var cs = this.childNodes;
35198             for(var i = 0, len = cs.length; i < len; i++) {
35199                 cs[i].collapse(true, false);
35200             }
35201         }
35202     },
35203
35204     // private
35205     delayedExpand : function(delay){
35206         if(!this.expandProcId){
35207             this.expandProcId = this.expand.defer(delay, this);
35208         }
35209     },
35210
35211     // private
35212     cancelExpand : function(){
35213         if(this.expandProcId){
35214             clearTimeout(this.expandProcId);
35215         }
35216         this.expandProcId = false;
35217     },
35218
35219     /**
35220      * Toggles expanded/collapsed state of the node
35221      */
35222     toggle : function(){
35223         if(this.expanded){
35224             this.collapse();
35225         }else{
35226             this.expand();
35227         }
35228     },
35229
35230     /**
35231      * Ensures all parent nodes are expanded
35232      */
35233     ensureVisible : function(callback){
35234         var tree = this.getOwnerTree();
35235         tree.expandPath(this.parentNode.getPath(), false, function(){
35236             tree.getTreeEl().scrollChildIntoView(this.ui.anchor);
35237             Roo.callback(callback);
35238         }.createDelegate(this));
35239     },
35240
35241     /**
35242      * Expand all child nodes
35243      * @param {Boolean} deep (optional) true if the child nodes should also expand their child nodes
35244      */
35245     expandChildNodes : function(deep){
35246         var cs = this.childNodes;
35247         for(var i = 0, len = cs.length; i < len; i++) {
35248                 cs[i].expand(deep);
35249         }
35250     },
35251
35252     /**
35253      * Collapse all child nodes
35254      * @param {Boolean} deep (optional) true if the child nodes should also collapse their child nodes
35255      */
35256     collapseChildNodes : function(deep){
35257         var cs = this.childNodes;
35258         for(var i = 0, len = cs.length; i < len; i++) {
35259                 cs[i].collapse(deep);
35260         }
35261     },
35262
35263     /**
35264      * Disables this node
35265      */
35266     disable : function(){
35267         this.disabled = true;
35268         this.unselect();
35269         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
35270             this.ui.onDisableChange(this, true);
35271         }
35272         this.fireEvent("disabledchange", this, true);
35273     },
35274
35275     /**
35276      * Enables this node
35277      */
35278     enable : function(){
35279         this.disabled = false;
35280         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
35281             this.ui.onDisableChange(this, false);
35282         }
35283         this.fireEvent("disabledchange", this, false);
35284     },
35285
35286     // private
35287     renderChildren : function(suppressEvent){
35288         if(suppressEvent !== false){
35289             this.fireEvent("beforechildrenrendered", this);
35290         }
35291         var cs = this.childNodes;
35292         for(var i = 0, len = cs.length; i < len; i++){
35293             cs[i].render(true);
35294         }
35295         this.childrenRendered = true;
35296     },
35297
35298     // private
35299     sort : function(fn, scope){
35300         Roo.tree.TreeNode.superclass.sort.apply(this, arguments);
35301         if(this.childrenRendered){
35302             var cs = this.childNodes;
35303             for(var i = 0, len = cs.length; i < len; i++){
35304                 cs[i].render(true);
35305             }
35306         }
35307     },
35308
35309     // private
35310     render : function(bulkRender){
35311         this.ui.render(bulkRender);
35312         if(!this.rendered){
35313             this.rendered = true;
35314             if(this.expanded){
35315                 this.expanded = false;
35316                 this.expand(false, false);
35317             }
35318         }
35319     },
35320
35321     // private
35322     renderIndent : function(deep, refresh){
35323         if(refresh){
35324             this.ui.childIndent = null;
35325         }
35326         this.ui.renderIndent();
35327         if(deep === true && this.childrenRendered){
35328             var cs = this.childNodes;
35329             for(var i = 0, len = cs.length; i < len; i++){
35330                 cs[i].renderIndent(true, refresh);
35331             }
35332         }
35333     }
35334 });/*
35335  * Based on:
35336  * Ext JS Library 1.1.1
35337  * Copyright(c) 2006-2007, Ext JS, LLC.
35338  *
35339  * Originally Released Under LGPL - original licence link has changed is not relivant.
35340  *
35341  * Fork - LGPL
35342  * <script type="text/javascript">
35343  */
35344  
35345 /**
35346  * @class Roo.tree.AsyncTreeNode
35347  * @extends Roo.tree.TreeNode
35348  * @cfg {TreeLoader} loader A TreeLoader to be used by this node (defaults to the loader defined on the tree)
35349  * @constructor
35350  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node 
35351  */
35352  Roo.tree.AsyncTreeNode = function(config){
35353     this.loaded = false;
35354     this.loading = false;
35355     Roo.tree.AsyncTreeNode.superclass.constructor.apply(this, arguments);
35356     /**
35357     * @event beforeload
35358     * Fires before this node is loaded, return false to cancel
35359     * @param {Node} this This node
35360     */
35361     this.addEvents({'beforeload':true, 'load': true});
35362     /**
35363     * @event load
35364     * Fires when this node is loaded
35365     * @param {Node} this This node
35366     */
35367     /**
35368      * The loader used by this node (defaults to using the tree's defined loader)
35369      * @type TreeLoader
35370      * @property loader
35371      */
35372 };
35373 Roo.extend(Roo.tree.AsyncTreeNode, Roo.tree.TreeNode, {
35374     expand : function(deep, anim, callback){
35375         if(this.loading){ // if an async load is already running, waiting til it's done
35376             var timer;
35377             var f = function(){
35378                 if(!this.loading){ // done loading
35379                     clearInterval(timer);
35380                     this.expand(deep, anim, callback);
35381                 }
35382             }.createDelegate(this);
35383             timer = setInterval(f, 200);
35384             return;
35385         }
35386         if(!this.loaded){
35387             if(this.fireEvent("beforeload", this) === false){
35388                 return;
35389             }
35390             this.loading = true;
35391             this.ui.beforeLoad(this);
35392             var loader = this.loader || this.attributes.loader || this.getOwnerTree().getLoader();
35393             if(loader){
35394                 loader.load(this, this.loadComplete.createDelegate(this, [deep, anim, callback]));
35395                 return;
35396             }
35397         }
35398         Roo.tree.AsyncTreeNode.superclass.expand.call(this, deep, anim, callback);
35399     },
35400     
35401     /**
35402      * Returns true if this node is currently loading
35403      * @return {Boolean}
35404      */
35405     isLoading : function(){
35406         return this.loading;  
35407     },
35408     
35409     loadComplete : function(deep, anim, callback){
35410         this.loading = false;
35411         this.loaded = true;
35412         this.ui.afterLoad(this);
35413         this.fireEvent("load", this);
35414         this.expand(deep, anim, callback);
35415     },
35416     
35417     /**
35418      * Returns true if this node has been loaded
35419      * @return {Boolean}
35420      */
35421     isLoaded : function(){
35422         return this.loaded;
35423     },
35424     
35425     hasChildNodes : function(){
35426         if(!this.isLeaf() && !this.loaded){
35427             return true;
35428         }else{
35429             return Roo.tree.AsyncTreeNode.superclass.hasChildNodes.call(this);
35430         }
35431     },
35432
35433     /**
35434      * Trigger a reload for this node
35435      * @param {Function} callback
35436      */
35437     reload : function(callback){
35438         this.collapse(false, false);
35439         while(this.firstChild){
35440             this.removeChild(this.firstChild);
35441         }
35442         this.childrenRendered = false;
35443         this.loaded = false;
35444         if(this.isHiddenRoot()){
35445             this.expanded = false;
35446         }
35447         this.expand(false, false, callback);
35448     }
35449 });/*
35450  * Based on:
35451  * Ext JS Library 1.1.1
35452  * Copyright(c) 2006-2007, Ext JS, LLC.
35453  *
35454  * Originally Released Under LGPL - original licence link has changed is not relivant.
35455  *
35456  * Fork - LGPL
35457  * <script type="text/javascript">
35458  */
35459  
35460 /**
35461  * @class Roo.tree.TreeNodeUI
35462  * @constructor
35463  * @param {Object} node The node to render
35464  * The TreeNode UI implementation is separate from the
35465  * tree implementation. Unless you are customizing the tree UI,
35466  * you should never have to use this directly.
35467  */
35468 Roo.tree.TreeNodeUI = function(node){
35469     this.node = node;
35470     this.rendered = false;
35471     this.animating = false;
35472     this.emptyIcon = Roo.BLANK_IMAGE_URL;
35473 };
35474
35475 Roo.tree.TreeNodeUI.prototype = {
35476     removeChild : function(node){
35477         if(this.rendered){
35478             this.ctNode.removeChild(node.ui.getEl());
35479         }
35480     },
35481
35482     beforeLoad : function(){
35483          this.addClass("x-tree-node-loading");
35484     },
35485
35486     afterLoad : function(){
35487          this.removeClass("x-tree-node-loading");
35488     },
35489
35490     onTextChange : function(node, text, oldText){
35491         if(this.rendered){
35492             this.textNode.innerHTML = text;
35493         }
35494     },
35495
35496     onDisableChange : function(node, state){
35497         this.disabled = state;
35498         if(state){
35499             this.addClass("x-tree-node-disabled");
35500         }else{
35501             this.removeClass("x-tree-node-disabled");
35502         }
35503     },
35504
35505     onSelectedChange : function(state){
35506         if(state){
35507             this.focus();
35508             this.addClass("x-tree-selected");
35509         }else{
35510             //this.blur();
35511             this.removeClass("x-tree-selected");
35512         }
35513     },
35514
35515     onMove : function(tree, node, oldParent, newParent, index, refNode){
35516         this.childIndent = null;
35517         if(this.rendered){
35518             var targetNode = newParent.ui.getContainer();
35519             if(!targetNode){//target not rendered
35520                 this.holder = document.createElement("div");
35521                 this.holder.appendChild(this.wrap);
35522                 return;
35523             }
35524             var insertBefore = refNode ? refNode.ui.getEl() : null;
35525             if(insertBefore){
35526                 targetNode.insertBefore(this.wrap, insertBefore);
35527             }else{
35528                 targetNode.appendChild(this.wrap);
35529             }
35530             this.node.renderIndent(true);
35531         }
35532     },
35533
35534     addClass : function(cls){
35535         if(this.elNode){
35536             Roo.fly(this.elNode).addClass(cls);
35537         }
35538     },
35539
35540     removeClass : function(cls){
35541         if(this.elNode){
35542             Roo.fly(this.elNode).removeClass(cls);
35543         }
35544     },
35545
35546     remove : function(){
35547         if(this.rendered){
35548             this.holder = document.createElement("div");
35549             this.holder.appendChild(this.wrap);
35550         }
35551     },
35552
35553     fireEvent : function(){
35554         return this.node.fireEvent.apply(this.node, arguments);
35555     },
35556
35557     initEvents : function(){
35558         this.node.on("move", this.onMove, this);
35559         var E = Roo.EventManager;
35560         var a = this.anchor;
35561
35562         var el = Roo.fly(a, '_treeui');
35563
35564         if(Roo.isOpera){ // opera render bug ignores the CSS
35565             el.setStyle("text-decoration", "none");
35566         }
35567
35568         el.on("click", this.onClick, this);
35569         el.on("dblclick", this.onDblClick, this);
35570
35571         if(this.checkbox){
35572             Roo.EventManager.on(this.checkbox,
35573                     Roo.isIE ? 'click' : 'change', this.onCheckChange, this);
35574         }
35575
35576         el.on("contextmenu", this.onContextMenu, this);
35577
35578         var icon = Roo.fly(this.iconNode);
35579         icon.on("click", this.onClick, this);
35580         icon.on("dblclick", this.onDblClick, this);
35581         icon.on("contextmenu", this.onContextMenu, this);
35582         E.on(this.ecNode, "click", this.ecClick, this, true);
35583
35584         if(this.node.disabled){
35585             this.addClass("x-tree-node-disabled");
35586         }
35587         if(this.node.hidden){
35588             this.addClass("x-tree-node-disabled");
35589         }
35590         var ot = this.node.getOwnerTree();
35591         var dd = ot.enableDD || ot.enableDrag || ot.enableDrop;
35592         if(dd && (!this.node.isRoot || ot.rootVisible)){
35593             Roo.dd.Registry.register(this.elNode, {
35594                 node: this.node,
35595                 handles: this.getDDHandles(),
35596                 isHandle: false
35597             });
35598         }
35599     },
35600
35601     getDDHandles : function(){
35602         return [this.iconNode, this.textNode];
35603     },
35604
35605     hide : function(){
35606         if(this.rendered){
35607             this.wrap.style.display = "none";
35608         }
35609     },
35610
35611     show : function(){
35612         if(this.rendered){
35613             this.wrap.style.display = "";
35614         }
35615     },
35616
35617     onContextMenu : function(e){
35618         if (this.node.hasListener("contextmenu") || this.node.getOwnerTree().hasListener("contextmenu")) {
35619             e.preventDefault();
35620             this.focus();
35621             this.fireEvent("contextmenu", this.node, e);
35622         }
35623     },
35624
35625     onClick : function(e){
35626         if(this.dropping){
35627             e.stopEvent();
35628             return;
35629         }
35630         if(this.fireEvent("beforeclick", this.node, e) !== false){
35631             if(!this.disabled && this.node.attributes.href){
35632                 this.fireEvent("click", this.node, e);
35633                 return;
35634             }
35635             e.preventDefault();
35636             if(this.disabled){
35637                 return;
35638             }
35639
35640             if(this.node.attributes.singleClickExpand && !this.animating && this.node.hasChildNodes()){
35641                 this.node.toggle();
35642             }
35643
35644             this.fireEvent("click", this.node, e);
35645         }else{
35646             e.stopEvent();
35647         }
35648     },
35649
35650     onDblClick : function(e){
35651         e.preventDefault();
35652         if(this.disabled){
35653             return;
35654         }
35655         if(this.checkbox){
35656             this.toggleCheck();
35657         }
35658         if(!this.animating && this.node.hasChildNodes()){
35659             this.node.toggle();
35660         }
35661         this.fireEvent("dblclick", this.node, e);
35662     },
35663
35664     onCheckChange : function(){
35665         var checked = this.checkbox.checked;
35666         this.node.attributes.checked = checked;
35667         this.fireEvent('checkchange', this.node, checked);
35668     },
35669
35670     ecClick : function(e){
35671         if(!this.animating && this.node.hasChildNodes()){
35672             this.node.toggle();
35673         }
35674     },
35675
35676     startDrop : function(){
35677         this.dropping = true;
35678     },
35679
35680     // delayed drop so the click event doesn't get fired on a drop
35681     endDrop : function(){
35682        setTimeout(function(){
35683            this.dropping = false;
35684        }.createDelegate(this), 50);
35685     },
35686
35687     expand : function(){
35688         this.updateExpandIcon();
35689         this.ctNode.style.display = "";
35690     },
35691
35692     focus : function(){
35693         if(!this.node.preventHScroll){
35694             try{this.anchor.focus();
35695             }catch(e){}
35696         }else if(!Roo.isIE){
35697             try{
35698                 var noscroll = this.node.getOwnerTree().getTreeEl().dom;
35699                 var l = noscroll.scrollLeft;
35700                 this.anchor.focus();
35701                 noscroll.scrollLeft = l;
35702             }catch(e){}
35703         }
35704     },
35705
35706     toggleCheck : function(value){
35707         var cb = this.checkbox;
35708         if(cb){
35709             cb.checked = (value === undefined ? !cb.checked : value);
35710         }
35711     },
35712
35713     blur : function(){
35714         try{
35715             this.anchor.blur();
35716         }catch(e){}
35717     },
35718
35719     animExpand : function(callback){
35720         var ct = Roo.get(this.ctNode);
35721         ct.stopFx();
35722         if(!this.node.hasChildNodes()){
35723             this.updateExpandIcon();
35724             this.ctNode.style.display = "";
35725             Roo.callback(callback);
35726             return;
35727         }
35728         this.animating = true;
35729         this.updateExpandIcon();
35730
35731         ct.slideIn('t', {
35732            callback : function(){
35733                this.animating = false;
35734                Roo.callback(callback);
35735             },
35736             scope: this,
35737             duration: this.node.ownerTree.duration || .25
35738         });
35739     },
35740
35741     highlight : function(){
35742         var tree = this.node.getOwnerTree();
35743         Roo.fly(this.wrap).highlight(
35744             tree.hlColor || "C3DAF9",
35745             {endColor: tree.hlBaseColor}
35746         );
35747     },
35748
35749     collapse : function(){
35750         this.updateExpandIcon();
35751         this.ctNode.style.display = "none";
35752     },
35753
35754     animCollapse : function(callback){
35755         var ct = Roo.get(this.ctNode);
35756         ct.enableDisplayMode('block');
35757         ct.stopFx();
35758
35759         this.animating = true;
35760         this.updateExpandIcon();
35761
35762         ct.slideOut('t', {
35763             callback : function(){
35764                this.animating = false;
35765                Roo.callback(callback);
35766             },
35767             scope: this,
35768             duration: this.node.ownerTree.duration || .25
35769         });
35770     },
35771
35772     getContainer : function(){
35773         return this.ctNode;
35774     },
35775
35776     getEl : function(){
35777         return this.wrap;
35778     },
35779
35780     appendDDGhost : function(ghostNode){
35781         ghostNode.appendChild(this.elNode.cloneNode(true));
35782     },
35783
35784     getDDRepairXY : function(){
35785         return Roo.lib.Dom.getXY(this.iconNode);
35786     },
35787
35788     onRender : function(){
35789         this.render();
35790     },
35791
35792     render : function(bulkRender){
35793         var n = this.node, a = n.attributes;
35794         var targetNode = n.parentNode ?
35795               n.parentNode.ui.getContainer() : n.ownerTree.innerCt.dom;
35796
35797         if(!this.rendered){
35798             this.rendered = true;
35799
35800             this.renderElements(n, a, targetNode, bulkRender);
35801
35802             if(a.qtip){
35803                if(this.textNode.setAttributeNS){
35804                    this.textNode.setAttributeNS("ext", "qtip", a.qtip);
35805                    if(a.qtipTitle){
35806                        this.textNode.setAttributeNS("ext", "qtitle", a.qtipTitle);
35807                    }
35808                }else{
35809                    this.textNode.setAttribute("ext:qtip", a.qtip);
35810                    if(a.qtipTitle){
35811                        this.textNode.setAttribute("ext:qtitle", a.qtipTitle);
35812                    }
35813                }
35814             }else if(a.qtipCfg){
35815                 a.qtipCfg.target = Roo.id(this.textNode);
35816                 Roo.QuickTips.register(a.qtipCfg);
35817             }
35818             this.initEvents();
35819             if(!this.node.expanded){
35820                 this.updateExpandIcon();
35821             }
35822         }else{
35823             if(bulkRender === true) {
35824                 targetNode.appendChild(this.wrap);
35825             }
35826         }
35827     },
35828
35829     renderElements : function(n, a, targetNode, bulkRender)
35830     {
35831         // add some indent caching, this helps performance when rendering a large tree
35832         this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
35833         var t = n.getOwnerTree();
35834         var txt = t.renderer ? t.renderer(n.attributes) : Roo.util.Format.htmlEncode(n.text);
35835         if (typeof(n.attributes.html) != 'undefined') {
35836             txt = n.attributes.html;
35837         }
35838         var tip = t.rendererTip ? t.rendererTip(n.attributes) : txt;
35839         var cb = typeof a.checked == 'boolean';
35840         var href = a.href ? a.href : Roo.isGecko ? "" : "#";
35841         var buf = ['<li class="x-tree-node"><div class="x-tree-node-el ', a.cls,'">',
35842             '<span class="x-tree-node-indent">',this.indentMarkup,"</span>",
35843             '<img src="', this.emptyIcon, '" class="x-tree-ec-icon" />',
35844             '<img src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',(a.icon ? " x-tree-node-inline-icon" : ""),(a.iconCls ? " "+a.iconCls : ""),'" unselectable="on" />',
35845             cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + (a.checked ? 'checked="checked" />' : ' />')) : '',
35846             '<a hidefocus="on" href="',href,'" tabIndex="1" ',
35847              a.hrefTarget ? ' target="'+a.hrefTarget+'"' : "", 
35848                 '><span unselectable="on" qtip="' , tip ,'">',txt,"</span></a></div>",
35849             '<ul class="x-tree-node-ct" style="display:none;"></ul>',
35850             "</li>"];
35851
35852         if(bulkRender !== true && n.nextSibling && n.nextSibling.ui.getEl()){
35853             this.wrap = Roo.DomHelper.insertHtml("beforeBegin",
35854                                 n.nextSibling.ui.getEl(), buf.join(""));
35855         }else{
35856             this.wrap = Roo.DomHelper.insertHtml("beforeEnd", targetNode, buf.join(""));
35857         }
35858
35859         this.elNode = this.wrap.childNodes[0];
35860         this.ctNode = this.wrap.childNodes[1];
35861         var cs = this.elNode.childNodes;
35862         this.indentNode = cs[0];
35863         this.ecNode = cs[1];
35864         this.iconNode = cs[2];
35865         var index = 3;
35866         if(cb){
35867             this.checkbox = cs[3];
35868             index++;
35869         }
35870         this.anchor = cs[index];
35871         this.textNode = cs[index].firstChild;
35872     },
35873
35874     getAnchor : function(){
35875         return this.anchor;
35876     },
35877
35878     getTextEl : function(){
35879         return this.textNode;
35880     },
35881
35882     getIconEl : function(){
35883         return this.iconNode;
35884     },
35885
35886     isChecked : function(){
35887         return this.checkbox ? this.checkbox.checked : false;
35888     },
35889
35890     updateExpandIcon : function(){
35891         if(this.rendered){
35892             var n = this.node, c1, c2;
35893             var cls = n.isLast() ? "x-tree-elbow-end" : "x-tree-elbow";
35894             var hasChild = n.hasChildNodes();
35895             if(hasChild){
35896                 if(n.expanded){
35897                     cls += "-minus";
35898                     c1 = "x-tree-node-collapsed";
35899                     c2 = "x-tree-node-expanded";
35900                 }else{
35901                     cls += "-plus";
35902                     c1 = "x-tree-node-expanded";
35903                     c2 = "x-tree-node-collapsed";
35904                 }
35905                 if(this.wasLeaf){
35906                     this.removeClass("x-tree-node-leaf");
35907                     this.wasLeaf = false;
35908                 }
35909                 if(this.c1 != c1 || this.c2 != c2){
35910                     Roo.fly(this.elNode).replaceClass(c1, c2);
35911                     this.c1 = c1; this.c2 = c2;
35912                 }
35913             }else{
35914                 // this changes non-leafs into leafs if they have no children.
35915                 // it's not very rational behaviour..
35916                 
35917                 if(!this.wasLeaf && this.node.leaf){
35918                     Roo.fly(this.elNode).replaceClass("x-tree-node-expanded", "x-tree-node-leaf");
35919                     delete this.c1;
35920                     delete this.c2;
35921                     this.wasLeaf = true;
35922                 }
35923             }
35924             var ecc = "x-tree-ec-icon "+cls;
35925             if(this.ecc != ecc){
35926                 this.ecNode.className = ecc;
35927                 this.ecc = ecc;
35928             }
35929         }
35930     },
35931
35932     getChildIndent : function(){
35933         if(!this.childIndent){
35934             var buf = [];
35935             var p = this.node;
35936             while(p){
35937                 if(!p.isRoot || (p.isRoot && p.ownerTree.rootVisible)){
35938                     if(!p.isLast()) {
35939                         buf.unshift('<img src="'+this.emptyIcon+'" class="x-tree-elbow-line" />');
35940                     } else {
35941                         buf.unshift('<img src="'+this.emptyIcon+'" class="x-tree-icon" />');
35942                     }
35943                 }
35944                 p = p.parentNode;
35945             }
35946             this.childIndent = buf.join("");
35947         }
35948         return this.childIndent;
35949     },
35950
35951     renderIndent : function(){
35952         if(this.rendered){
35953             var indent = "";
35954             var p = this.node.parentNode;
35955             if(p){
35956                 indent = p.ui.getChildIndent();
35957             }
35958             if(this.indentMarkup != indent){ // don't rerender if not required
35959                 this.indentNode.innerHTML = indent;
35960                 this.indentMarkup = indent;
35961             }
35962             this.updateExpandIcon();
35963         }
35964     }
35965 };
35966
35967 Roo.tree.RootTreeNodeUI = function(){
35968     Roo.tree.RootTreeNodeUI.superclass.constructor.apply(this, arguments);
35969 };
35970 Roo.extend(Roo.tree.RootTreeNodeUI, Roo.tree.TreeNodeUI, {
35971     render : function(){
35972         if(!this.rendered){
35973             var targetNode = this.node.ownerTree.innerCt.dom;
35974             this.node.expanded = true;
35975             targetNode.innerHTML = '<div class="x-tree-root-node"></div>';
35976             this.wrap = this.ctNode = targetNode.firstChild;
35977         }
35978     },
35979     collapse : function(){
35980     },
35981     expand : function(){
35982     }
35983 });/*
35984  * Based on:
35985  * Ext JS Library 1.1.1
35986  * Copyright(c) 2006-2007, Ext JS, LLC.
35987  *
35988  * Originally Released Under LGPL - original licence link has changed is not relivant.
35989  *
35990  * Fork - LGPL
35991  * <script type="text/javascript">
35992  */
35993 /**
35994  * @class Roo.tree.TreeLoader
35995  * @extends Roo.util.Observable
35996  * A TreeLoader provides for lazy loading of an {@link Roo.tree.TreeNode}'s child
35997  * nodes from a specified URL. The response must be a javascript Array definition
35998  * who's elements are node definition objects. eg:
35999  * <pre><code>
36000 {  success : true,
36001    data :      [
36002    
36003     { 'id': 1, 'text': 'A folder Node', 'leaf': false },
36004     { 'id': 2, 'text': 'A leaf Node', 'leaf': true }
36005     ]
36006 }
36007
36008
36009 </code></pre>
36010  * <br><br>
36011  * The old style respose with just an array is still supported, but not recommended.
36012  * <br><br>
36013  *
36014  * A server request is sent, and child nodes are loaded only when a node is expanded.
36015  * The loading node's id is passed to the server under the parameter name "node" to
36016  * enable the server to produce the correct child nodes.
36017  * <br><br>
36018  * To pass extra parameters, an event handler may be attached to the "beforeload"
36019  * event, and the parameters specified in the TreeLoader's baseParams property:
36020  * <pre><code>
36021     myTreeLoader.on("beforeload", function(treeLoader, node) {
36022         this.baseParams.category = node.attributes.category;
36023     }, this);
36024     
36025 </code></pre>
36026  *
36027  * This would pass an HTTP parameter called "category" to the server containing
36028  * the value of the Node's "category" attribute.
36029  * @constructor
36030  * Creates a new Treeloader.
36031  * @param {Object} config A config object containing config properties.
36032  */
36033 Roo.tree.TreeLoader = function(config){
36034     this.baseParams = {};
36035     this.requestMethod = "POST";
36036     Roo.apply(this, config);
36037
36038     this.addEvents({
36039     
36040         /**
36041          * @event beforeload
36042          * Fires before a network request is made to retrieve the Json text which specifies a node's children.
36043          * @param {Object} This TreeLoader object.
36044          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36045          * @param {Object} callback The callback function specified in the {@link #load} call.
36046          */
36047         beforeload : true,
36048         /**
36049          * @event load
36050          * Fires when the node has been successfuly loaded.
36051          * @param {Object} This TreeLoader object.
36052          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36053          * @param {Object} response The response object containing the data from the server.
36054          */
36055         load : true,
36056         /**
36057          * @event loadexception
36058          * Fires if the network request failed.
36059          * @param {Object} This TreeLoader object.
36060          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36061          * @param {Object} response The response object containing the data from the server.
36062          */
36063         loadexception : true,
36064         /**
36065          * @event create
36066          * Fires before a node is created, enabling you to return custom Node types 
36067          * @param {Object} This TreeLoader object.
36068          * @param {Object} attr - the data returned from the AJAX call (modify it to suit)
36069          */
36070         create : true
36071     });
36072
36073     Roo.tree.TreeLoader.superclass.constructor.call(this);
36074 };
36075
36076 Roo.extend(Roo.tree.TreeLoader, Roo.util.Observable, {
36077     /**
36078     * @cfg {String} dataUrl The URL from which to request a Json string which
36079     * specifies an array of node definition object representing the child nodes
36080     * to be loaded.
36081     */
36082     /**
36083     * @cfg {String} requestMethod either GET or POST
36084     * defaults to POST (due to BC)
36085     * to be loaded.
36086     */
36087     /**
36088     * @cfg {Object} baseParams (optional) An object containing properties which
36089     * specify HTTP parameters to be passed to each request for child nodes.
36090     */
36091     /**
36092     * @cfg {Object} baseAttrs (optional) An object containing attributes to be added to all nodes
36093     * created by this loader. If the attributes sent by the server have an attribute in this object,
36094     * they take priority.
36095     */
36096     /**
36097     * @cfg {Object} uiProviders (optional) An object containing properties which
36098     * 
36099     * DEPRECATED - use 'create' event handler to modify attributes - which affect creation.
36100     * specify custom {@link Roo.tree.TreeNodeUI} implementations. If the optional
36101     * <i>uiProvider</i> attribute of a returned child node is a string rather
36102     * than a reference to a TreeNodeUI implementation, this that string value
36103     * is used as a property name in the uiProviders object. You can define the provider named
36104     * 'default' , and this will be used for all nodes (if no uiProvider is delivered by the node data)
36105     */
36106     uiProviders : {},
36107
36108     /**
36109     * @cfg {Boolean} clearOnLoad (optional) Default to true. Remove previously existing
36110     * child nodes before loading.
36111     */
36112     clearOnLoad : true,
36113
36114     /**
36115     * @cfg {String} root (optional) Default to false. Use this to read data from an object 
36116     * property on loading, rather than expecting an array. (eg. more compatible to a standard
36117     * Grid query { data : [ .....] }
36118     */
36119     
36120     root : false,
36121      /**
36122     * @cfg {String} queryParam (optional) 
36123     * Name of the query as it will be passed on the querystring (defaults to 'node')
36124     * eg. the request will be ?node=[id]
36125     */
36126     
36127     
36128     queryParam: false,
36129     
36130     /**
36131      * Load an {@link Roo.tree.TreeNode} from the URL specified in the constructor.
36132      * This is called automatically when a node is expanded, but may be used to reload
36133      * a node (or append new children if the {@link #clearOnLoad} option is false.)
36134      * @param {Roo.tree.TreeNode} node
36135      * @param {Function} callback
36136      */
36137     load : function(node, callback){
36138         if(this.clearOnLoad){
36139             while(node.firstChild){
36140                 node.removeChild(node.firstChild);
36141             }
36142         }
36143         if(node.attributes.children){ // preloaded json children
36144             var cs = node.attributes.children;
36145             for(var i = 0, len = cs.length; i < len; i++){
36146                 Roo.log('appendchild');
36147                 Roo.log(cs[i]);
36148                 node.appendChild(this.createNode(cs[i]));
36149             }
36150             if(typeof callback == "function"){
36151                 callback();
36152             }
36153         }else if(this.dataUrl){
36154             this.requestData(node, callback);
36155         }
36156     },
36157
36158     getParams: function(node){
36159         var buf = [], bp = this.baseParams;
36160         for(var key in bp){
36161             if(typeof bp[key] != "function"){
36162                 buf.push(encodeURIComponent(key), "=", encodeURIComponent(bp[key]), "&");
36163             }
36164         }
36165         var n = this.queryParam === false ? 'node' : this.queryParam;
36166         buf.push(n + "=", encodeURIComponent(node.id));
36167         return buf.join("");
36168     },
36169
36170     requestData : function(node, callback){
36171         if(this.fireEvent("beforeload", this, node, callback) !== false){
36172             this.transId = Roo.Ajax.request({
36173                 method:this.requestMethod,
36174                 url: this.dataUrl||this.url,
36175                 success: this.handleResponse,
36176                 failure: this.handleFailure,
36177                 scope: this,
36178                 argument: {callback: callback, node: node},
36179                 params: this.getParams(node)
36180             });
36181         }else{
36182             // if the load is cancelled, make sure we notify
36183             // the node that we are done
36184             if(typeof callback == "function"){
36185                 callback();
36186             }
36187         }
36188     },
36189
36190     isLoading : function(){
36191         return this.transId ? true : false;
36192     },
36193
36194     abort : function(){
36195         if(this.isLoading()){
36196             Roo.Ajax.abort(this.transId);
36197         }
36198     },
36199
36200     // private
36201     createNode : function(attr)
36202     {
36203         // apply baseAttrs, nice idea Corey!
36204         if(this.baseAttrs){
36205             Roo.applyIf(attr, this.baseAttrs);
36206         }
36207         if(this.applyLoader !== false){
36208             attr.loader = this;
36209         }
36210         // uiProvider = depreciated..
36211         
36212         if(typeof(attr.uiProvider) == 'string'){
36213            attr.uiProvider = this.uiProviders[attr.uiProvider] || 
36214                 /**  eval:var:attr */ eval(attr.uiProvider);
36215         }
36216         if(typeof(this.uiProviders['default']) != 'undefined') {
36217             attr.uiProvider = this.uiProviders['default'];
36218         }
36219         
36220         this.fireEvent('create', this, attr);
36221         
36222         attr.leaf  = typeof(attr.leaf) == 'string' ? attr.leaf * 1 : attr.leaf;
36223         return(attr.leaf ?
36224                         new Roo.tree.TreeNode(attr) :
36225                         new Roo.tree.AsyncTreeNode(attr));
36226     },
36227
36228     processResponse : function(response, node, callback)
36229     {
36230         var json = response.responseText;
36231         try {
36232             
36233             var o = Roo.decode(json);
36234             
36235             if (this.root === false && typeof(o.success) != undefined) {
36236                 this.root = 'data'; // the default behaviour for list like data..
36237                 }
36238                 
36239             if (this.root !== false &&  !o.success) {
36240                 // it's a failure condition.
36241                 var a = response.argument;
36242                 this.fireEvent("loadexception", this, a.node, response);
36243                 Roo.log("Load failed - should have a handler really");
36244                 return;
36245             }
36246             
36247             
36248             
36249             if (this.root !== false) {
36250                  o = o[this.root];
36251             }
36252             
36253             for(var i = 0, len = o.length; i < len; i++){
36254                 var n = this.createNode(o[i]);
36255                 if(n){
36256                     node.appendChild(n);
36257                 }
36258             }
36259             if(typeof callback == "function"){
36260                 callback(this, node);
36261             }
36262         }catch(e){
36263             this.handleFailure(response);
36264         }
36265     },
36266
36267     handleResponse : function(response){
36268         this.transId = false;
36269         var a = response.argument;
36270         this.processResponse(response, a.node, a.callback);
36271         this.fireEvent("load", this, a.node, response);
36272     },
36273
36274     handleFailure : function(response)
36275     {
36276         // should handle failure better..
36277         this.transId = false;
36278         var a = response.argument;
36279         this.fireEvent("loadexception", this, a.node, response);
36280         if(typeof a.callback == "function"){
36281             a.callback(this, a.node);
36282         }
36283     }
36284 });/*
36285  * Based on:
36286  * Ext JS Library 1.1.1
36287  * Copyright(c) 2006-2007, Ext JS, LLC.
36288  *
36289  * Originally Released Under LGPL - original licence link has changed is not relivant.
36290  *
36291  * Fork - LGPL
36292  * <script type="text/javascript">
36293  */
36294
36295 /**
36296 * @class Roo.tree.TreeFilter
36297 * Note this class is experimental and doesn't update the indent (lines) or expand collapse icons of the nodes
36298 * @param {TreePanel} tree
36299 * @param {Object} config (optional)
36300  */
36301 Roo.tree.TreeFilter = function(tree, config){
36302     this.tree = tree;
36303     this.filtered = {};
36304     Roo.apply(this, config);
36305 };
36306
36307 Roo.tree.TreeFilter.prototype = {
36308     clearBlank:false,
36309     reverse:false,
36310     autoClear:false,
36311     remove:false,
36312
36313      /**
36314      * Filter the data by a specific attribute.
36315      * @param {String/RegExp} value Either string that the attribute value
36316      * should start with or a RegExp to test against the attribute
36317      * @param {String} attr (optional) The attribute passed in your node's attributes collection. Defaults to "text".
36318      * @param {TreeNode} startNode (optional) The node to start the filter at.
36319      */
36320     filter : function(value, attr, startNode){
36321         attr = attr || "text";
36322         var f;
36323         if(typeof value == "string"){
36324             var vlen = value.length;
36325             // auto clear empty filter
36326             if(vlen == 0 && this.clearBlank){
36327                 this.clear();
36328                 return;
36329             }
36330             value = value.toLowerCase();
36331             f = function(n){
36332                 return n.attributes[attr].substr(0, vlen).toLowerCase() == value;
36333             };
36334         }else if(value.exec){ // regex?
36335             f = function(n){
36336                 return value.test(n.attributes[attr]);
36337             };
36338         }else{
36339             throw 'Illegal filter type, must be string or regex';
36340         }
36341         this.filterBy(f, null, startNode);
36342         },
36343
36344     /**
36345      * Filter by a function. The passed function will be called with each
36346      * node in the tree (or from the startNode). If the function returns true, the node is kept
36347      * otherwise it is filtered. If a node is filtered, its children are also filtered.
36348      * @param {Function} fn The filter function
36349      * @param {Object} scope (optional) The scope of the function (defaults to the current node)
36350      */
36351     filterBy : function(fn, scope, startNode){
36352         startNode = startNode || this.tree.root;
36353         if(this.autoClear){
36354             this.clear();
36355         }
36356         var af = this.filtered, rv = this.reverse;
36357         var f = function(n){
36358             if(n == startNode){
36359                 return true;
36360             }
36361             if(af[n.id]){
36362                 return false;
36363             }
36364             var m = fn.call(scope || n, n);
36365             if(!m || rv){
36366                 af[n.id] = n;
36367                 n.ui.hide();
36368                 return false;
36369             }
36370             return true;
36371         };
36372         startNode.cascade(f);
36373         if(this.remove){
36374            for(var id in af){
36375                if(typeof id != "function"){
36376                    var n = af[id];
36377                    if(n && n.parentNode){
36378                        n.parentNode.removeChild(n);
36379                    }
36380                }
36381            }
36382         }
36383     },
36384
36385     /**
36386      * Clears the current filter. Note: with the "remove" option
36387      * set a filter cannot be cleared.
36388      */
36389     clear : function(){
36390         var t = this.tree;
36391         var af = this.filtered;
36392         for(var id in af){
36393             if(typeof id != "function"){
36394                 var n = af[id];
36395                 if(n){
36396                     n.ui.show();
36397                 }
36398             }
36399         }
36400         this.filtered = {};
36401     }
36402 };
36403 /*
36404  * Based on:
36405  * Ext JS Library 1.1.1
36406  * Copyright(c) 2006-2007, Ext JS, LLC.
36407  *
36408  * Originally Released Under LGPL - original licence link has changed is not relivant.
36409  *
36410  * Fork - LGPL
36411  * <script type="text/javascript">
36412  */
36413  
36414
36415 /**
36416  * @class Roo.tree.TreeSorter
36417  * Provides sorting of nodes in a TreePanel
36418  * 
36419  * @cfg {Boolean} folderSort True to sort leaf nodes under non leaf nodes
36420  * @cfg {String} property The named attribute on the node to sort by (defaults to text)
36421  * @cfg {String} dir The direction to sort (asc or desc) (defaults to asc)
36422  * @cfg {String} leafAttr The attribute used to determine leaf nodes in folder sort (defaults to "leaf")
36423  * @cfg {Boolean} caseSensitive true for case sensitive sort (defaults to false)
36424  * @cfg {Function} sortType A custom "casting" function used to convert node values before sorting
36425  * @constructor
36426  * @param {TreePanel} tree
36427  * @param {Object} config
36428  */
36429 Roo.tree.TreeSorter = function(tree, config){
36430     Roo.apply(this, config);
36431     tree.on("beforechildrenrendered", this.doSort, this);
36432     tree.on("append", this.updateSort, this);
36433     tree.on("insert", this.updateSort, this);
36434     
36435     var dsc = this.dir && this.dir.toLowerCase() == "desc";
36436     var p = this.property || "text";
36437     var sortType = this.sortType;
36438     var fs = this.folderSort;
36439     var cs = this.caseSensitive === true;
36440     var leafAttr = this.leafAttr || 'leaf';
36441
36442     this.sortFn = function(n1, n2){
36443         if(fs){
36444             if(n1.attributes[leafAttr] && !n2.attributes[leafAttr]){
36445                 return 1;
36446             }
36447             if(!n1.attributes[leafAttr] && n2.attributes[leafAttr]){
36448                 return -1;
36449             }
36450         }
36451         var v1 = sortType ? sortType(n1) : (cs ? n1.attributes[p] : n1.attributes[p].toUpperCase());
36452         var v2 = sortType ? sortType(n2) : (cs ? n2.attributes[p] : n2.attributes[p].toUpperCase());
36453         if(v1 < v2){
36454                         return dsc ? +1 : -1;
36455                 }else if(v1 > v2){
36456                         return dsc ? -1 : +1;
36457         }else{
36458                 return 0;
36459         }
36460     };
36461 };
36462
36463 Roo.tree.TreeSorter.prototype = {
36464     doSort : function(node){
36465         node.sort(this.sortFn);
36466     },
36467     
36468     compareNodes : function(n1, n2){
36469         return (n1.text.toUpperCase() > n2.text.toUpperCase() ? 1 : -1);
36470     },
36471     
36472     updateSort : function(tree, node){
36473         if(node.childrenRendered){
36474             this.doSort.defer(1, this, [node]);
36475         }
36476     }
36477 };/*
36478  * Based on:
36479  * Ext JS Library 1.1.1
36480  * Copyright(c) 2006-2007, Ext JS, LLC.
36481  *
36482  * Originally Released Under LGPL - original licence link has changed is not relivant.
36483  *
36484  * Fork - LGPL
36485  * <script type="text/javascript">
36486  */
36487
36488 if(Roo.dd.DropZone){
36489     
36490 Roo.tree.TreeDropZone = function(tree, config){
36491     this.allowParentInsert = false;
36492     this.allowContainerDrop = false;
36493     this.appendOnly = false;
36494     Roo.tree.TreeDropZone.superclass.constructor.call(this, tree.innerCt, config);
36495     this.tree = tree;
36496     this.lastInsertClass = "x-tree-no-status";
36497     this.dragOverData = {};
36498 };
36499
36500 Roo.extend(Roo.tree.TreeDropZone, Roo.dd.DropZone, {
36501     ddGroup : "TreeDD",
36502     scroll:  true,
36503     
36504     expandDelay : 1000,
36505     
36506     expandNode : function(node){
36507         if(node.hasChildNodes() && !node.isExpanded()){
36508             node.expand(false, null, this.triggerCacheRefresh.createDelegate(this));
36509         }
36510     },
36511     
36512     queueExpand : function(node){
36513         this.expandProcId = this.expandNode.defer(this.expandDelay, this, [node]);
36514     },
36515     
36516     cancelExpand : function(){
36517         if(this.expandProcId){
36518             clearTimeout(this.expandProcId);
36519             this.expandProcId = false;
36520         }
36521     },
36522     
36523     isValidDropPoint : function(n, pt, dd, e, data){
36524         if(!n || !data){ return false; }
36525         var targetNode = n.node;
36526         var dropNode = data.node;
36527         // default drop rules
36528         if(!(targetNode && targetNode.isTarget && pt)){
36529             return false;
36530         }
36531         if(pt == "append" && targetNode.allowChildren === false){
36532             return false;
36533         }
36534         if((pt == "above" || pt == "below") && (targetNode.parentNode && targetNode.parentNode.allowChildren === false)){
36535             return false;
36536         }
36537         if(dropNode && (targetNode == dropNode || dropNode.contains(targetNode))){
36538             return false;
36539         }
36540         // reuse the object
36541         var overEvent = this.dragOverData;
36542         overEvent.tree = this.tree;
36543         overEvent.target = targetNode;
36544         overEvent.data = data;
36545         overEvent.point = pt;
36546         overEvent.source = dd;
36547         overEvent.rawEvent = e;
36548         overEvent.dropNode = dropNode;
36549         overEvent.cancel = false;  
36550         var result = this.tree.fireEvent("nodedragover", overEvent);
36551         return overEvent.cancel === false && result !== false;
36552     },
36553     
36554     getDropPoint : function(e, n, dd)
36555     {
36556         var tn = n.node;
36557         if(tn.isRoot){
36558             return tn.allowChildren !== false ? "append" : false; // always append for root
36559         }
36560         var dragEl = n.ddel;
36561         var t = Roo.lib.Dom.getY(dragEl), b = t + dragEl.offsetHeight;
36562         var y = Roo.lib.Event.getPageY(e);
36563         //var noAppend = tn.allowChildren === false || tn.isLeaf();
36564         
36565         // we may drop nodes anywhere, as long as allowChildren has not been set to false..
36566         var noAppend = tn.allowChildren === false;
36567         if(this.appendOnly || tn.parentNode.allowChildren === false){
36568             return noAppend ? false : "append";
36569         }
36570         var noBelow = false;
36571         if(!this.allowParentInsert){
36572             noBelow = tn.hasChildNodes() && tn.isExpanded();
36573         }
36574         var q = (b - t) / (noAppend ? 2 : 3);
36575         if(y >= t && y < (t + q)){
36576             return "above";
36577         }else if(!noBelow && (noAppend || y >= b-q && y <= b)){
36578             return "below";
36579         }else{
36580             return "append";
36581         }
36582     },
36583     
36584     onNodeEnter : function(n, dd, e, data)
36585     {
36586         this.cancelExpand();
36587     },
36588     
36589     onNodeOver : function(n, dd, e, data)
36590     {
36591        
36592         var pt = this.getDropPoint(e, n, dd);
36593         var node = n.node;
36594         
36595         // auto node expand check
36596         if(!this.expandProcId && pt == "append" && node.hasChildNodes() && !n.node.isExpanded()){
36597             this.queueExpand(node);
36598         }else if(pt != "append"){
36599             this.cancelExpand();
36600         }
36601         
36602         // set the insert point style on the target node
36603         var returnCls = this.dropNotAllowed;
36604         if(this.isValidDropPoint(n, pt, dd, e, data)){
36605            if(pt){
36606                var el = n.ddel;
36607                var cls;
36608                if(pt == "above"){
36609                    returnCls = n.node.isFirst() ? "x-tree-drop-ok-above" : "x-tree-drop-ok-between";
36610                    cls = "x-tree-drag-insert-above";
36611                }else if(pt == "below"){
36612                    returnCls = n.node.isLast() ? "x-tree-drop-ok-below" : "x-tree-drop-ok-between";
36613                    cls = "x-tree-drag-insert-below";
36614                }else{
36615                    returnCls = "x-tree-drop-ok-append";
36616                    cls = "x-tree-drag-append";
36617                }
36618                if(this.lastInsertClass != cls){
36619                    Roo.fly(el).replaceClass(this.lastInsertClass, cls);
36620                    this.lastInsertClass = cls;
36621                }
36622            }
36623        }
36624        return returnCls;
36625     },
36626     
36627     onNodeOut : function(n, dd, e, data){
36628         
36629         this.cancelExpand();
36630         this.removeDropIndicators(n);
36631     },
36632     
36633     onNodeDrop : function(n, dd, e, data){
36634         var point = this.getDropPoint(e, n, dd);
36635         var targetNode = n.node;
36636         targetNode.ui.startDrop();
36637         if(!this.isValidDropPoint(n, point, dd, e, data)){
36638             targetNode.ui.endDrop();
36639             return false;
36640         }
36641         // first try to find the drop node
36642         var dropNode = data.node || (dd.getTreeNode ? dd.getTreeNode(data, targetNode, point, e) : null);
36643         var dropEvent = {
36644             tree : this.tree,
36645             target: targetNode,
36646             data: data,
36647             point: point,
36648             source: dd,
36649             rawEvent: e,
36650             dropNode: dropNode,
36651             cancel: !dropNode   
36652         };
36653         var retval = this.tree.fireEvent("beforenodedrop", dropEvent);
36654         if(retval === false || dropEvent.cancel === true || !dropEvent.dropNode){
36655             targetNode.ui.endDrop();
36656             return false;
36657         }
36658         // allow target changing
36659         targetNode = dropEvent.target;
36660         if(point == "append" && !targetNode.isExpanded()){
36661             targetNode.expand(false, null, function(){
36662                 this.completeDrop(dropEvent);
36663             }.createDelegate(this));
36664         }else{
36665             this.completeDrop(dropEvent);
36666         }
36667         return true;
36668     },
36669     
36670     completeDrop : function(de){
36671         var ns = de.dropNode, p = de.point, t = de.target;
36672         if(!(ns instanceof Array)){
36673             ns = [ns];
36674         }
36675         var n;
36676         for(var i = 0, len = ns.length; i < len; i++){
36677             n = ns[i];
36678             if(p == "above"){
36679                 t.parentNode.insertBefore(n, t);
36680             }else if(p == "below"){
36681                 t.parentNode.insertBefore(n, t.nextSibling);
36682             }else{
36683                 t.appendChild(n);
36684             }
36685         }
36686         n.ui.focus();
36687         if(this.tree.hlDrop){
36688             n.ui.highlight();
36689         }
36690         t.ui.endDrop();
36691         this.tree.fireEvent("nodedrop", de);
36692     },
36693     
36694     afterNodeMoved : function(dd, data, e, targetNode, dropNode){
36695         if(this.tree.hlDrop){
36696             dropNode.ui.focus();
36697             dropNode.ui.highlight();
36698         }
36699         this.tree.fireEvent("nodedrop", this.tree, targetNode, data, dd, e);
36700     },
36701     
36702     getTree : function(){
36703         return this.tree;
36704     },
36705     
36706     removeDropIndicators : function(n){
36707         if(n && n.ddel){
36708             var el = n.ddel;
36709             Roo.fly(el).removeClass([
36710                     "x-tree-drag-insert-above",
36711                     "x-tree-drag-insert-below",
36712                     "x-tree-drag-append"]);
36713             this.lastInsertClass = "_noclass";
36714         }
36715     },
36716     
36717     beforeDragDrop : function(target, e, id){
36718         this.cancelExpand();
36719         return true;
36720     },
36721     
36722     afterRepair : function(data){
36723         if(data && Roo.enableFx){
36724             data.node.ui.highlight();
36725         }
36726         this.hideProxy();
36727     } 
36728     
36729 });
36730
36731 }
36732 /*
36733  * Based on:
36734  * Ext JS Library 1.1.1
36735  * Copyright(c) 2006-2007, Ext JS, LLC.
36736  *
36737  * Originally Released Under LGPL - original licence link has changed is not relivant.
36738  *
36739  * Fork - LGPL
36740  * <script type="text/javascript">
36741  */
36742  
36743
36744 if(Roo.dd.DragZone){
36745 Roo.tree.TreeDragZone = function(tree, config){
36746     Roo.tree.TreeDragZone.superclass.constructor.call(this, tree.getTreeEl(), config);
36747     this.tree = tree;
36748 };
36749
36750 Roo.extend(Roo.tree.TreeDragZone, Roo.dd.DragZone, {
36751     ddGroup : "TreeDD",
36752    
36753     onBeforeDrag : function(data, e){
36754         var n = data.node;
36755         return n && n.draggable && !n.disabled;
36756     },
36757      
36758     
36759     onInitDrag : function(e){
36760         var data = this.dragData;
36761         this.tree.getSelectionModel().select(data.node);
36762         this.proxy.update("");
36763         data.node.ui.appendDDGhost(this.proxy.ghost.dom);
36764         this.tree.fireEvent("startdrag", this.tree, data.node, e);
36765     },
36766     
36767     getRepairXY : function(e, data){
36768         return data.node.ui.getDDRepairXY();
36769     },
36770     
36771     onEndDrag : function(data, e){
36772         this.tree.fireEvent("enddrag", this.tree, data.node, e);
36773         
36774         
36775     },
36776     
36777     onValidDrop : function(dd, e, id){
36778         this.tree.fireEvent("dragdrop", this.tree, this.dragData.node, dd, e);
36779         this.hideProxy();
36780     },
36781     
36782     beforeInvalidDrop : function(e, id){
36783         // this scrolls the original position back into view
36784         var sm = this.tree.getSelectionModel();
36785         sm.clearSelections();
36786         sm.select(this.dragData.node);
36787     }
36788 });
36789 }/*
36790  * Based on:
36791  * Ext JS Library 1.1.1
36792  * Copyright(c) 2006-2007, Ext JS, LLC.
36793  *
36794  * Originally Released Under LGPL - original licence link has changed is not relivant.
36795  *
36796  * Fork - LGPL
36797  * <script type="text/javascript">
36798  */
36799 /**
36800  * @class Roo.tree.TreeEditor
36801  * @extends Roo.Editor
36802  * Provides editor functionality for inline tree node editing.  Any valid {@link Roo.form.Field} can be used
36803  * as the editor field.
36804  * @constructor
36805  * @param {Object} config (used to be the tree panel.)
36806  * @param {Object} oldconfig DEPRECIATED Either a prebuilt {@link Roo.form.Field} instance or a Field config object
36807  * 
36808  * @cfg {Roo.tree.TreePanel} tree The tree to bind to.
36809  * @cfg {Roo.form.TextField|Object} field The field configuration
36810  *
36811  * 
36812  */
36813 Roo.tree.TreeEditor = function(config, oldconfig) { // was -- (tree, config){
36814     var tree = config;
36815     var field;
36816     if (oldconfig) { // old style..
36817         field = oldconfig.events ? oldconfig : new Roo.form.TextField(oldconfig);
36818     } else {
36819         // new style..
36820         tree = config.tree;
36821         config.field = config.field  || {};
36822         config.field.xtype = 'TextField';
36823         field = Roo.factory(config.field, Roo.form);
36824     }
36825     config = config || {};
36826     
36827     
36828     this.addEvents({
36829         /**
36830          * @event beforenodeedit
36831          * Fires when editing is initiated, but before the value changes.  Editing can be canceled by returning
36832          * false from the handler of this event.
36833          * @param {Editor} this
36834          * @param {Roo.tree.Node} node 
36835          */
36836         "beforenodeedit" : true
36837     });
36838     
36839     //Roo.log(config);
36840     Roo.tree.TreeEditor.superclass.constructor.call(this, field, config);
36841
36842     this.tree = tree;
36843
36844     tree.on('beforeclick', this.beforeNodeClick, this);
36845     tree.getTreeEl().on('mousedown', this.hide, this);
36846     this.on('complete', this.updateNode, this);
36847     this.on('beforestartedit', this.fitToTree, this);
36848     this.on('startedit', this.bindScroll, this, {delay:10});
36849     this.on('specialkey', this.onSpecialKey, this);
36850 };
36851
36852 Roo.extend(Roo.tree.TreeEditor, Roo.Editor, {
36853     /**
36854      * @cfg {String} alignment
36855      * The position to align to (see {@link Roo.Element#alignTo} for more details, defaults to "l-l").
36856      */
36857     alignment: "l-l",
36858     // inherit
36859     autoSize: false,
36860     /**
36861      * @cfg {Boolean} hideEl
36862      * True to hide the bound element while the editor is displayed (defaults to false)
36863      */
36864     hideEl : false,
36865     /**
36866      * @cfg {String} cls
36867      * CSS class to apply to the editor (defaults to "x-small-editor x-tree-editor")
36868      */
36869     cls: "x-small-editor x-tree-editor",
36870     /**
36871      * @cfg {Boolean} shim
36872      * True to shim the editor if selects/iframes could be displayed beneath it (defaults to false)
36873      */
36874     shim:false,
36875     // inherit
36876     shadow:"frame",
36877     /**
36878      * @cfg {Number} maxWidth
36879      * The maximum width in pixels of the editor field (defaults to 250).  Note that if the maxWidth would exceed
36880      * the containing tree element's size, it will be automatically limited for you to the container width, taking
36881      * scroll and client offsets into account prior to each edit.
36882      */
36883     maxWidth: 250,
36884
36885     editDelay : 350,
36886
36887     // private
36888     fitToTree : function(ed, el){
36889         var td = this.tree.getTreeEl().dom, nd = el.dom;
36890         if(td.scrollLeft >  nd.offsetLeft){ // ensure the node left point is visible
36891             td.scrollLeft = nd.offsetLeft;
36892         }
36893         var w = Math.min(
36894                 this.maxWidth,
36895                 (td.clientWidth > 20 ? td.clientWidth : td.offsetWidth) - Math.max(0, nd.offsetLeft-td.scrollLeft) - /*cushion*/5);
36896         this.setSize(w, '');
36897         
36898         return this.fireEvent('beforenodeedit', this, this.editNode);
36899         
36900     },
36901
36902     // private
36903     triggerEdit : function(node){
36904         this.completeEdit();
36905         this.editNode = node;
36906         this.startEdit(node.ui.textNode, node.text);
36907     },
36908
36909     // private
36910     bindScroll : function(){
36911         this.tree.getTreeEl().on('scroll', this.cancelEdit, this);
36912     },
36913
36914     // private
36915     beforeNodeClick : function(node, e){
36916         var sinceLast = (this.lastClick ? this.lastClick.getElapsed() : 0);
36917         this.lastClick = new Date();
36918         if(sinceLast > this.editDelay && this.tree.getSelectionModel().isSelected(node)){
36919             e.stopEvent();
36920             this.triggerEdit(node);
36921             return false;
36922         }
36923         return true;
36924     },
36925
36926     // private
36927     updateNode : function(ed, value){
36928         this.tree.getTreeEl().un('scroll', this.cancelEdit, this);
36929         this.editNode.setText(value);
36930     },
36931
36932     // private
36933     onHide : function(){
36934         Roo.tree.TreeEditor.superclass.onHide.call(this);
36935         if(this.editNode){
36936             this.editNode.ui.focus();
36937         }
36938     },
36939
36940     // private
36941     onSpecialKey : function(field, e){
36942         var k = e.getKey();
36943         if(k == e.ESC){
36944             e.stopEvent();
36945             this.cancelEdit();
36946         }else if(k == e.ENTER && !e.hasModifier()){
36947             e.stopEvent();
36948             this.completeEdit();
36949         }
36950     }
36951 });//<Script type="text/javascript">
36952 /*
36953  * Based on:
36954  * Ext JS Library 1.1.1
36955  * Copyright(c) 2006-2007, Ext JS, LLC.
36956  *
36957  * Originally Released Under LGPL - original licence link has changed is not relivant.
36958  *
36959  * Fork - LGPL
36960  * <script type="text/javascript">
36961  */
36962  
36963 /**
36964  * Not documented??? - probably should be...
36965  */
36966
36967 Roo.tree.ColumnNodeUI = Roo.extend(Roo.tree.TreeNodeUI, {
36968     //focus: Roo.emptyFn, // prevent odd scrolling behavior
36969     
36970     renderElements : function(n, a, targetNode, bulkRender){
36971         //consel.log("renderElements?");
36972         this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
36973
36974         var t = n.getOwnerTree();
36975         var tid = Pman.Tab.Document_TypesTree.tree.el.id;
36976         
36977         var cols = t.columns;
36978         var bw = t.borderWidth;
36979         var c = cols[0];
36980         var href = a.href ? a.href : Roo.isGecko ? "" : "#";
36981          var cb = typeof a.checked == "boolean";
36982         var tx = String.format('{0}',n.text || (c.renderer ? c.renderer(a[c.dataIndex], n, a) : a[c.dataIndex]));
36983         var colcls = 'x-t-' + tid + '-c0';
36984         var buf = [
36985             '<li class="x-tree-node">',
36986             
36987                 
36988                 '<div class="x-tree-node-el ', a.cls,'">',
36989                     // extran...
36990                     '<div class="x-tree-col ', colcls, '" style="width:', c.width-bw, 'px;">',
36991                 
36992                 
36993                         '<span class="x-tree-node-indent">',this.indentMarkup,'</span>',
36994                         '<img src="', this.emptyIcon, '" class="x-tree-ec-icon  " />',
36995                         '<img src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',
36996                            (a.icon ? ' x-tree-node-inline-icon' : ''),
36997                            (a.iconCls ? ' '+a.iconCls : ''),
36998                            '" unselectable="on" />',
36999                         (cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + 
37000                              (a.checked ? 'checked="checked" />' : ' />')) : ''),
37001                              
37002                         '<a class="x-tree-node-anchor" hidefocus="on" href="',href,'" tabIndex="1" ',
37003                             (a.hrefTarget ? ' target="' +a.hrefTarget + '"' : ''), '>',
37004                             '<span unselectable="on" qtip="' + tx + '">',
37005                              tx,
37006                              '</span></a>' ,
37007                     '</div>',
37008                      '<a class="x-tree-node-anchor" hidefocus="on" href="',href,'" tabIndex="1" ',
37009                             (a.hrefTarget ? ' target="' +a.hrefTarget + '"' : ''), '>'
37010                  ];
37011         for(var i = 1, len = cols.length; i < len; i++){
37012             c = cols[i];
37013             colcls = 'x-t-' + tid + '-c' +i;
37014             tx = String.format('{0}', (c.renderer ? c.renderer(a[c.dataIndex], n, a) : a[c.dataIndex]));
37015             buf.push('<div class="x-tree-col ', colcls, ' ' ,(c.cls?c.cls:''),'" style="width:',c.width-bw,'px;">',
37016                         '<div class="x-tree-col-text" qtip="' + tx +'">',tx,"</div>",
37017                       "</div>");
37018          }
37019          
37020          buf.push(
37021             '</a>',
37022             '<div class="x-clear"></div></div>',
37023             '<ul class="x-tree-node-ct" style="display:none;"></ul>',
37024             "</li>");
37025         
37026         if(bulkRender !== true && n.nextSibling && n.nextSibling.ui.getEl()){
37027             this.wrap = Roo.DomHelper.insertHtml("beforeBegin",
37028                                 n.nextSibling.ui.getEl(), buf.join(""));
37029         }else{
37030             this.wrap = Roo.DomHelper.insertHtml("beforeEnd", targetNode, buf.join(""));
37031         }
37032         var el = this.wrap.firstChild;
37033         this.elRow = el;
37034         this.elNode = el.firstChild;
37035         this.ranchor = el.childNodes[1];
37036         this.ctNode = this.wrap.childNodes[1];
37037         var cs = el.firstChild.childNodes;
37038         this.indentNode = cs[0];
37039         this.ecNode = cs[1];
37040         this.iconNode = cs[2];
37041         var index = 3;
37042         if(cb){
37043             this.checkbox = cs[3];
37044             index++;
37045         }
37046         this.anchor = cs[index];
37047         
37048         this.textNode = cs[index].firstChild;
37049         
37050         //el.on("click", this.onClick, this);
37051         //el.on("dblclick", this.onDblClick, this);
37052         
37053         
37054        // console.log(this);
37055     },
37056     initEvents : function(){
37057         Roo.tree.ColumnNodeUI.superclass.initEvents.call(this);
37058         
37059             
37060         var a = this.ranchor;
37061
37062         var el = Roo.get(a);
37063
37064         if(Roo.isOpera){ // opera render bug ignores the CSS
37065             el.setStyle("text-decoration", "none");
37066         }
37067
37068         el.on("click", this.onClick, this);
37069         el.on("dblclick", this.onDblClick, this);
37070         el.on("contextmenu", this.onContextMenu, this);
37071         
37072     },
37073     
37074     /*onSelectedChange : function(state){
37075         if(state){
37076             this.focus();
37077             this.addClass("x-tree-selected");
37078         }else{
37079             //this.blur();
37080             this.removeClass("x-tree-selected");
37081         }
37082     },*/
37083     addClass : function(cls){
37084         if(this.elRow){
37085             Roo.fly(this.elRow).addClass(cls);
37086         }
37087         
37088     },
37089     
37090     
37091     removeClass : function(cls){
37092         if(this.elRow){
37093             Roo.fly(this.elRow).removeClass(cls);
37094         }
37095     }
37096
37097     
37098     
37099 });//<Script type="text/javascript">
37100
37101 /*
37102  * Based on:
37103  * Ext JS Library 1.1.1
37104  * Copyright(c) 2006-2007, Ext JS, LLC.
37105  *
37106  * Originally Released Under LGPL - original licence link has changed is not relivant.
37107  *
37108  * Fork - LGPL
37109  * <script type="text/javascript">
37110  */
37111  
37112
37113 /**
37114  * @class Roo.tree.ColumnTree
37115  * @extends Roo.data.TreePanel
37116  * @cfg {Object} columns  Including width, header, renderer, cls, dataIndex 
37117  * @cfg {int} borderWidth  compined right/left border allowance
37118  * @constructor
37119  * @param {String/HTMLElement/Element} el The container element
37120  * @param {Object} config
37121  */
37122 Roo.tree.ColumnTree =  function(el, config)
37123 {
37124    Roo.tree.ColumnTree.superclass.constructor.call(this, el , config);
37125    this.addEvents({
37126         /**
37127         * @event resize
37128         * Fire this event on a container when it resizes
37129         * @param {int} w Width
37130         * @param {int} h Height
37131         */
37132        "resize" : true
37133     });
37134     this.on('resize', this.onResize, this);
37135 };
37136
37137 Roo.extend(Roo.tree.ColumnTree, Roo.tree.TreePanel, {
37138     //lines:false,
37139     
37140     
37141     borderWidth: Roo.isBorderBox ? 0 : 2, 
37142     headEls : false,
37143     
37144     render : function(){
37145         // add the header.....
37146        
37147         Roo.tree.ColumnTree.superclass.render.apply(this);
37148         
37149         this.el.addClass('x-column-tree');
37150         
37151         this.headers = this.el.createChild(
37152             {cls:'x-tree-headers'},this.innerCt.dom);
37153    
37154         var cols = this.columns, c;
37155         var totalWidth = 0;
37156         this.headEls = [];
37157         var  len = cols.length;
37158         for(var i = 0; i < len; i++){
37159              c = cols[i];
37160              totalWidth += c.width;
37161             this.headEls.push(this.headers.createChild({
37162                  cls:'x-tree-hd ' + (c.cls?c.cls+'-hd':''),
37163                  cn: {
37164                      cls:'x-tree-hd-text',
37165                      html: c.header
37166                  },
37167                  style:'width:'+(c.width-this.borderWidth)+'px;'
37168              }));
37169         }
37170         this.headers.createChild({cls:'x-clear'});
37171         // prevent floats from wrapping when clipped
37172         this.headers.setWidth(totalWidth);
37173         //this.innerCt.setWidth(totalWidth);
37174         this.innerCt.setStyle({ overflow: 'auto' });
37175         this.onResize(this.width, this.height);
37176              
37177         
37178     },
37179     onResize : function(w,h)
37180     {
37181         this.height = h;
37182         this.width = w;
37183         // resize cols..
37184         this.innerCt.setWidth(this.width);
37185         this.innerCt.setHeight(this.height-20);
37186         
37187         // headers...
37188         var cols = this.columns, c;
37189         var totalWidth = 0;
37190         var expEl = false;
37191         var len = cols.length;
37192         for(var i = 0; i < len; i++){
37193             c = cols[i];
37194             if (this.autoExpandColumn !== false && c.dataIndex == this.autoExpandColumn) {
37195                 // it's the expander..
37196                 expEl  = this.headEls[i];
37197                 continue;
37198             }
37199             totalWidth += c.width;
37200             
37201         }
37202         if (expEl) {
37203             expEl.setWidth(  ((w - totalWidth)-this.borderWidth - 20));
37204         }
37205         this.headers.setWidth(w-20);
37206
37207         
37208         
37209         
37210     }
37211 });
37212 /*
37213  * Based on:
37214  * Ext JS Library 1.1.1
37215  * Copyright(c) 2006-2007, Ext JS, LLC.
37216  *
37217  * Originally Released Under LGPL - original licence link has changed is not relivant.
37218  *
37219  * Fork - LGPL
37220  * <script type="text/javascript">
37221  */
37222  
37223 /**
37224  * @class Roo.menu.Menu
37225  * @extends Roo.util.Observable
37226  * A menu object.  This is the container to which you add all other menu items.  Menu can also serve a as a base class
37227  * when you want a specialzed menu based off of another component (like {@link Roo.menu.DateMenu} for example).
37228  * @constructor
37229  * Creates a new Menu
37230  * @param {Object} config Configuration options
37231  */
37232 Roo.menu.Menu = function(config){
37233     
37234     Roo.menu.Menu.superclass.constructor.call(this, config);
37235     
37236     this.id = this.id || Roo.id();
37237     this.addEvents({
37238         /**
37239          * @event beforeshow
37240          * Fires before this menu is displayed
37241          * @param {Roo.menu.Menu} this
37242          */
37243         beforeshow : true,
37244         /**
37245          * @event beforehide
37246          * Fires before this menu is hidden
37247          * @param {Roo.menu.Menu} this
37248          */
37249         beforehide : true,
37250         /**
37251          * @event show
37252          * Fires after this menu is displayed
37253          * @param {Roo.menu.Menu} this
37254          */
37255         show : true,
37256         /**
37257          * @event hide
37258          * Fires after this menu is hidden
37259          * @param {Roo.menu.Menu} this
37260          */
37261         hide : true,
37262         /**
37263          * @event click
37264          * Fires when this menu is clicked (or when the enter key is pressed while it is active)
37265          * @param {Roo.menu.Menu} this
37266          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37267          * @param {Roo.EventObject} e
37268          */
37269         click : true,
37270         /**
37271          * @event mouseover
37272          * Fires when the mouse is hovering over this menu
37273          * @param {Roo.menu.Menu} this
37274          * @param {Roo.EventObject} e
37275          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37276          */
37277         mouseover : true,
37278         /**
37279          * @event mouseout
37280          * Fires when the mouse exits this menu
37281          * @param {Roo.menu.Menu} this
37282          * @param {Roo.EventObject} e
37283          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37284          */
37285         mouseout : true,
37286         /**
37287          * @event itemclick
37288          * Fires when a menu item contained in this menu is clicked
37289          * @param {Roo.menu.BaseItem} baseItem The BaseItem that was clicked
37290          * @param {Roo.EventObject} e
37291          */
37292         itemclick: true
37293     });
37294     if (this.registerMenu) {
37295         Roo.menu.MenuMgr.register(this);
37296     }
37297     
37298     var mis = this.items;
37299     this.items = new Roo.util.MixedCollection();
37300     if(mis){
37301         this.add.apply(this, mis);
37302     }
37303 };
37304
37305 Roo.extend(Roo.menu.Menu, Roo.util.Observable, {
37306     /**
37307      * @cfg {Number} minWidth The minimum width of the menu in pixels (defaults to 120)
37308      */
37309     minWidth : 120,
37310     /**
37311      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop"
37312      * for bottom-right shadow (defaults to "sides")
37313      */
37314     shadow : "sides",
37315     /**
37316      * @cfg {String} subMenuAlign The {@link Roo.Element#alignTo} anchor position value to use for submenus of
37317      * this menu (defaults to "tl-tr?")
37318      */
37319     subMenuAlign : "tl-tr?",
37320     /**
37321      * @cfg {String} defaultAlign The default {@link Roo.Element#alignTo) anchor position value for this menu
37322      * relative to its element of origin (defaults to "tl-bl?")
37323      */
37324     defaultAlign : "tl-bl?",
37325     /**
37326      * @cfg {Boolean} allowOtherMenus True to allow multiple menus to be displayed at the same time (defaults to false)
37327      */
37328     allowOtherMenus : false,
37329     /**
37330      * @cfg {Boolean} registerMenu True (default) - means that clicking on screen etc. hides it.
37331      */
37332     registerMenu : true,
37333
37334     hidden:true,
37335
37336     // private
37337     render : function(){
37338         if(this.el){
37339             return;
37340         }
37341         var el = this.el = new Roo.Layer({
37342             cls: "x-menu",
37343             shadow:this.shadow,
37344             constrain: false,
37345             parentEl: this.parentEl || document.body,
37346             zindex:15000
37347         });
37348
37349         this.keyNav = new Roo.menu.MenuNav(this);
37350
37351         if(this.plain){
37352             el.addClass("x-menu-plain");
37353         }
37354         if(this.cls){
37355             el.addClass(this.cls);
37356         }
37357         // generic focus element
37358         this.focusEl = el.createChild({
37359             tag: "a", cls: "x-menu-focus", href: "#", onclick: "return false;", tabIndex:"-1"
37360         });
37361         var ul = el.createChild({tag: "ul", cls: "x-menu-list"});
37362         //disabling touch- as it's causing issues ..
37363         //ul.on(Roo.isTouch ? 'touchstart' : 'click'   , this.onClick, this);
37364         ul.on('click'   , this.onClick, this);
37365         
37366         
37367         ul.on("mouseover", this.onMouseOver, this);
37368         ul.on("mouseout", this.onMouseOut, this);
37369         this.items.each(function(item){
37370             if (item.hidden) {
37371                 return;
37372             }
37373             
37374             var li = document.createElement("li");
37375             li.className = "x-menu-list-item";
37376             ul.dom.appendChild(li);
37377             item.render(li, this);
37378         }, this);
37379         this.ul = ul;
37380         this.autoWidth();
37381     },
37382
37383     // private
37384     autoWidth : function(){
37385         var el = this.el, ul = this.ul;
37386         if(!el){
37387             return;
37388         }
37389         var w = this.width;
37390         if(w){
37391             el.setWidth(w);
37392         }else if(Roo.isIE){
37393             el.setWidth(this.minWidth);
37394             var t = el.dom.offsetWidth; // force recalc
37395             el.setWidth(ul.getWidth()+el.getFrameWidth("lr"));
37396         }
37397     },
37398
37399     // private
37400     delayAutoWidth : function(){
37401         if(this.rendered){
37402             if(!this.awTask){
37403                 this.awTask = new Roo.util.DelayedTask(this.autoWidth, this);
37404             }
37405             this.awTask.delay(20);
37406         }
37407     },
37408
37409     // private
37410     findTargetItem : function(e){
37411         var t = e.getTarget(".x-menu-list-item", this.ul,  true);
37412         if(t && t.menuItemId){
37413             return this.items.get(t.menuItemId);
37414         }
37415     },
37416
37417     // private
37418     onClick : function(e){
37419         Roo.log("menu.onClick");
37420         var t = this.findTargetItem(e);
37421         if(!t){
37422             return;
37423         }
37424         Roo.log(e);
37425         if (Roo.isTouch && e.type == 'touchstart' && t.menu  && !t.disabled) {
37426             if(t == this.activeItem && t.shouldDeactivate(e)){
37427                 this.activeItem.deactivate();
37428                 delete this.activeItem;
37429                 return;
37430             }
37431             if(t.canActivate){
37432                 this.setActiveItem(t, true);
37433             }
37434             return;
37435             
37436             
37437         }
37438         
37439         t.onClick(e);
37440         this.fireEvent("click", this, t, e);
37441     },
37442
37443     // private
37444     setActiveItem : function(item, autoExpand){
37445         if(item != this.activeItem){
37446             if(this.activeItem){
37447                 this.activeItem.deactivate();
37448             }
37449             this.activeItem = item;
37450             item.activate(autoExpand);
37451         }else if(autoExpand){
37452             item.expandMenu();
37453         }
37454     },
37455
37456     // private
37457     tryActivate : function(start, step){
37458         var items = this.items;
37459         for(var i = start, len = items.length; i >= 0 && i < len; i+= step){
37460             var item = items.get(i);
37461             if(!item.disabled && item.canActivate){
37462                 this.setActiveItem(item, false);
37463                 return item;
37464             }
37465         }
37466         return false;
37467     },
37468
37469     // private
37470     onMouseOver : function(e){
37471         var t;
37472         if(t = this.findTargetItem(e)){
37473             if(t.canActivate && !t.disabled){
37474                 this.setActiveItem(t, true);
37475             }
37476         }
37477         this.fireEvent("mouseover", this, e, t);
37478     },
37479
37480     // private
37481     onMouseOut : function(e){
37482         var t;
37483         if(t = this.findTargetItem(e)){
37484             if(t == this.activeItem && t.shouldDeactivate(e)){
37485                 this.activeItem.deactivate();
37486                 delete this.activeItem;
37487             }
37488         }
37489         this.fireEvent("mouseout", this, e, t);
37490     },
37491
37492     /**
37493      * Read-only.  Returns true if the menu is currently displayed, else false.
37494      * @type Boolean
37495      */
37496     isVisible : function(){
37497         return this.el && !this.hidden;
37498     },
37499
37500     /**
37501      * Displays this menu relative to another element
37502      * @param {String/HTMLElement/Roo.Element} element The element to align to
37503      * @param {String} position (optional) The {@link Roo.Element#alignTo} anchor position to use in aligning to
37504      * the element (defaults to this.defaultAlign)
37505      * @param {Roo.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
37506      */
37507     show : function(el, pos, parentMenu){
37508         this.parentMenu = parentMenu;
37509         if(!this.el){
37510             this.render();
37511         }
37512         this.fireEvent("beforeshow", this);
37513         this.showAt(this.el.getAlignToXY(el, pos || this.defaultAlign), parentMenu, false);
37514     },
37515
37516     /**
37517      * Displays this menu at a specific xy position
37518      * @param {Array} xyPosition Contains X & Y [x, y] values for the position at which to show the menu (coordinates are page-based)
37519      * @param {Roo.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
37520      */
37521     showAt : function(xy, parentMenu, /* private: */_e){
37522         this.parentMenu = parentMenu;
37523         if(!this.el){
37524             this.render();
37525         }
37526         if(_e !== false){
37527             this.fireEvent("beforeshow", this);
37528             xy = this.el.adjustForConstraints(xy);
37529         }
37530         this.el.setXY(xy);
37531         this.el.show();
37532         this.hidden = false;
37533         this.focus();
37534         this.fireEvent("show", this);
37535     },
37536
37537     focus : function(){
37538         if(!this.hidden){
37539             this.doFocus.defer(50, this);
37540         }
37541     },
37542
37543     doFocus : function(){
37544         if(!this.hidden){
37545             this.focusEl.focus();
37546         }
37547     },
37548
37549     /**
37550      * Hides this menu and optionally all parent menus
37551      * @param {Boolean} deep (optional) True to hide all parent menus recursively, if any (defaults to false)
37552      */
37553     hide : function(deep){
37554         if(this.el && this.isVisible()){
37555             this.fireEvent("beforehide", this);
37556             if(this.activeItem){
37557                 this.activeItem.deactivate();
37558                 this.activeItem = null;
37559             }
37560             this.el.hide();
37561             this.hidden = true;
37562             this.fireEvent("hide", this);
37563         }
37564         if(deep === true && this.parentMenu){
37565             this.parentMenu.hide(true);
37566         }
37567     },
37568
37569     /**
37570      * Addds one or more items of any type supported by the Menu class, or that can be converted into menu items.
37571      * Any of the following are valid:
37572      * <ul>
37573      * <li>Any menu item object based on {@link Roo.menu.Item}</li>
37574      * <li>An HTMLElement object which will be converted to a menu item</li>
37575      * <li>A menu item config object that will be created as a new menu item</li>
37576      * <li>A string, which can either be '-' or 'separator' to add a menu separator, otherwise
37577      * it will be converted into a {@link Roo.menu.TextItem} and added</li>
37578      * </ul>
37579      * Usage:
37580      * <pre><code>
37581 // Create the menu
37582 var menu = new Roo.menu.Menu();
37583
37584 // Create a menu item to add by reference
37585 var menuItem = new Roo.menu.Item({ text: 'New Item!' });
37586
37587 // Add a bunch of items at once using different methods.
37588 // Only the last item added will be returned.
37589 var item = menu.add(
37590     menuItem,                // add existing item by ref
37591     'Dynamic Item',          // new TextItem
37592     '-',                     // new separator
37593     { text: 'Config Item' }  // new item by config
37594 );
37595 </code></pre>
37596      * @param {Mixed} args One or more menu items, menu item configs or other objects that can be converted to menu items
37597      * @return {Roo.menu.Item} The menu item that was added, or the last one if multiple items were added
37598      */
37599     add : function(){
37600         var a = arguments, l = a.length, item;
37601         for(var i = 0; i < l; i++){
37602             var el = a[i];
37603             if ((typeof(el) == "object") && el.xtype && el.xns) {
37604                 el = Roo.factory(el, Roo.menu);
37605             }
37606             
37607             if(el.render){ // some kind of Item
37608                 item = this.addItem(el);
37609             }else if(typeof el == "string"){ // string
37610                 if(el == "separator" || el == "-"){
37611                     item = this.addSeparator();
37612                 }else{
37613                     item = this.addText(el);
37614                 }
37615             }else if(el.tagName || el.el){ // element
37616                 item = this.addElement(el);
37617             }else if(typeof el == "object"){ // must be menu item config?
37618                 item = this.addMenuItem(el);
37619             }
37620         }
37621         return item;
37622     },
37623
37624     /**
37625      * Returns this menu's underlying {@link Roo.Element} object
37626      * @return {Roo.Element} The element
37627      */
37628     getEl : function(){
37629         if(!this.el){
37630             this.render();
37631         }
37632         return this.el;
37633     },
37634
37635     /**
37636      * Adds a separator bar to the menu
37637      * @return {Roo.menu.Item} The menu item that was added
37638      */
37639     addSeparator : function(){
37640         return this.addItem(new Roo.menu.Separator());
37641     },
37642
37643     /**
37644      * Adds an {@link Roo.Element} object to the menu
37645      * @param {String/HTMLElement/Roo.Element} el The element or DOM node to add, or its id
37646      * @return {Roo.menu.Item} The menu item that was added
37647      */
37648     addElement : function(el){
37649         return this.addItem(new Roo.menu.BaseItem(el));
37650     },
37651
37652     /**
37653      * Adds an existing object based on {@link Roo.menu.Item} to the menu
37654      * @param {Roo.menu.Item} item The menu item to add
37655      * @return {Roo.menu.Item} The menu item that was added
37656      */
37657     addItem : function(item){
37658         this.items.add(item);
37659         if(this.ul){
37660             var li = document.createElement("li");
37661             li.className = "x-menu-list-item";
37662             this.ul.dom.appendChild(li);
37663             item.render(li, this);
37664             this.delayAutoWidth();
37665         }
37666         return item;
37667     },
37668
37669     /**
37670      * Creates a new {@link Roo.menu.Item} based an the supplied config object and adds it to the menu
37671      * @param {Object} config A MenuItem config object
37672      * @return {Roo.menu.Item} The menu item that was added
37673      */
37674     addMenuItem : function(config){
37675         if(!(config instanceof Roo.menu.Item)){
37676             if(typeof config.checked == "boolean"){ // must be check menu item config?
37677                 config = new Roo.menu.CheckItem(config);
37678             }else{
37679                 config = new Roo.menu.Item(config);
37680             }
37681         }
37682         return this.addItem(config);
37683     },
37684
37685     /**
37686      * Creates a new {@link Roo.menu.TextItem} with the supplied text and adds it to the menu
37687      * @param {String} text The text to display in the menu item
37688      * @return {Roo.menu.Item} The menu item that was added
37689      */
37690     addText : function(text){
37691         return this.addItem(new Roo.menu.TextItem({ text : text }));
37692     },
37693
37694     /**
37695      * Inserts an existing object based on {@link Roo.menu.Item} to the menu at a specified index
37696      * @param {Number} index The index in the menu's list of current items where the new item should be inserted
37697      * @param {Roo.menu.Item} item The menu item to add
37698      * @return {Roo.menu.Item} The menu item that was added
37699      */
37700     insert : function(index, item){
37701         this.items.insert(index, item);
37702         if(this.ul){
37703             var li = document.createElement("li");
37704             li.className = "x-menu-list-item";
37705             this.ul.dom.insertBefore(li, this.ul.dom.childNodes[index]);
37706             item.render(li, this);
37707             this.delayAutoWidth();
37708         }
37709         return item;
37710     },
37711
37712     /**
37713      * Removes an {@link Roo.menu.Item} from the menu and destroys the object
37714      * @param {Roo.menu.Item} item The menu item to remove
37715      */
37716     remove : function(item){
37717         this.items.removeKey(item.id);
37718         item.destroy();
37719     },
37720
37721     /**
37722      * Removes and destroys all items in the menu
37723      */
37724     removeAll : function(){
37725         var f;
37726         while(f = this.items.first()){
37727             this.remove(f);
37728         }
37729     }
37730 });
37731
37732 // MenuNav is a private utility class used internally by the Menu
37733 Roo.menu.MenuNav = function(menu){
37734     Roo.menu.MenuNav.superclass.constructor.call(this, menu.el);
37735     this.scope = this.menu = menu;
37736 };
37737
37738 Roo.extend(Roo.menu.MenuNav, Roo.KeyNav, {
37739     doRelay : function(e, h){
37740         var k = e.getKey();
37741         if(!this.menu.activeItem && e.isNavKeyPress() && k != e.SPACE && k != e.RETURN){
37742             this.menu.tryActivate(0, 1);
37743             return false;
37744         }
37745         return h.call(this.scope || this, e, this.menu);
37746     },
37747
37748     up : function(e, m){
37749         if(!m.tryActivate(m.items.indexOf(m.activeItem)-1, -1)){
37750             m.tryActivate(m.items.length-1, -1);
37751         }
37752     },
37753
37754     down : function(e, m){
37755         if(!m.tryActivate(m.items.indexOf(m.activeItem)+1, 1)){
37756             m.tryActivate(0, 1);
37757         }
37758     },
37759
37760     right : function(e, m){
37761         if(m.activeItem){
37762             m.activeItem.expandMenu(true);
37763         }
37764     },
37765
37766     left : function(e, m){
37767         m.hide();
37768         if(m.parentMenu && m.parentMenu.activeItem){
37769             m.parentMenu.activeItem.activate();
37770         }
37771     },
37772
37773     enter : function(e, m){
37774         if(m.activeItem){
37775             e.stopPropagation();
37776             m.activeItem.onClick(e);
37777             m.fireEvent("click", this, m.activeItem);
37778             return true;
37779         }
37780     }
37781 });/*
37782  * Based on:
37783  * Ext JS Library 1.1.1
37784  * Copyright(c) 2006-2007, Ext JS, LLC.
37785  *
37786  * Originally Released Under LGPL - original licence link has changed is not relivant.
37787  *
37788  * Fork - LGPL
37789  * <script type="text/javascript">
37790  */
37791  
37792 /**
37793  * @class Roo.menu.MenuMgr
37794  * Provides a common registry of all menu items on a page so that they can be easily accessed by id.
37795  * @singleton
37796  */
37797 Roo.menu.MenuMgr = function(){
37798    var menus, active, groups = {}, attached = false, lastShow = new Date();
37799
37800    // private - called when first menu is created
37801    function init(){
37802        menus = {};
37803        active = new Roo.util.MixedCollection();
37804        Roo.get(document).addKeyListener(27, function(){
37805            if(active.length > 0){
37806                hideAll();
37807            }
37808        });
37809    }
37810
37811    // private
37812    function hideAll(){
37813        if(active && active.length > 0){
37814            var c = active.clone();
37815            c.each(function(m){
37816                m.hide();
37817            });
37818        }
37819    }
37820
37821    // private
37822    function onHide(m){
37823        active.remove(m);
37824        if(active.length < 1){
37825            Roo.get(document).un("mousedown", onMouseDown);
37826            attached = false;
37827        }
37828    }
37829
37830    // private
37831    function onShow(m){
37832        var last = active.last();
37833        lastShow = new Date();
37834        active.add(m);
37835        if(!attached){
37836            Roo.get(document).on("mousedown", onMouseDown);
37837            attached = true;
37838        }
37839        if(m.parentMenu){
37840           m.getEl().setZIndex(parseInt(m.parentMenu.getEl().getStyle("z-index"), 10) + 3);
37841           m.parentMenu.activeChild = m;
37842        }else if(last && last.isVisible()){
37843           m.getEl().setZIndex(parseInt(last.getEl().getStyle("z-index"), 10) + 3);
37844        }
37845    }
37846
37847    // private
37848    function onBeforeHide(m){
37849        if(m.activeChild){
37850            m.activeChild.hide();
37851        }
37852        if(m.autoHideTimer){
37853            clearTimeout(m.autoHideTimer);
37854            delete m.autoHideTimer;
37855        }
37856    }
37857
37858    // private
37859    function onBeforeShow(m){
37860        var pm = m.parentMenu;
37861        if(!pm && !m.allowOtherMenus){
37862            hideAll();
37863        }else if(pm && pm.activeChild && active != m){
37864            pm.activeChild.hide();
37865        }
37866    }
37867
37868    // private
37869    function onMouseDown(e){
37870        if(lastShow.getElapsed() > 50 && active.length > 0 && !e.getTarget(".x-menu")){
37871            hideAll();
37872        }
37873    }
37874
37875    // private
37876    function onBeforeCheck(mi, state){
37877        if(state){
37878            var g = groups[mi.group];
37879            for(var i = 0, l = g.length; i < l; i++){
37880                if(g[i] != mi){
37881                    g[i].setChecked(false);
37882                }
37883            }
37884        }
37885    }
37886
37887    return {
37888
37889        /**
37890         * Hides all menus that are currently visible
37891         */
37892        hideAll : function(){
37893             hideAll();  
37894        },
37895
37896        // private
37897        register : function(menu){
37898            if(!menus){
37899                init();
37900            }
37901            menus[menu.id] = menu;
37902            menu.on("beforehide", onBeforeHide);
37903            menu.on("hide", onHide);
37904            menu.on("beforeshow", onBeforeShow);
37905            menu.on("show", onShow);
37906            var g = menu.group;
37907            if(g && menu.events["checkchange"]){
37908                if(!groups[g]){
37909                    groups[g] = [];
37910                }
37911                groups[g].push(menu);
37912                menu.on("checkchange", onCheck);
37913            }
37914        },
37915
37916         /**
37917          * Returns a {@link Roo.menu.Menu} object
37918          * @param {String/Object} menu The string menu id, an existing menu object reference, or a Menu config that will
37919          * be used to generate and return a new Menu instance.
37920          */
37921        get : function(menu){
37922            if(typeof menu == "string"){ // menu id
37923                return menus[menu];
37924            }else if(menu.events){  // menu instance
37925                return menu;
37926            }else if(typeof menu.length == 'number'){ // array of menu items?
37927                return new Roo.menu.Menu({items:menu});
37928            }else{ // otherwise, must be a config
37929                return new Roo.menu.Menu(menu);
37930            }
37931        },
37932
37933        // private
37934        unregister : function(menu){
37935            delete menus[menu.id];
37936            menu.un("beforehide", onBeforeHide);
37937            menu.un("hide", onHide);
37938            menu.un("beforeshow", onBeforeShow);
37939            menu.un("show", onShow);
37940            var g = menu.group;
37941            if(g && menu.events["checkchange"]){
37942                groups[g].remove(menu);
37943                menu.un("checkchange", onCheck);
37944            }
37945        },
37946
37947        // private
37948        registerCheckable : function(menuItem){
37949            var g = menuItem.group;
37950            if(g){
37951                if(!groups[g]){
37952                    groups[g] = [];
37953                }
37954                groups[g].push(menuItem);
37955                menuItem.on("beforecheckchange", onBeforeCheck);
37956            }
37957        },
37958
37959        // private
37960        unregisterCheckable : function(menuItem){
37961            var g = menuItem.group;
37962            if(g){
37963                groups[g].remove(menuItem);
37964                menuItem.un("beforecheckchange", onBeforeCheck);
37965            }
37966        }
37967    };
37968 }();/*
37969  * Based on:
37970  * Ext JS Library 1.1.1
37971  * Copyright(c) 2006-2007, Ext JS, LLC.
37972  *
37973  * Originally Released Under LGPL - original licence link has changed is not relivant.
37974  *
37975  * Fork - LGPL
37976  * <script type="text/javascript">
37977  */
37978  
37979
37980 /**
37981  * @class Roo.menu.BaseItem
37982  * @extends Roo.Component
37983  * The base class for all items that render into menus.  BaseItem provides default rendering, activated state
37984  * management and base configuration options shared by all menu components.
37985  * @constructor
37986  * Creates a new BaseItem
37987  * @param {Object} config Configuration options
37988  */
37989 Roo.menu.BaseItem = function(config){
37990     Roo.menu.BaseItem.superclass.constructor.call(this, config);
37991
37992     this.addEvents({
37993         /**
37994          * @event click
37995          * Fires when this item is clicked
37996          * @param {Roo.menu.BaseItem} this
37997          * @param {Roo.EventObject} e
37998          */
37999         click: true,
38000         /**
38001          * @event activate
38002          * Fires when this item is activated
38003          * @param {Roo.menu.BaseItem} this
38004          */
38005         activate : true,
38006         /**
38007          * @event deactivate
38008          * Fires when this item is deactivated
38009          * @param {Roo.menu.BaseItem} this
38010          */
38011         deactivate : true
38012     });
38013
38014     if(this.handler){
38015         this.on("click", this.handler, this.scope, true);
38016     }
38017 };
38018
38019 Roo.extend(Roo.menu.BaseItem, Roo.Component, {
38020     /**
38021      * @cfg {Function} handler
38022      * A function that will handle the click event of this menu item (defaults to undefined)
38023      */
38024     /**
38025      * @cfg {Boolean} canActivate True if this item can be visually activated (defaults to false)
38026      */
38027     canActivate : false,
38028     
38029      /**
38030      * @cfg {Boolean} hidden True to prevent creation of this menu item (defaults to false)
38031      */
38032     hidden: false,
38033     
38034     /**
38035      * @cfg {String} activeClass The CSS class to use when the item becomes activated (defaults to "x-menu-item-active")
38036      */
38037     activeClass : "x-menu-item-active",
38038     /**
38039      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to true)
38040      */
38041     hideOnClick : true,
38042     /**
38043      * @cfg {Number} hideDelay Length of time in milliseconds to wait before hiding after a click (defaults to 100)
38044      */
38045     hideDelay : 100,
38046
38047     // private
38048     ctype: "Roo.menu.BaseItem",
38049
38050     // private
38051     actionMode : "container",
38052
38053     // private
38054     render : function(container, parentMenu){
38055         this.parentMenu = parentMenu;
38056         Roo.menu.BaseItem.superclass.render.call(this, container);
38057         this.container.menuItemId = this.id;
38058     },
38059
38060     // private
38061     onRender : function(container, position){
38062         this.el = Roo.get(this.el);
38063         container.dom.appendChild(this.el.dom);
38064     },
38065
38066     // private
38067     onClick : function(e){
38068         if(!this.disabled && this.fireEvent("click", this, e) !== false
38069                 && this.parentMenu.fireEvent("itemclick", this, e) !== false){
38070             this.handleClick(e);
38071         }else{
38072             e.stopEvent();
38073         }
38074     },
38075
38076     // private
38077     activate : function(){
38078         if(this.disabled){
38079             return false;
38080         }
38081         var li = this.container;
38082         li.addClass(this.activeClass);
38083         this.region = li.getRegion().adjust(2, 2, -2, -2);
38084         this.fireEvent("activate", this);
38085         return true;
38086     },
38087
38088     // private
38089     deactivate : function(){
38090         this.container.removeClass(this.activeClass);
38091         this.fireEvent("deactivate", this);
38092     },
38093
38094     // private
38095     shouldDeactivate : function(e){
38096         return !this.region || !this.region.contains(e.getPoint());
38097     },
38098
38099     // private
38100     handleClick : function(e){
38101         if(this.hideOnClick){
38102             this.parentMenu.hide.defer(this.hideDelay, this.parentMenu, [true]);
38103         }
38104     },
38105
38106     // private
38107     expandMenu : function(autoActivate){
38108         // do nothing
38109     },
38110
38111     // private
38112     hideMenu : function(){
38113         // do nothing
38114     }
38115 });/*
38116  * Based on:
38117  * Ext JS Library 1.1.1
38118  * Copyright(c) 2006-2007, Ext JS, LLC.
38119  *
38120  * Originally Released Under LGPL - original licence link has changed is not relivant.
38121  *
38122  * Fork - LGPL
38123  * <script type="text/javascript">
38124  */
38125  
38126 /**
38127  * @class Roo.menu.Adapter
38128  * @extends Roo.menu.BaseItem
38129  * 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.
38130  * It provides basic rendering, activation management and enable/disable logic required to work in menus.
38131  * @constructor
38132  * Creates a new Adapter
38133  * @param {Object} config Configuration options
38134  */
38135 Roo.menu.Adapter = function(component, config){
38136     Roo.menu.Adapter.superclass.constructor.call(this, config);
38137     this.component = component;
38138 };
38139 Roo.extend(Roo.menu.Adapter, Roo.menu.BaseItem, {
38140     // private
38141     canActivate : true,
38142
38143     // private
38144     onRender : function(container, position){
38145         this.component.render(container);
38146         this.el = this.component.getEl();
38147     },
38148
38149     // private
38150     activate : function(){
38151         if(this.disabled){
38152             return false;
38153         }
38154         this.component.focus();
38155         this.fireEvent("activate", this);
38156         return true;
38157     },
38158
38159     // private
38160     deactivate : function(){
38161         this.fireEvent("deactivate", this);
38162     },
38163
38164     // private
38165     disable : function(){
38166         this.component.disable();
38167         Roo.menu.Adapter.superclass.disable.call(this);
38168     },
38169
38170     // private
38171     enable : function(){
38172         this.component.enable();
38173         Roo.menu.Adapter.superclass.enable.call(this);
38174     }
38175 });/*
38176  * Based on:
38177  * Ext JS Library 1.1.1
38178  * Copyright(c) 2006-2007, Ext JS, LLC.
38179  *
38180  * Originally Released Under LGPL - original licence link has changed is not relivant.
38181  *
38182  * Fork - LGPL
38183  * <script type="text/javascript">
38184  */
38185
38186 /**
38187  * @class Roo.menu.TextItem
38188  * @extends Roo.menu.BaseItem
38189  * Adds a static text string to a menu, usually used as either a heading or group separator.
38190  * Note: old style constructor with text is still supported.
38191  * 
38192  * @constructor
38193  * Creates a new TextItem
38194  * @param {Object} cfg Configuration
38195  */
38196 Roo.menu.TextItem = function(cfg){
38197     if (typeof(cfg) == 'string') {
38198         this.text = cfg;
38199     } else {
38200         Roo.apply(this,cfg);
38201     }
38202     
38203     Roo.menu.TextItem.superclass.constructor.call(this);
38204 };
38205
38206 Roo.extend(Roo.menu.TextItem, Roo.menu.BaseItem, {
38207     /**
38208      * @cfg {Boolean} text Text to show on item.
38209      */
38210     text : '',
38211     
38212     /**
38213      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to false)
38214      */
38215     hideOnClick : false,
38216     /**
38217      * @cfg {String} itemCls The default CSS class to use for text items (defaults to "x-menu-text")
38218      */
38219     itemCls : "x-menu-text",
38220
38221     // private
38222     onRender : function(){
38223         var s = document.createElement("span");
38224         s.className = this.itemCls;
38225         s.innerHTML = this.text;
38226         this.el = s;
38227         Roo.menu.TextItem.superclass.onRender.apply(this, arguments);
38228     }
38229 });/*
38230  * Based on:
38231  * Ext JS Library 1.1.1
38232  * Copyright(c) 2006-2007, Ext JS, LLC.
38233  *
38234  * Originally Released Under LGPL - original licence link has changed is not relivant.
38235  *
38236  * Fork - LGPL
38237  * <script type="text/javascript">
38238  */
38239
38240 /**
38241  * @class Roo.menu.Separator
38242  * @extends Roo.menu.BaseItem
38243  * Adds a separator bar to a menu, used to divide logical groups of menu items. Generally you will
38244  * add one of these by using "-" in you call to add() or in your items config rather than creating one directly.
38245  * @constructor
38246  * @param {Object} config Configuration options
38247  */
38248 Roo.menu.Separator = function(config){
38249     Roo.menu.Separator.superclass.constructor.call(this, config);
38250 };
38251
38252 Roo.extend(Roo.menu.Separator, Roo.menu.BaseItem, {
38253     /**
38254      * @cfg {String} itemCls The default CSS class to use for separators (defaults to "x-menu-sep")
38255      */
38256     itemCls : "x-menu-sep",
38257     /**
38258      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to false)
38259      */
38260     hideOnClick : false,
38261
38262     // private
38263     onRender : function(li){
38264         var s = document.createElement("span");
38265         s.className = this.itemCls;
38266         s.innerHTML = "&#160;";
38267         this.el = s;
38268         li.addClass("x-menu-sep-li");
38269         Roo.menu.Separator.superclass.onRender.apply(this, arguments);
38270     }
38271 });/*
38272  * Based on:
38273  * Ext JS Library 1.1.1
38274  * Copyright(c) 2006-2007, Ext JS, LLC.
38275  *
38276  * Originally Released Under LGPL - original licence link has changed is not relivant.
38277  *
38278  * Fork - LGPL
38279  * <script type="text/javascript">
38280  */
38281 /**
38282  * @class Roo.menu.Item
38283  * @extends Roo.menu.BaseItem
38284  * A base class for all menu items that require menu-related functionality (like sub-menus) and are not static
38285  * display items.  Item extends the base functionality of {@link Roo.menu.BaseItem} by adding menu-specific
38286  * activation and click handling.
38287  * @constructor
38288  * Creates a new Item
38289  * @param {Object} config Configuration options
38290  */
38291 Roo.menu.Item = function(config){
38292     Roo.menu.Item.superclass.constructor.call(this, config);
38293     if(this.menu){
38294         this.menu = Roo.menu.MenuMgr.get(this.menu);
38295     }
38296 };
38297 Roo.extend(Roo.menu.Item, Roo.menu.BaseItem, {
38298     
38299     /**
38300      * @cfg {String} text
38301      * The text to show on the menu item.
38302      */
38303     text: '',
38304      /**
38305      * @cfg {String} HTML to render in menu
38306      * The text to show on the menu item (HTML version).
38307      */
38308     html: '',
38309     /**
38310      * @cfg {String} icon
38311      * The path to an icon to display in this menu item (defaults to Roo.BLANK_IMAGE_URL)
38312      */
38313     icon: undefined,
38314     /**
38315      * @cfg {String} itemCls The default CSS class to use for menu items (defaults to "x-menu-item")
38316      */
38317     itemCls : "x-menu-item",
38318     /**
38319      * @cfg {Boolean} canActivate True if this item can be visually activated (defaults to true)
38320      */
38321     canActivate : true,
38322     /**
38323      * @cfg {Number} showDelay Length of time in milliseconds to wait before showing this item (defaults to 200)
38324      */
38325     showDelay: 200,
38326     // doc'd in BaseItem
38327     hideDelay: 200,
38328
38329     // private
38330     ctype: "Roo.menu.Item",
38331     
38332     // private
38333     onRender : function(container, position){
38334         var el = document.createElement("a");
38335         el.hideFocus = true;
38336         el.unselectable = "on";
38337         el.href = this.href || "#";
38338         if(this.hrefTarget){
38339             el.target = this.hrefTarget;
38340         }
38341         el.className = this.itemCls + (this.menu ?  " x-menu-item-arrow" : "") + (this.cls ?  " " + this.cls : "");
38342         
38343         var html = this.html.length ? this.html  : String.format('{0}',this.text);
38344         
38345         el.innerHTML = String.format(
38346                 '<img src="{0}" class="x-menu-item-icon {1}" />' + html,
38347                 this.icon || Roo.BLANK_IMAGE_URL, this.iconCls || '');
38348         this.el = el;
38349         Roo.menu.Item.superclass.onRender.call(this, container, position);
38350     },
38351
38352     /**
38353      * Sets the text to display in this menu item
38354      * @param {String} text The text to display
38355      * @param {Boolean} isHTML true to indicate text is pure html.
38356      */
38357     setText : function(text, isHTML){
38358         if (isHTML) {
38359             this.html = text;
38360         } else {
38361             this.text = text;
38362             this.html = '';
38363         }
38364         if(this.rendered){
38365             var html = this.html.length ? this.html  : String.format('{0}',this.text);
38366      
38367             this.el.update(String.format(
38368                 '<img src="{0}" class="x-menu-item-icon {2}">' + html,
38369                 this.icon || Roo.BLANK_IMAGE_URL, this.text, this.iconCls || ''));
38370             this.parentMenu.autoWidth();
38371         }
38372     },
38373
38374     // private
38375     handleClick : function(e){
38376         if(!this.href){ // if no link defined, stop the event automatically
38377             e.stopEvent();
38378         }
38379         Roo.menu.Item.superclass.handleClick.apply(this, arguments);
38380     },
38381
38382     // private
38383     activate : function(autoExpand){
38384         if(Roo.menu.Item.superclass.activate.apply(this, arguments)){
38385             this.focus();
38386             if(autoExpand){
38387                 this.expandMenu();
38388             }
38389         }
38390         return true;
38391     },
38392
38393     // private
38394     shouldDeactivate : function(e){
38395         if(Roo.menu.Item.superclass.shouldDeactivate.call(this, e)){
38396             if(this.menu && this.menu.isVisible()){
38397                 return !this.menu.getEl().getRegion().contains(e.getPoint());
38398             }
38399             return true;
38400         }
38401         return false;
38402     },
38403
38404     // private
38405     deactivate : function(){
38406         Roo.menu.Item.superclass.deactivate.apply(this, arguments);
38407         this.hideMenu();
38408     },
38409
38410     // private
38411     expandMenu : function(autoActivate){
38412         if(!this.disabled && this.menu){
38413             clearTimeout(this.hideTimer);
38414             delete this.hideTimer;
38415             if(!this.menu.isVisible() && !this.showTimer){
38416                 this.showTimer = this.deferExpand.defer(this.showDelay, this, [autoActivate]);
38417             }else if (this.menu.isVisible() && autoActivate){
38418                 this.menu.tryActivate(0, 1);
38419             }
38420         }
38421     },
38422
38423     // private
38424     deferExpand : function(autoActivate){
38425         delete this.showTimer;
38426         this.menu.show(this.container, this.parentMenu.subMenuAlign || "tl-tr?", this.parentMenu);
38427         if(autoActivate){
38428             this.menu.tryActivate(0, 1);
38429         }
38430     },
38431
38432     // private
38433     hideMenu : function(){
38434         clearTimeout(this.showTimer);
38435         delete this.showTimer;
38436         if(!this.hideTimer && this.menu && this.menu.isVisible()){
38437             this.hideTimer = this.deferHide.defer(this.hideDelay, this);
38438         }
38439     },
38440
38441     // private
38442     deferHide : function(){
38443         delete this.hideTimer;
38444         this.menu.hide();
38445     }
38446 });/*
38447  * Based on:
38448  * Ext JS Library 1.1.1
38449  * Copyright(c) 2006-2007, Ext JS, LLC.
38450  *
38451  * Originally Released Under LGPL - original licence link has changed is not relivant.
38452  *
38453  * Fork - LGPL
38454  * <script type="text/javascript">
38455  */
38456  
38457 /**
38458  * @class Roo.menu.CheckItem
38459  * @extends Roo.menu.Item
38460  * Adds a menu item that contains a checkbox by default, but can also be part of a radio group.
38461  * @constructor
38462  * Creates a new CheckItem
38463  * @param {Object} config Configuration options
38464  */
38465 Roo.menu.CheckItem = function(config){
38466     Roo.menu.CheckItem.superclass.constructor.call(this, config);
38467     this.addEvents({
38468         /**
38469          * @event beforecheckchange
38470          * Fires before the checked value is set, providing an opportunity to cancel if needed
38471          * @param {Roo.menu.CheckItem} this
38472          * @param {Boolean} checked The new checked value that will be set
38473          */
38474         "beforecheckchange" : true,
38475         /**
38476          * @event checkchange
38477          * Fires after the checked value has been set
38478          * @param {Roo.menu.CheckItem} this
38479          * @param {Boolean} checked The checked value that was set
38480          */
38481         "checkchange" : true
38482     });
38483     if(this.checkHandler){
38484         this.on('checkchange', this.checkHandler, this.scope);
38485     }
38486 };
38487 Roo.extend(Roo.menu.CheckItem, Roo.menu.Item, {
38488     /**
38489      * @cfg {String} group
38490      * All check items with the same group name will automatically be grouped into a single-select
38491      * radio button group (defaults to '')
38492      */
38493     /**
38494      * @cfg {String} itemCls The default CSS class to use for check items (defaults to "x-menu-item x-menu-check-item")
38495      */
38496     itemCls : "x-menu-item x-menu-check-item",
38497     /**
38498      * @cfg {String} groupClass The default CSS class to use for radio group check items (defaults to "x-menu-group-item")
38499      */
38500     groupClass : "x-menu-group-item",
38501
38502     /**
38503      * @cfg {Boolean} checked True to initialize this checkbox as checked (defaults to false).  Note that
38504      * if this checkbox is part of a radio group (group = true) only the last item in the group that is
38505      * initialized with checked = true will be rendered as checked.
38506      */
38507     checked: false,
38508
38509     // private
38510     ctype: "Roo.menu.CheckItem",
38511
38512     // private
38513     onRender : function(c){
38514         Roo.menu.CheckItem.superclass.onRender.apply(this, arguments);
38515         if(this.group){
38516             this.el.addClass(this.groupClass);
38517         }
38518         Roo.menu.MenuMgr.registerCheckable(this);
38519         if(this.checked){
38520             this.checked = false;
38521             this.setChecked(true, true);
38522         }
38523     },
38524
38525     // private
38526     destroy : function(){
38527         if(this.rendered){
38528             Roo.menu.MenuMgr.unregisterCheckable(this);
38529         }
38530         Roo.menu.CheckItem.superclass.destroy.apply(this, arguments);
38531     },
38532
38533     /**
38534      * Set the checked state of this item
38535      * @param {Boolean} checked The new checked value
38536      * @param {Boolean} suppressEvent (optional) True to prevent the checkchange event from firing (defaults to false)
38537      */
38538     setChecked : function(state, suppressEvent){
38539         if(this.checked != state && this.fireEvent("beforecheckchange", this, state) !== false){
38540             if(this.container){
38541                 this.container[state ? "addClass" : "removeClass"]("x-menu-item-checked");
38542             }
38543             this.checked = state;
38544             if(suppressEvent !== true){
38545                 this.fireEvent("checkchange", this, state);
38546             }
38547         }
38548     },
38549
38550     // private
38551     handleClick : function(e){
38552        if(!this.disabled && !(this.checked && this.group)){// disable unselect on radio item
38553            this.setChecked(!this.checked);
38554        }
38555        Roo.menu.CheckItem.superclass.handleClick.apply(this, arguments);
38556     }
38557 });/*
38558  * Based on:
38559  * Ext JS Library 1.1.1
38560  * Copyright(c) 2006-2007, Ext JS, LLC.
38561  *
38562  * Originally Released Under LGPL - original licence link has changed is not relivant.
38563  *
38564  * Fork - LGPL
38565  * <script type="text/javascript">
38566  */
38567  
38568 /**
38569  * @class Roo.menu.DateItem
38570  * @extends Roo.menu.Adapter
38571  * A menu item that wraps the {@link Roo.DatPicker} component.
38572  * @constructor
38573  * Creates a new DateItem
38574  * @param {Object} config Configuration options
38575  */
38576 Roo.menu.DateItem = function(config){
38577     Roo.menu.DateItem.superclass.constructor.call(this, new Roo.DatePicker(config), config);
38578     /** The Roo.DatePicker object @type Roo.DatePicker */
38579     this.picker = this.component;
38580     this.addEvents({select: true});
38581     
38582     this.picker.on("render", function(picker){
38583         picker.getEl().swallowEvent("click");
38584         picker.container.addClass("x-menu-date-item");
38585     });
38586
38587     this.picker.on("select", this.onSelect, this);
38588 };
38589
38590 Roo.extend(Roo.menu.DateItem, Roo.menu.Adapter, {
38591     // private
38592     onSelect : function(picker, date){
38593         this.fireEvent("select", this, date, picker);
38594         Roo.menu.DateItem.superclass.handleClick.call(this);
38595     }
38596 });/*
38597  * Based on:
38598  * Ext JS Library 1.1.1
38599  * Copyright(c) 2006-2007, Ext JS, LLC.
38600  *
38601  * Originally Released Under LGPL - original licence link has changed is not relivant.
38602  *
38603  * Fork - LGPL
38604  * <script type="text/javascript">
38605  */
38606  
38607 /**
38608  * @class Roo.menu.ColorItem
38609  * @extends Roo.menu.Adapter
38610  * A menu item that wraps the {@link Roo.ColorPalette} component.
38611  * @constructor
38612  * Creates a new ColorItem
38613  * @param {Object} config Configuration options
38614  */
38615 Roo.menu.ColorItem = function(config){
38616     Roo.menu.ColorItem.superclass.constructor.call(this, new Roo.ColorPalette(config), config);
38617     /** The Roo.ColorPalette object @type Roo.ColorPalette */
38618     this.palette = this.component;
38619     this.relayEvents(this.palette, ["select"]);
38620     if(this.selectHandler){
38621         this.on('select', this.selectHandler, this.scope);
38622     }
38623 };
38624 Roo.extend(Roo.menu.ColorItem, Roo.menu.Adapter);/*
38625  * Based on:
38626  * Ext JS Library 1.1.1
38627  * Copyright(c) 2006-2007, Ext JS, LLC.
38628  *
38629  * Originally Released Under LGPL - original licence link has changed is not relivant.
38630  *
38631  * Fork - LGPL
38632  * <script type="text/javascript">
38633  */
38634  
38635
38636 /**
38637  * @class Roo.menu.DateMenu
38638  * @extends Roo.menu.Menu
38639  * A menu containing a {@link Roo.menu.DateItem} component (which provides a date picker).
38640  * @constructor
38641  * Creates a new DateMenu
38642  * @param {Object} config Configuration options
38643  */
38644 Roo.menu.DateMenu = function(config){
38645     Roo.menu.DateMenu.superclass.constructor.call(this, config);
38646     this.plain = true;
38647     var di = new Roo.menu.DateItem(config);
38648     this.add(di);
38649     /**
38650      * The {@link Roo.DatePicker} instance for this DateMenu
38651      * @type DatePicker
38652      */
38653     this.picker = di.picker;
38654     /**
38655      * @event select
38656      * @param {DatePicker} picker
38657      * @param {Date} date
38658      */
38659     this.relayEvents(di, ["select"]);
38660     this.on('beforeshow', function(){
38661         if(this.picker){
38662             this.picker.hideMonthPicker(false);
38663         }
38664     }, this);
38665 };
38666 Roo.extend(Roo.menu.DateMenu, Roo.menu.Menu, {
38667     cls:'x-date-menu'
38668 });/*
38669  * Based on:
38670  * Ext JS Library 1.1.1
38671  * Copyright(c) 2006-2007, Ext JS, LLC.
38672  *
38673  * Originally Released Under LGPL - original licence link has changed is not relivant.
38674  *
38675  * Fork - LGPL
38676  * <script type="text/javascript">
38677  */
38678  
38679
38680 /**
38681  * @class Roo.menu.ColorMenu
38682  * @extends Roo.menu.Menu
38683  * A menu containing a {@link Roo.menu.ColorItem} component (which provides a basic color picker).
38684  * @constructor
38685  * Creates a new ColorMenu
38686  * @param {Object} config Configuration options
38687  */
38688 Roo.menu.ColorMenu = function(config){
38689     Roo.menu.ColorMenu.superclass.constructor.call(this, config);
38690     this.plain = true;
38691     var ci = new Roo.menu.ColorItem(config);
38692     this.add(ci);
38693     /**
38694      * The {@link Roo.ColorPalette} instance for this ColorMenu
38695      * @type ColorPalette
38696      */
38697     this.palette = ci.palette;
38698     /**
38699      * @event select
38700      * @param {ColorPalette} palette
38701      * @param {String} color
38702      */
38703     this.relayEvents(ci, ["select"]);
38704 };
38705 Roo.extend(Roo.menu.ColorMenu, Roo.menu.Menu);/*
38706  * Based on:
38707  * Ext JS Library 1.1.1
38708  * Copyright(c) 2006-2007, Ext JS, LLC.
38709  *
38710  * Originally Released Under LGPL - original licence link has changed is not relivant.
38711  *
38712  * Fork - LGPL
38713  * <script type="text/javascript">
38714  */
38715  
38716 /**
38717  * @class Roo.form.TextItem
38718  * @extends Roo.BoxComponent
38719  * Base class for form fields that provides default event handling, sizing, value handling and other functionality.
38720  * @constructor
38721  * Creates a new TextItem
38722  * @param {Object} config Configuration options
38723  */
38724 Roo.form.TextItem = function(config){
38725     Roo.form.TextItem.superclass.constructor.call(this, config);
38726 };
38727
38728 Roo.extend(Roo.form.TextItem, Roo.BoxComponent,  {
38729     
38730     /**
38731      * @cfg {String} tag the tag for this item (default div)
38732      */
38733     tag : 'div',
38734     /**
38735      * @cfg {String} html the content for this item
38736      */
38737     html : '',
38738     
38739     getAutoCreate : function()
38740     {
38741         var cfg = {
38742             id: this.id,
38743             tag: this.tag,
38744             html: this.html,
38745             cls: 'x-form-item'
38746         };
38747         
38748         return cfg;
38749         
38750     },
38751     
38752     onRender : function(ct, position)
38753     {
38754         Roo.form.TextItem.superclass.onRender.call(this, ct, position);
38755         
38756         if(!this.el){
38757             var cfg = this.getAutoCreate();
38758             if(!cfg.name){
38759                 cfg.name = typeof(this.name) == 'undefined' ? this.id : this.name;
38760             }
38761             if (!cfg.name.length) {
38762                 delete cfg.name;
38763             }
38764             this.el = ct.createChild(cfg, position);
38765         }
38766     }
38767     
38768 });/*
38769  * Based on:
38770  * Ext JS Library 1.1.1
38771  * Copyright(c) 2006-2007, Ext JS, LLC.
38772  *
38773  * Originally Released Under LGPL - original licence link has changed is not relivant.
38774  *
38775  * Fork - LGPL
38776  * <script type="text/javascript">
38777  */
38778  
38779 /**
38780  * @class Roo.form.Field
38781  * @extends Roo.BoxComponent
38782  * Base class for form fields that provides default event handling, sizing, value handling and other functionality.
38783  * @constructor
38784  * Creates a new Field
38785  * @param {Object} config Configuration options
38786  */
38787 Roo.form.Field = function(config){
38788     Roo.form.Field.superclass.constructor.call(this, config);
38789 };
38790
38791 Roo.extend(Roo.form.Field, Roo.BoxComponent,  {
38792     /**
38793      * @cfg {String} fieldLabel Label to use when rendering a form.
38794      */
38795        /**
38796      * @cfg {String} qtip Mouse over tip
38797      */
38798      
38799     /**
38800      * @cfg {String} invalidClass The CSS class to use when marking a field invalid (defaults to "x-form-invalid")
38801      */
38802     invalidClass : "x-form-invalid",
38803     /**
38804      * @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")
38805      */
38806     invalidText : "The value in this field is invalid",
38807     /**
38808      * @cfg {String} focusClass The CSS class to use when the field receives focus (defaults to "x-form-focus")
38809      */
38810     focusClass : "x-form-focus",
38811     /**
38812      * @cfg {String/Boolean} validationEvent The event that should initiate field validation. Set to false to disable
38813       automatic validation (defaults to "keyup").
38814      */
38815     validationEvent : "keyup",
38816     /**
38817      * @cfg {Boolean} validateOnBlur Whether the field should validate when it loses focus (defaults to true).
38818      */
38819     validateOnBlur : true,
38820     /**
38821      * @cfg {Number} validationDelay The length of time in milliseconds after user input begins until validation is initiated (defaults to 250)
38822      */
38823     validationDelay : 250,
38824     /**
38825      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
38826      * {tag: "input", type: "text", size: "20", autocomplete: "off"})
38827      */
38828     defaultAutoCreate : {tag: "input", type: "text", size: "20", autocomplete: "new-password"},
38829     /**
38830      * @cfg {String} fieldClass The default CSS class for the field (defaults to "x-form-field")
38831      */
38832     fieldClass : "x-form-field",
38833     /**
38834      * @cfg {String} msgTarget The location where error text should display.  Should be one of the following values (defaults to 'qtip'):
38835      *<pre>
38836 Value         Description
38837 -----------   ----------------------------------------------------------------------
38838 qtip          Display a quick tip when the user hovers over the field
38839 title         Display a default browser title attribute popup
38840 under         Add a block div beneath the field containing the error text
38841 side          Add an error icon to the right of the field with a popup on hover
38842 [element id]  Add the error text directly to the innerHTML of the specified element
38843 </pre>
38844      */
38845     msgTarget : 'qtip',
38846     /**
38847      * @cfg {String} msgFx <b>Experimental</b> The effect used when displaying a validation message under the field (defaults to 'normal').
38848      */
38849     msgFx : 'normal',
38850
38851     /**
38852      * @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.
38853      */
38854     readOnly : false,
38855
38856     /**
38857      * @cfg {Boolean} disabled True to disable the field (defaults to false).
38858      */
38859     disabled : false,
38860
38861     /**
38862      * @cfg {String} inputType The type attribute for input fields -- e.g. radio, text, password (defaults to "text").
38863      */
38864     inputType : undefined,
38865     
38866     /**
38867      * @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).
38868          */
38869         tabIndex : undefined,
38870         
38871     // private
38872     isFormField : true,
38873
38874     // private
38875     hasFocus : false,
38876     /**
38877      * @property {Roo.Element} fieldEl
38878      * Element Containing the rendered Field (with label etc.)
38879      */
38880     /**
38881      * @cfg {Mixed} value A value to initialize this field with.
38882      */
38883     value : undefined,
38884
38885     /**
38886      * @cfg {String} name The field's HTML name attribute.
38887      */
38888     /**
38889      * @cfg {String} cls A CSS class to apply to the field's underlying element.
38890      */
38891     // private
38892     loadedValue : false,
38893      
38894      
38895         // private ??
38896         initComponent : function(){
38897         Roo.form.Field.superclass.initComponent.call(this);
38898         this.addEvents({
38899             /**
38900              * @event focus
38901              * Fires when this field receives input focus.
38902              * @param {Roo.form.Field} this
38903              */
38904             focus : true,
38905             /**
38906              * @event blur
38907              * Fires when this field loses input focus.
38908              * @param {Roo.form.Field} this
38909              */
38910             blur : true,
38911             /**
38912              * @event specialkey
38913              * Fires when any key related to navigation (arrows, tab, enter, esc, etc.) is pressed.  You can check
38914              * {@link Roo.EventObject#getKey} to determine which key was pressed.
38915              * @param {Roo.form.Field} this
38916              * @param {Roo.EventObject} e The event object
38917              */
38918             specialkey : true,
38919             /**
38920              * @event change
38921              * Fires just before the field blurs if the field value has changed.
38922              * @param {Roo.form.Field} this
38923              * @param {Mixed} newValue The new value
38924              * @param {Mixed} oldValue The original value
38925              */
38926             change : true,
38927             /**
38928              * @event invalid
38929              * Fires after the field has been marked as invalid.
38930              * @param {Roo.form.Field} this
38931              * @param {String} msg The validation message
38932              */
38933             invalid : true,
38934             /**
38935              * @event valid
38936              * Fires after the field has been validated with no errors.
38937              * @param {Roo.form.Field} this
38938              */
38939             valid : true,
38940              /**
38941              * @event keyup
38942              * Fires after the key up
38943              * @param {Roo.form.Field} this
38944              * @param {Roo.EventObject}  e The event Object
38945              */
38946             keyup : true
38947         });
38948     },
38949
38950     /**
38951      * Returns the name attribute of the field if available
38952      * @return {String} name The field name
38953      */
38954     getName: function(){
38955          return this.rendered && this.el.dom.name ? this.el.dom.name : (this.hiddenName || '');
38956     },
38957
38958     // private
38959     onRender : function(ct, position){
38960         Roo.form.Field.superclass.onRender.call(this, ct, position);
38961         if(!this.el){
38962             var cfg = this.getAutoCreate();
38963             if(!cfg.name){
38964                 cfg.name = typeof(this.name) == 'undefined' ? this.id : this.name;
38965             }
38966             if (!cfg.name.length) {
38967                 delete cfg.name;
38968             }
38969             if(this.inputType){
38970                 cfg.type = this.inputType;
38971             }
38972             this.el = ct.createChild(cfg, position);
38973         }
38974         var type = this.el.dom.type;
38975         if(type){
38976             if(type == 'password'){
38977                 type = 'text';
38978             }
38979             this.el.addClass('x-form-'+type);
38980         }
38981         if(this.readOnly){
38982             this.el.dom.readOnly = true;
38983         }
38984         if(this.tabIndex !== undefined){
38985             this.el.dom.setAttribute('tabIndex', this.tabIndex);
38986         }
38987
38988         this.el.addClass([this.fieldClass, this.cls]);
38989         this.initValue();
38990     },
38991
38992     /**
38993      * Apply the behaviors of this component to an existing element. <b>This is used instead of render().</b>
38994      * @param {String/HTMLElement/Element} el The id of the node, a DOM node or an existing Element
38995      * @return {Roo.form.Field} this
38996      */
38997     applyTo : function(target){
38998         this.allowDomMove = false;
38999         this.el = Roo.get(target);
39000         this.render(this.el.dom.parentNode);
39001         return this;
39002     },
39003
39004     // private
39005     initValue : function(){
39006         if(this.value !== undefined){
39007             this.setValue(this.value);
39008         }else if(this.el.dom.value.length > 0){
39009             this.setValue(this.el.dom.value);
39010         }
39011     },
39012
39013     /**
39014      * Returns true if this field has been changed since it was originally loaded and is not disabled.
39015      * DEPRICATED  - it never worked well - use hasChanged/resetHasChanged.
39016      */
39017     isDirty : function() {
39018         if(this.disabled) {
39019             return false;
39020         }
39021         return String(this.getValue()) !== String(this.originalValue);
39022     },
39023
39024     /**
39025      * stores the current value in loadedValue
39026      */
39027     resetHasChanged : function()
39028     {
39029         this.loadedValue = String(this.getValue());
39030     },
39031     /**
39032      * checks the current value against the 'loaded' value.
39033      * Note - will return false if 'resetHasChanged' has not been called first.
39034      */
39035     hasChanged : function()
39036     {
39037         if(this.disabled || this.readOnly) {
39038             return false;
39039         }
39040         return this.loadedValue !== false && String(this.getValue()) !== this.loadedValue;
39041     },
39042     
39043     
39044     
39045     // private
39046     afterRender : function(){
39047         Roo.form.Field.superclass.afterRender.call(this);
39048         this.initEvents();
39049     },
39050
39051     // private
39052     fireKey : function(e){
39053         //Roo.log('field ' + e.getKey());
39054         if(e.isNavKeyPress()){
39055             this.fireEvent("specialkey", this, e);
39056         }
39057     },
39058
39059     /**
39060      * Resets the current field value to the originally loaded value and clears any validation messages
39061      */
39062     reset : function(){
39063         this.setValue(this.resetValue);
39064         this.originalValue = this.getValue();
39065         this.clearInvalid();
39066     },
39067
39068     // private
39069     initEvents : function(){
39070         // safari killled keypress - so keydown is now used..
39071         this.el.on("keydown" , this.fireKey,  this);
39072         this.el.on("focus", this.onFocus,  this);
39073         this.el.on("blur", this.onBlur,  this);
39074         this.el.relayEvent('keyup', this);
39075
39076         // reference to original value for reset
39077         this.originalValue = this.getValue();
39078         this.resetValue =  this.getValue();
39079     },
39080
39081     // private
39082     onFocus : function(){
39083         if(!Roo.isOpera && this.focusClass){ // don't touch in Opera
39084             this.el.addClass(this.focusClass);
39085         }
39086         if(!this.hasFocus){
39087             this.hasFocus = true;
39088             this.startValue = this.getValue();
39089             this.fireEvent("focus", this);
39090         }
39091     },
39092
39093     beforeBlur : Roo.emptyFn,
39094
39095     // private
39096     onBlur : function(){
39097         this.beforeBlur();
39098         if(!Roo.isOpera && this.focusClass){ // don't touch in Opera
39099             this.el.removeClass(this.focusClass);
39100         }
39101         this.hasFocus = false;
39102         if(this.validationEvent !== false && this.validateOnBlur && this.validationEvent != "blur"){
39103             this.validate();
39104         }
39105         var v = this.getValue();
39106         if(String(v) !== String(this.startValue)){
39107             this.fireEvent('change', this, v, this.startValue);
39108         }
39109         this.fireEvent("blur", this);
39110     },
39111
39112     /**
39113      * Returns whether or not the field value is currently valid
39114      * @param {Boolean} preventMark True to disable marking the field invalid
39115      * @return {Boolean} True if the value is valid, else false
39116      */
39117     isValid : function(preventMark){
39118         if(this.disabled){
39119             return true;
39120         }
39121         var restore = this.preventMark;
39122         this.preventMark = preventMark === true;
39123         var v = this.validateValue(this.processValue(this.getRawValue()));
39124         this.preventMark = restore;
39125         return v;
39126     },
39127
39128     /**
39129      * Validates the field value
39130      * @return {Boolean} True if the value is valid, else false
39131      */
39132     validate : function(){
39133         if(this.disabled || this.validateValue(this.processValue(this.getRawValue()))){
39134             this.clearInvalid();
39135             return true;
39136         }
39137         return false;
39138     },
39139
39140     processValue : function(value){
39141         return value;
39142     },
39143
39144     // private
39145     // Subclasses should provide the validation implementation by overriding this
39146     validateValue : function(value){
39147         return true;
39148     },
39149
39150     /**
39151      * Mark this field as invalid
39152      * @param {String} msg The validation message
39153      */
39154     markInvalid : function(msg){
39155         if(!this.rendered || this.preventMark){ // not rendered
39156             return;
39157         }
39158         
39159         var obj = (typeof(this.combo) != 'undefined') ? this.combo : this; // fix the combox array!!
39160         
39161         obj.el.addClass(this.invalidClass);
39162         msg = msg || this.invalidText;
39163         switch(this.msgTarget){
39164             case 'qtip':
39165                 obj.el.dom.qtip = msg;
39166                 obj.el.dom.qclass = 'x-form-invalid-tip';
39167                 if(Roo.QuickTips){ // fix for floating editors interacting with DND
39168                     Roo.QuickTips.enable();
39169                 }
39170                 break;
39171             case 'title':
39172                 this.el.dom.title = msg;
39173                 break;
39174             case 'under':
39175                 if(!this.errorEl){
39176                     var elp = this.el.findParent('.x-form-element', 5, true);
39177                     this.errorEl = elp.createChild({cls:'x-form-invalid-msg'});
39178                     this.errorEl.setWidth(elp.getWidth(true)-20);
39179                 }
39180                 this.errorEl.update(msg);
39181                 Roo.form.Field.msgFx[this.msgFx].show(this.errorEl, this);
39182                 break;
39183             case 'side':
39184                 if(!this.errorIcon){
39185                     var elp = this.el.findParent('.x-form-element', 5, true);
39186                     this.errorIcon = elp.createChild({cls:'x-form-invalid-icon'});
39187                 }
39188                 this.alignErrorIcon();
39189                 this.errorIcon.dom.qtip = msg;
39190                 this.errorIcon.dom.qclass = 'x-form-invalid-tip';
39191                 this.errorIcon.show();
39192                 this.on('resize', this.alignErrorIcon, this);
39193                 break;
39194             default:
39195                 var t = Roo.getDom(this.msgTarget);
39196                 t.innerHTML = msg;
39197                 t.style.display = this.msgDisplay;
39198                 break;
39199         }
39200         this.fireEvent('invalid', this, msg);
39201     },
39202
39203     // private
39204     alignErrorIcon : function(){
39205         this.errorIcon.alignTo(this.el, 'tl-tr', [2, 0]);
39206     },
39207
39208     /**
39209      * Clear any invalid styles/messages for this field
39210      */
39211     clearInvalid : function(){
39212         if(!this.rendered || this.preventMark){ // not rendered
39213             return;
39214         }
39215         var obj = (typeof(this.combo) != 'undefined') ? this.combo : this; // fix the combox array!!
39216         
39217         obj.el.removeClass(this.invalidClass);
39218         switch(this.msgTarget){
39219             case 'qtip':
39220                 obj.el.dom.qtip = '';
39221                 break;
39222             case 'title':
39223                 this.el.dom.title = '';
39224                 break;
39225             case 'under':
39226                 if(this.errorEl){
39227                     Roo.form.Field.msgFx[this.msgFx].hide(this.errorEl, this);
39228                 }
39229                 break;
39230             case 'side':
39231                 if(this.errorIcon){
39232                     this.errorIcon.dom.qtip = '';
39233                     this.errorIcon.hide();
39234                     this.un('resize', this.alignErrorIcon, this);
39235                 }
39236                 break;
39237             default:
39238                 var t = Roo.getDom(this.msgTarget);
39239                 t.innerHTML = '';
39240                 t.style.display = 'none';
39241                 break;
39242         }
39243         this.fireEvent('valid', this);
39244     },
39245
39246     /**
39247      * Returns the raw data value which may or may not be a valid, defined value.  To return a normalized value see {@link #getValue}.
39248      * @return {Mixed} value The field value
39249      */
39250     getRawValue : function(){
39251         var v = this.el.getValue();
39252         
39253         return v;
39254     },
39255
39256     /**
39257      * Returns the normalized data value (undefined or emptyText will be returned as '').  To return the raw value see {@link #getRawValue}.
39258      * @return {Mixed} value The field value
39259      */
39260     getValue : function(){
39261         var v = this.el.getValue();
39262          
39263         return v;
39264     },
39265
39266     /**
39267      * Sets the underlying DOM field's value directly, bypassing validation.  To set the value with validation see {@link #setValue}.
39268      * @param {Mixed} value The value to set
39269      */
39270     setRawValue : function(v){
39271         return this.el.dom.value = (v === null || v === undefined ? '' : v);
39272     },
39273
39274     /**
39275      * Sets a data value into the field and validates it.  To set the value directly without validation see {@link #setRawValue}.
39276      * @param {Mixed} value The value to set
39277      */
39278     setValue : function(v){
39279         this.value = v;
39280         if(this.rendered){
39281             this.el.dom.value = (v === null || v === undefined ? '' : v);
39282              this.validate();
39283         }
39284     },
39285
39286     adjustSize : function(w, h){
39287         var s = Roo.form.Field.superclass.adjustSize.call(this, w, h);
39288         s.width = this.adjustWidth(this.el.dom.tagName, s.width);
39289         return s;
39290     },
39291
39292     adjustWidth : function(tag, w){
39293         tag = tag.toLowerCase();
39294         if(typeof w == 'number' && Roo.isStrict && !Roo.isSafari){
39295             if(Roo.isIE && (tag == 'input' || tag == 'textarea')){
39296                 if(tag == 'input'){
39297                     return w + 2;
39298                 }
39299                 if(tag == 'textarea'){
39300                     return w-2;
39301                 }
39302             }else if(Roo.isOpera){
39303                 if(tag == 'input'){
39304                     return w + 2;
39305                 }
39306                 if(tag == 'textarea'){
39307                     return w-2;
39308                 }
39309             }
39310         }
39311         return w;
39312     }
39313 });
39314
39315
39316 // anything other than normal should be considered experimental
39317 Roo.form.Field.msgFx = {
39318     normal : {
39319         show: function(msgEl, f){
39320             msgEl.setDisplayed('block');
39321         },
39322
39323         hide : function(msgEl, f){
39324             msgEl.setDisplayed(false).update('');
39325         }
39326     },
39327
39328     slide : {
39329         show: function(msgEl, f){
39330             msgEl.slideIn('t', {stopFx:true});
39331         },
39332
39333         hide : function(msgEl, f){
39334             msgEl.slideOut('t', {stopFx:true,useDisplay:true});
39335         }
39336     },
39337
39338     slideRight : {
39339         show: function(msgEl, f){
39340             msgEl.fixDisplay();
39341             msgEl.alignTo(f.el, 'tl-tr');
39342             msgEl.slideIn('l', {stopFx:true});
39343         },
39344
39345         hide : function(msgEl, f){
39346             msgEl.slideOut('l', {stopFx:true,useDisplay:true});
39347         }
39348     }
39349 };/*
39350  * Based on:
39351  * Ext JS Library 1.1.1
39352  * Copyright(c) 2006-2007, Ext JS, LLC.
39353  *
39354  * Originally Released Under LGPL - original licence link has changed is not relivant.
39355  *
39356  * Fork - LGPL
39357  * <script type="text/javascript">
39358  */
39359  
39360
39361 /**
39362  * @class Roo.form.TextField
39363  * @extends Roo.form.Field
39364  * Basic text field.  Can be used as a direct replacement for traditional text inputs, or as the base
39365  * class for more sophisticated input controls (like {@link Roo.form.TextArea} and {@link Roo.form.ComboBox}).
39366  * @constructor
39367  * Creates a new TextField
39368  * @param {Object} config Configuration options
39369  */
39370 Roo.form.TextField = function(config){
39371     Roo.form.TextField.superclass.constructor.call(this, config);
39372     this.addEvents({
39373         /**
39374          * @event autosize
39375          * Fires when the autosize function is triggered.  The field may or may not have actually changed size
39376          * according to the default logic, but this event provides a hook for the developer to apply additional
39377          * logic at runtime to resize the field if needed.
39378              * @param {Roo.form.Field} this This text field
39379              * @param {Number} width The new field width
39380              */
39381         autosize : true
39382     });
39383 };
39384
39385 Roo.extend(Roo.form.TextField, Roo.form.Field,  {
39386     /**
39387      * @cfg {Boolean} grow True if this field should automatically grow and shrink to its content
39388      */
39389     grow : false,
39390     /**
39391      * @cfg {Number} growMin The minimum width to allow when grow = true (defaults to 30)
39392      */
39393     growMin : 30,
39394     /**
39395      * @cfg {Number} growMax The maximum width to allow when grow = true (defaults to 800)
39396      */
39397     growMax : 800,
39398     /**
39399      * @cfg {String} vtype A validation type name as defined in {@link Roo.form.VTypes} (defaults to null)
39400      */
39401     vtype : null,
39402     /**
39403      * @cfg {String} maskRe An input mask regular expression that will be used to filter keystrokes that don't match (defaults to null)
39404      */
39405     maskRe : null,
39406     /**
39407      * @cfg {Boolean} disableKeyFilter True to disable input keystroke filtering (defaults to false)
39408      */
39409     disableKeyFilter : false,
39410     /**
39411      * @cfg {Boolean} allowBlank False to validate that the value length > 0 (defaults to true)
39412      */
39413     allowBlank : true,
39414     /**
39415      * @cfg {Number} minLength Minimum input field length required (defaults to 0)
39416      */
39417     minLength : 0,
39418     /**
39419      * @cfg {Number} maxLength Maximum input field length allowed (defaults to Number.MAX_VALUE)
39420      */
39421     maxLength : Number.MAX_VALUE,
39422     /**
39423      * @cfg {String} minLengthText Error text to display if the minimum length validation fails (defaults to "The minimum length for this field is {minLength}")
39424      */
39425     minLengthText : "The minimum length for this field is {0}",
39426     /**
39427      * @cfg {String} maxLengthText Error text to display if the maximum length validation fails (defaults to "The maximum length for this field is {maxLength}")
39428      */
39429     maxLengthText : "The maximum length for this field is {0}",
39430     /**
39431      * @cfg {Boolean} selectOnFocus True to automatically select any existing field text when the field receives input focus (defaults to false)
39432      */
39433     selectOnFocus : false,
39434     /**
39435      * @cfg {String} blankText Error text to display if the allow blank validation fails (defaults to "This field is required")
39436      */
39437     blankText : "This field is required",
39438     /**
39439      * @cfg {Function} validator A custom validation function to be called during field validation (defaults to null).
39440      * If available, this function will be called only after the basic validators all return true, and will be passed the
39441      * current field value and expected to return boolean true if the value is valid or a string error message if invalid.
39442      */
39443     validator : null,
39444     /**
39445      * @cfg {RegExp} regex A JavaScript RegExp object to be tested against the field value during validation (defaults to null).
39446      * If available, this regex will be evaluated only after the basic validators all return true, and will be passed the
39447      * current field value.  If the test fails, the field will be marked invalid using {@link #regexText}.
39448      */
39449     regex : null,
39450     /**
39451      * @cfg {String} regexText The error text to display if {@link #regex} is used and the test fails during validation (defaults to "")
39452      */
39453     regexText : "",
39454     /**
39455      * @cfg {String} emptyText The default text to display in an empty field - placeholder... (defaults to null).
39456      */
39457     emptyText : null,
39458    
39459
39460     // private
39461     initEvents : function()
39462     {
39463         if (this.emptyText) {
39464             this.el.attr('placeholder', this.emptyText);
39465         }
39466         
39467         Roo.form.TextField.superclass.initEvents.call(this);
39468         if(this.validationEvent == 'keyup'){
39469             this.validationTask = new Roo.util.DelayedTask(this.validate, this);
39470             this.el.on('keyup', this.filterValidation, this);
39471         }
39472         else if(this.validationEvent !== false){
39473             this.el.on(this.validationEvent, this.validate, this, {buffer: this.validationDelay});
39474         }
39475         
39476         if(this.selectOnFocus){
39477             this.on("focus", this.preFocus, this);
39478             
39479         }
39480         if(this.maskRe || (this.vtype && this.disableKeyFilter !== true && (this.maskRe = Roo.form.VTypes[this.vtype+'Mask']))){
39481             this.el.on("keypress", this.filterKeys, this);
39482         }
39483         if(this.grow){
39484             this.el.on("keyup", this.onKeyUp,  this, {buffer:50});
39485             this.el.on("click", this.autoSize,  this);
39486         }
39487         if(this.el.is('input[type=password]') && Roo.isSafari){
39488             this.el.on('keydown', this.SafariOnKeyDown, this);
39489         }
39490     },
39491
39492     processValue : function(value){
39493         if(this.stripCharsRe){
39494             var newValue = value.replace(this.stripCharsRe, '');
39495             if(newValue !== value){
39496                 this.setRawValue(newValue);
39497                 return newValue;
39498             }
39499         }
39500         return value;
39501     },
39502
39503     filterValidation : function(e){
39504         if(!e.isNavKeyPress()){
39505             this.validationTask.delay(this.validationDelay);
39506         }
39507     },
39508
39509     // private
39510     onKeyUp : function(e){
39511         if(!e.isNavKeyPress()){
39512             this.autoSize();
39513         }
39514     },
39515
39516     /**
39517      * Resets the current field value to the originally-loaded value and clears any validation messages.
39518      *  
39519      */
39520     reset : function(){
39521         Roo.form.TextField.superclass.reset.call(this);
39522        
39523     },
39524
39525     
39526     // private
39527     preFocus : function(){
39528         
39529         if(this.selectOnFocus){
39530             this.el.dom.select();
39531         }
39532     },
39533
39534     
39535     // private
39536     filterKeys : function(e){
39537         var k = e.getKey();
39538         if(!Roo.isIE && (e.isNavKeyPress() || k == e.BACKSPACE || (k == e.DELETE && e.button == -1))){
39539             return;
39540         }
39541         var c = e.getCharCode(), cc = String.fromCharCode(c);
39542         if(Roo.isIE && (e.isSpecialKey() || !cc)){
39543             return;
39544         }
39545         if(!this.maskRe.test(cc)){
39546             e.stopEvent();
39547         }
39548     },
39549
39550     setValue : function(v){
39551         
39552         Roo.form.TextField.superclass.setValue.apply(this, arguments);
39553         
39554         this.autoSize();
39555     },
39556
39557     /**
39558      * Validates a value according to the field's validation rules and marks the field as invalid
39559      * if the validation fails
39560      * @param {Mixed} value The value to validate
39561      * @return {Boolean} True if the value is valid, else false
39562      */
39563     validateValue : function(value){
39564         if(value.length < 1)  { // if it's blank
39565              if(this.allowBlank){
39566                 this.clearInvalid();
39567                 return true;
39568              }else{
39569                 this.markInvalid(this.blankText);
39570                 return false;
39571              }
39572         }
39573         if(value.length < this.minLength){
39574             this.markInvalid(String.format(this.minLengthText, this.minLength));
39575             return false;
39576         }
39577         if(value.length > this.maxLength){
39578             this.markInvalid(String.format(this.maxLengthText, this.maxLength));
39579             return false;
39580         }
39581         if(this.vtype){
39582             var vt = Roo.form.VTypes;
39583             if(!vt[this.vtype](value, this)){
39584                 this.markInvalid(this.vtypeText || vt[this.vtype +'Text']);
39585                 return false;
39586             }
39587         }
39588         if(typeof this.validator == "function"){
39589             var msg = this.validator(value);
39590             if(msg !== true){
39591                 this.markInvalid(msg);
39592                 return false;
39593             }
39594         }
39595         if(this.regex && !this.regex.test(value)){
39596             this.markInvalid(this.regexText);
39597             return false;
39598         }
39599         return true;
39600     },
39601
39602     /**
39603      * Selects text in this field
39604      * @param {Number} start (optional) The index where the selection should start (defaults to 0)
39605      * @param {Number} end (optional) The index where the selection should end (defaults to the text length)
39606      */
39607     selectText : function(start, end){
39608         var v = this.getRawValue();
39609         if(v.length > 0){
39610             start = start === undefined ? 0 : start;
39611             end = end === undefined ? v.length : end;
39612             var d = this.el.dom;
39613             if(d.setSelectionRange){
39614                 d.setSelectionRange(start, end);
39615             }else if(d.createTextRange){
39616                 var range = d.createTextRange();
39617                 range.moveStart("character", start);
39618                 range.moveEnd("character", v.length-end);
39619                 range.select();
39620             }
39621         }
39622     },
39623
39624     /**
39625      * Automatically grows the field to accomodate the width of the text up to the maximum field width allowed.
39626      * This only takes effect if grow = true, and fires the autosize event.
39627      */
39628     autoSize : function(){
39629         if(!this.grow || !this.rendered){
39630             return;
39631         }
39632         if(!this.metrics){
39633             this.metrics = Roo.util.TextMetrics.createInstance(this.el);
39634         }
39635         var el = this.el;
39636         var v = el.dom.value;
39637         var d = document.createElement('div');
39638         d.appendChild(document.createTextNode(v));
39639         v = d.innerHTML;
39640         d = null;
39641         v += "&#160;";
39642         var w = Math.min(this.growMax, Math.max(this.metrics.getWidth(v) + /* add extra padding */ 10, this.growMin));
39643         this.el.setWidth(w);
39644         this.fireEvent("autosize", this, w);
39645     },
39646     
39647     // private
39648     SafariOnKeyDown : function(event)
39649     {
39650         // this is a workaround for a password hang bug on chrome/ webkit.
39651         
39652         var isSelectAll = false;
39653         
39654         if(this.el.dom.selectionEnd > 0){
39655             isSelectAll = (this.el.dom.selectionEnd - this.el.dom.selectionStart - this.getValue().length == 0) ? true : false;
39656         }
39657         if(((event.getKey() == 8 || event.getKey() == 46) && this.getValue().length ==1)){ // backspace and delete key
39658             event.preventDefault();
39659             this.setValue('');
39660             return;
39661         }
39662         
39663         if(isSelectAll && event.getCharCode() > 31){ // backspace and delete key
39664             
39665             event.preventDefault();
39666             // this is very hacky as keydown always get's upper case.
39667             
39668             var cc = String.fromCharCode(event.getCharCode());
39669             
39670             
39671             this.setValue( event.shiftKey ?  cc : cc.toLowerCase());
39672             
39673         }
39674         
39675         
39676     }
39677 });/*
39678  * Based on:
39679  * Ext JS Library 1.1.1
39680  * Copyright(c) 2006-2007, Ext JS, LLC.
39681  *
39682  * Originally Released Under LGPL - original licence link has changed is not relivant.
39683  *
39684  * Fork - LGPL
39685  * <script type="text/javascript">
39686  */
39687  
39688 /**
39689  * @class Roo.form.Hidden
39690  * @extends Roo.form.TextField
39691  * Simple Hidden element used on forms 
39692  * 
39693  * usage: form.add(new Roo.form.HiddenField({ 'name' : 'test1' }));
39694  * 
39695  * @constructor
39696  * Creates a new Hidden form element.
39697  * @param {Object} config Configuration options
39698  */
39699
39700
39701
39702 // easy hidden field...
39703 Roo.form.Hidden = function(config){
39704     Roo.form.Hidden.superclass.constructor.call(this, config);
39705 };
39706   
39707 Roo.extend(Roo.form.Hidden, Roo.form.TextField, {
39708     fieldLabel:      '',
39709     inputType:      'hidden',
39710     width:          50,
39711     allowBlank:     true,
39712     labelSeparator: '',
39713     hidden:         true,
39714     itemCls :       'x-form-item-display-none'
39715
39716
39717 });
39718
39719
39720 /*
39721  * Based on:
39722  * Ext JS Library 1.1.1
39723  * Copyright(c) 2006-2007, Ext JS, LLC.
39724  *
39725  * Originally Released Under LGPL - original licence link has changed is not relivant.
39726  *
39727  * Fork - LGPL
39728  * <script type="text/javascript">
39729  */
39730  
39731 /**
39732  * @class Roo.form.TriggerField
39733  * @extends Roo.form.TextField
39734  * Provides a convenient wrapper for TextFields that adds a clickable trigger button (looks like a combobox by default).
39735  * The trigger has no default action, so you must assign a function to implement the trigger click handler by
39736  * overriding {@link #onTriggerClick}. You can create a TriggerField directly, as it renders exactly like a combobox
39737  * for which you can provide a custom implementation.  For example:
39738  * <pre><code>
39739 var trigger = new Roo.form.TriggerField();
39740 trigger.onTriggerClick = myTriggerFn;
39741 trigger.applyTo('my-field');
39742 </code></pre>
39743  *
39744  * However, in general you will most likely want to use TriggerField as the base class for a reusable component.
39745  * {@link Roo.form.DateField} and {@link Roo.form.ComboBox} are perfect examples of this.
39746  * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
39747  * class 'x-form-trigger' by default and triggerClass will be <b>appended</b> if specified.
39748  * @constructor
39749  * Create a new TriggerField.
39750  * @param {Object} config Configuration options (valid {@Roo.form.TextField} config options will also be applied
39751  * to the base TextField)
39752  */
39753 Roo.form.TriggerField = function(config){
39754     this.mimicing = false;
39755     Roo.form.TriggerField.superclass.constructor.call(this, config);
39756 };
39757
39758 Roo.extend(Roo.form.TriggerField, Roo.form.TextField,  {
39759     /**
39760      * @cfg {String} triggerClass A CSS class to apply to the trigger
39761      */
39762     /**
39763      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
39764      * {tag: "input", type: "text", size: "16", autocomplete: "off"})
39765      */
39766     defaultAutoCreate : {tag: "input", type: "text", size: "16", autocomplete: "new-password"},
39767     /**
39768      * @cfg {Boolean} hideTrigger True to hide the trigger element and display only the base text field (defaults to false)
39769      */
39770     hideTrigger:false,
39771
39772     /** @cfg {Boolean} grow @hide */
39773     /** @cfg {Number} growMin @hide */
39774     /** @cfg {Number} growMax @hide */
39775
39776     /**
39777      * @hide 
39778      * @method
39779      */
39780     autoSize: Roo.emptyFn,
39781     // private
39782     monitorTab : true,
39783     // private
39784     deferHeight : true,
39785
39786     
39787     actionMode : 'wrap',
39788     // private
39789     onResize : function(w, h){
39790         Roo.form.TriggerField.superclass.onResize.apply(this, arguments);
39791         if(typeof w == 'number'){
39792             var x = w - this.trigger.getWidth();
39793             this.el.setWidth(this.adjustWidth('input', x));
39794             this.trigger.setStyle('left', x+'px');
39795         }
39796     },
39797
39798     // private
39799     adjustSize : Roo.BoxComponent.prototype.adjustSize,
39800
39801     // private
39802     getResizeEl : function(){
39803         return this.wrap;
39804     },
39805
39806     // private
39807     getPositionEl : function(){
39808         return this.wrap;
39809     },
39810
39811     // private
39812     alignErrorIcon : function(){
39813         this.errorIcon.alignTo(this.wrap, 'tl-tr', [2, 0]);
39814     },
39815
39816     // private
39817     onRender : function(ct, position){
39818         Roo.form.TriggerField.superclass.onRender.call(this, ct, position);
39819         this.wrap = this.el.wrap({cls: "x-form-field-wrap"});
39820         this.trigger = this.wrap.createChild(this.triggerConfig ||
39821                 {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.triggerClass});
39822         if(this.hideTrigger){
39823             this.trigger.setDisplayed(false);
39824         }
39825         this.initTrigger();
39826         if(!this.width){
39827             this.wrap.setWidth(this.el.getWidth()+this.trigger.getWidth());
39828         }
39829     },
39830
39831     // private
39832     initTrigger : function(){
39833         this.trigger.on("click", this.onTriggerClick, this, {preventDefault:true});
39834         this.trigger.addClassOnOver('x-form-trigger-over');
39835         this.trigger.addClassOnClick('x-form-trigger-click');
39836     },
39837
39838     // private
39839     onDestroy : function(){
39840         if(this.trigger){
39841             this.trigger.removeAllListeners();
39842             this.trigger.remove();
39843         }
39844         if(this.wrap){
39845             this.wrap.remove();
39846         }
39847         Roo.form.TriggerField.superclass.onDestroy.call(this);
39848     },
39849
39850     // private
39851     onFocus : function(){
39852         Roo.form.TriggerField.superclass.onFocus.call(this);
39853         if(!this.mimicing){
39854             this.wrap.addClass('x-trigger-wrap-focus');
39855             this.mimicing = true;
39856             Roo.get(Roo.isIE ? document.body : document).on("mousedown", this.mimicBlur, this);
39857             if(this.monitorTab){
39858                 this.el.on("keydown", this.checkTab, this);
39859             }
39860         }
39861     },
39862
39863     // private
39864     checkTab : function(e){
39865         if(e.getKey() == e.TAB){
39866             this.triggerBlur();
39867         }
39868     },
39869
39870     // private
39871     onBlur : function(){
39872         // do nothing
39873     },
39874
39875     // private
39876     mimicBlur : function(e, t){
39877         if(!this.wrap.contains(t) && this.validateBlur()){
39878             this.triggerBlur();
39879         }
39880     },
39881
39882     // private
39883     triggerBlur : function(){
39884         this.mimicing = false;
39885         Roo.get(Roo.isIE ? document.body : document).un("mousedown", this.mimicBlur);
39886         if(this.monitorTab){
39887             this.el.un("keydown", this.checkTab, this);
39888         }
39889         this.wrap.removeClass('x-trigger-wrap-focus');
39890         Roo.form.TriggerField.superclass.onBlur.call(this);
39891     },
39892
39893     // private
39894     // This should be overriden by any subclass that needs to check whether or not the field can be blurred.
39895     validateBlur : function(e, t){
39896         return true;
39897     },
39898
39899     // private
39900     onDisable : function(){
39901         Roo.form.TriggerField.superclass.onDisable.call(this);
39902         if(this.wrap){
39903             this.wrap.addClass('x-item-disabled');
39904         }
39905     },
39906
39907     // private
39908     onEnable : function(){
39909         Roo.form.TriggerField.superclass.onEnable.call(this);
39910         if(this.wrap){
39911             this.wrap.removeClass('x-item-disabled');
39912         }
39913     },
39914
39915     // private
39916     onShow : function(){
39917         var ae = this.getActionEl();
39918         
39919         if(ae){
39920             ae.dom.style.display = '';
39921             ae.dom.style.visibility = 'visible';
39922         }
39923     },
39924
39925     // private
39926     
39927     onHide : function(){
39928         var ae = this.getActionEl();
39929         ae.dom.style.display = 'none';
39930     },
39931
39932     /**
39933      * The function that should handle the trigger's click event.  This method does nothing by default until overridden
39934      * by an implementing function.
39935      * @method
39936      * @param {EventObject} e
39937      */
39938     onTriggerClick : Roo.emptyFn
39939 });
39940
39941 // TwinTriggerField is not a public class to be used directly.  It is meant as an abstract base class
39942 // to be extended by an implementing class.  For an example of implementing this class, see the custom
39943 // SearchField implementation here: http://extjs.com/deploy/ext/examples/form/custom.html
39944 Roo.form.TwinTriggerField = Roo.extend(Roo.form.TriggerField, {
39945     initComponent : function(){
39946         Roo.form.TwinTriggerField.superclass.initComponent.call(this);
39947
39948         this.triggerConfig = {
39949             tag:'span', cls:'x-form-twin-triggers', cn:[
39950             {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.trigger1Class},
39951             {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.trigger2Class}
39952         ]};
39953     },
39954
39955     getTrigger : function(index){
39956         return this.triggers[index];
39957     },
39958
39959     initTrigger : function(){
39960         var ts = this.trigger.select('.x-form-trigger', true);
39961         this.wrap.setStyle('overflow', 'hidden');
39962         var triggerField = this;
39963         ts.each(function(t, all, index){
39964             t.hide = function(){
39965                 var w = triggerField.wrap.getWidth();
39966                 this.dom.style.display = 'none';
39967                 triggerField.el.setWidth(w-triggerField.trigger.getWidth());
39968             };
39969             t.show = function(){
39970                 var w = triggerField.wrap.getWidth();
39971                 this.dom.style.display = '';
39972                 triggerField.el.setWidth(w-triggerField.trigger.getWidth());
39973             };
39974             var triggerIndex = 'Trigger'+(index+1);
39975
39976             if(this['hide'+triggerIndex]){
39977                 t.dom.style.display = 'none';
39978             }
39979             t.on("click", this['on'+triggerIndex+'Click'], this, {preventDefault:true});
39980             t.addClassOnOver('x-form-trigger-over');
39981             t.addClassOnClick('x-form-trigger-click');
39982         }, this);
39983         this.triggers = ts.elements;
39984     },
39985
39986     onTrigger1Click : Roo.emptyFn,
39987     onTrigger2Click : Roo.emptyFn
39988 });/*
39989  * Based on:
39990  * Ext JS Library 1.1.1
39991  * Copyright(c) 2006-2007, Ext JS, LLC.
39992  *
39993  * Originally Released Under LGPL - original licence link has changed is not relivant.
39994  *
39995  * Fork - LGPL
39996  * <script type="text/javascript">
39997  */
39998  
39999 /**
40000  * @class Roo.form.TextArea
40001  * @extends Roo.form.TextField
40002  * Multiline text field.  Can be used as a direct replacement for traditional textarea fields, plus adds
40003  * support for auto-sizing.
40004  * @constructor
40005  * Creates a new TextArea
40006  * @param {Object} config Configuration options
40007  */
40008 Roo.form.TextArea = function(config){
40009     Roo.form.TextArea.superclass.constructor.call(this, config);
40010     // these are provided exchanges for backwards compat
40011     // minHeight/maxHeight were replaced by growMin/growMax to be
40012     // compatible with TextField growing config values
40013     if(this.minHeight !== undefined){
40014         this.growMin = this.minHeight;
40015     }
40016     if(this.maxHeight !== undefined){
40017         this.growMax = this.maxHeight;
40018     }
40019 };
40020
40021 Roo.extend(Roo.form.TextArea, Roo.form.TextField,  {
40022     /**
40023      * @cfg {Number} growMin The minimum height to allow when grow = true (defaults to 60)
40024      */
40025     growMin : 60,
40026     /**
40027      * @cfg {Number} growMax The maximum height to allow when grow = true (defaults to 1000)
40028      */
40029     growMax: 1000,
40030     /**
40031      * @cfg {Boolean} preventScrollbars True to prevent scrollbars from appearing regardless of how much text is
40032      * in the field (equivalent to setting overflow: hidden, defaults to false)
40033      */
40034     preventScrollbars: false,
40035     /**
40036      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
40037      * {tag: "textarea", style: "width:300px;height:60px;", autocomplete: "off"})
40038      */
40039
40040     // private
40041     onRender : function(ct, position){
40042         if(!this.el){
40043             this.defaultAutoCreate = {
40044                 tag: "textarea",
40045                 style:"width:300px;height:60px;",
40046                 autocomplete: "new-password"
40047             };
40048         }
40049         Roo.form.TextArea.superclass.onRender.call(this, ct, position);
40050         if(this.grow){
40051             this.textSizeEl = Roo.DomHelper.append(document.body, {
40052                 tag: "pre", cls: "x-form-grow-sizer"
40053             });
40054             if(this.preventScrollbars){
40055                 this.el.setStyle("overflow", "hidden");
40056             }
40057             this.el.setHeight(this.growMin);
40058         }
40059     },
40060
40061     onDestroy : function(){
40062         if(this.textSizeEl){
40063             this.textSizeEl.parentNode.removeChild(this.textSizeEl);
40064         }
40065         Roo.form.TextArea.superclass.onDestroy.call(this);
40066     },
40067
40068     // private
40069     onKeyUp : function(e){
40070         if(!e.isNavKeyPress() || e.getKey() == e.ENTER){
40071             this.autoSize();
40072         }
40073     },
40074
40075     /**
40076      * Automatically grows the field to accomodate the height of the text up to the maximum field height allowed.
40077      * This only takes effect if grow = true, and fires the autosize event if the height changes.
40078      */
40079     autoSize : function(){
40080         if(!this.grow || !this.textSizeEl){
40081             return;
40082         }
40083         var el = this.el;
40084         var v = el.dom.value;
40085         var ts = this.textSizeEl;
40086
40087         ts.innerHTML = '';
40088         ts.appendChild(document.createTextNode(v));
40089         v = ts.innerHTML;
40090
40091         Roo.fly(ts).setWidth(this.el.getWidth());
40092         if(v.length < 1){
40093             v = "&#160;&#160;";
40094         }else{
40095             if(Roo.isIE){
40096                 v = v.replace(/\n/g, '<p>&#160;</p>');
40097             }
40098             v += "&#160;\n&#160;";
40099         }
40100         ts.innerHTML = v;
40101         var h = Math.min(this.growMax, Math.max(ts.offsetHeight, this.growMin));
40102         if(h != this.lastHeight){
40103             this.lastHeight = h;
40104             this.el.setHeight(h);
40105             this.fireEvent("autosize", this, h);
40106         }
40107     }
40108 });/*
40109  * Based on:
40110  * Ext JS Library 1.1.1
40111  * Copyright(c) 2006-2007, Ext JS, LLC.
40112  *
40113  * Originally Released Under LGPL - original licence link has changed is not relivant.
40114  *
40115  * Fork - LGPL
40116  * <script type="text/javascript">
40117  */
40118  
40119
40120 /**
40121  * @class Roo.form.NumberField
40122  * @extends Roo.form.TextField
40123  * Numeric text field that provides automatic keystroke filtering and numeric validation.
40124  * @constructor
40125  * Creates a new NumberField
40126  * @param {Object} config Configuration options
40127  */
40128 Roo.form.NumberField = function(config){
40129     Roo.form.NumberField.superclass.constructor.call(this, config);
40130 };
40131
40132 Roo.extend(Roo.form.NumberField, Roo.form.TextField,  {
40133     /**
40134      * @cfg {String} fieldClass The default CSS class for the field (defaults to "x-form-field x-form-num-field")
40135      */
40136     fieldClass: "x-form-field x-form-num-field",
40137     /**
40138      * @cfg {Boolean} allowDecimals False to disallow decimal values (defaults to true)
40139      */
40140     allowDecimals : true,
40141     /**
40142      * @cfg {String} decimalSeparator Character(s) to allow as the decimal separator (defaults to '.')
40143      */
40144     decimalSeparator : ".",
40145     /**
40146      * @cfg {Number} decimalPrecision The maximum precision to display after the decimal separator (defaults to 2)
40147      */
40148     decimalPrecision : 2,
40149     /**
40150      * @cfg {Boolean} allowNegative False to prevent entering a negative sign (defaults to true)
40151      */
40152     allowNegative : true,
40153     /**
40154      * @cfg {Number} minValue The minimum allowed value (defaults to Number.NEGATIVE_INFINITY)
40155      */
40156     minValue : Number.NEGATIVE_INFINITY,
40157     /**
40158      * @cfg {Number} maxValue The maximum allowed value (defaults to Number.MAX_VALUE)
40159      */
40160     maxValue : Number.MAX_VALUE,
40161     /**
40162      * @cfg {String} minText Error text to display if the minimum value validation fails (defaults to "The minimum value for this field is {minValue}")
40163      */
40164     minText : "The minimum value for this field is {0}",
40165     /**
40166      * @cfg {String} maxText Error text to display if the maximum value validation fails (defaults to "The maximum value for this field is {maxValue}")
40167      */
40168     maxText : "The maximum value for this field is {0}",
40169     /**
40170      * @cfg {String} nanText Error text to display if the value is not a valid number.  For example, this can happen
40171      * if a valid character like '.' or '-' is left in the field with no number (defaults to "{value} is not a valid number")
40172      */
40173     nanText : "{0} is not a valid number",
40174
40175     // private
40176     initEvents : function(){
40177         Roo.form.NumberField.superclass.initEvents.call(this);
40178         var allowed = "0123456789";
40179         if(this.allowDecimals){
40180             allowed += this.decimalSeparator;
40181         }
40182         if(this.allowNegative){
40183             allowed += "-";
40184         }
40185         this.stripCharsRe = new RegExp('[^'+allowed+']', 'gi');
40186         var keyPress = function(e){
40187             var k = e.getKey();
40188             if(!Roo.isIE && (e.isSpecialKey() || k == e.BACKSPACE || k == e.DELETE)){
40189                 return;
40190             }
40191             var c = e.getCharCode();
40192             if(allowed.indexOf(String.fromCharCode(c)) === -1){
40193                 e.stopEvent();
40194             }
40195         };
40196         this.el.on("keypress", keyPress, this);
40197     },
40198
40199     // private
40200     validateValue : function(value){
40201         if(!Roo.form.NumberField.superclass.validateValue.call(this, value)){
40202             return false;
40203         }
40204         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40205              return true;
40206         }
40207         var num = this.parseValue(value);
40208         if(isNaN(num)){
40209             this.markInvalid(String.format(this.nanText, value));
40210             return false;
40211         }
40212         if(num < this.minValue){
40213             this.markInvalid(String.format(this.minText, this.minValue));
40214             return false;
40215         }
40216         if(num > this.maxValue){
40217             this.markInvalid(String.format(this.maxText, this.maxValue));
40218             return false;
40219         }
40220         return true;
40221     },
40222
40223     getValue : function(){
40224         return this.fixPrecision(this.parseValue(Roo.form.NumberField.superclass.getValue.call(this)));
40225     },
40226
40227     // private
40228     parseValue : function(value){
40229         value = parseFloat(String(value).replace(this.decimalSeparator, "."));
40230         return isNaN(value) ? '' : value;
40231     },
40232
40233     // private
40234     fixPrecision : function(value){
40235         var nan = isNaN(value);
40236         if(!this.allowDecimals || this.decimalPrecision == -1 || nan || !value){
40237             return nan ? '' : value;
40238         }
40239         return parseFloat(value).toFixed(this.decimalPrecision);
40240     },
40241
40242     setValue : function(v){
40243         v = this.fixPrecision(v);
40244         Roo.form.NumberField.superclass.setValue.call(this, String(v).replace(".", this.decimalSeparator));
40245     },
40246
40247     // private
40248     decimalPrecisionFcn : function(v){
40249         return Math.floor(v);
40250     },
40251
40252     beforeBlur : function(){
40253         var v = this.parseValue(this.getRawValue());
40254         if(v){
40255             this.setValue(v);
40256         }
40257     }
40258 });/*
40259  * Based on:
40260  * Ext JS Library 1.1.1
40261  * Copyright(c) 2006-2007, Ext JS, LLC.
40262  *
40263  * Originally Released Under LGPL - original licence link has changed is not relivant.
40264  *
40265  * Fork - LGPL
40266  * <script type="text/javascript">
40267  */
40268  
40269 /**
40270  * @class Roo.form.DateField
40271  * @extends Roo.form.TriggerField
40272  * Provides a date input field with a {@link Roo.DatePicker} dropdown and automatic date validation.
40273 * @constructor
40274 * Create a new DateField
40275 * @param {Object} config
40276  */
40277 Roo.form.DateField = function(config){
40278     Roo.form.DateField.superclass.constructor.call(this, config);
40279     
40280       this.addEvents({
40281          
40282         /**
40283          * @event select
40284          * Fires when a date is selected
40285              * @param {Roo.form.DateField} combo This combo box
40286              * @param {Date} date The date selected
40287              */
40288         'select' : true
40289          
40290     });
40291     
40292     
40293     if(typeof this.minValue == "string") {
40294         this.minValue = this.parseDate(this.minValue);
40295     }
40296     if(typeof this.maxValue == "string") {
40297         this.maxValue = this.parseDate(this.maxValue);
40298     }
40299     this.ddMatch = null;
40300     if(this.disabledDates){
40301         var dd = this.disabledDates;
40302         var re = "(?:";
40303         for(var i = 0; i < dd.length; i++){
40304             re += dd[i];
40305             if(i != dd.length-1) {
40306                 re += "|";
40307             }
40308         }
40309         this.ddMatch = new RegExp(re + ")");
40310     }
40311 };
40312
40313 Roo.extend(Roo.form.DateField, Roo.form.TriggerField,  {
40314     /**
40315      * @cfg {String} format
40316      * The default date format string which can be overriden for localization support.  The format must be
40317      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
40318      */
40319     format : "m/d/y",
40320     /**
40321      * @cfg {String} altFormats
40322      * Multiple date formats separated by "|" to try when parsing a user input value and it doesn't match the defined
40323      * format (defaults to 'm/d/Y|m-d-y|m-d-Y|m/d|m-d|d').
40324      */
40325     altFormats : "m/d/Y|m-d-y|m-d-Y|m/d|m-d|md|mdy|mdY|d",
40326     /**
40327      * @cfg {Array} disabledDays
40328      * An array of days to disable, 0 based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
40329      */
40330     disabledDays : null,
40331     /**
40332      * @cfg {String} disabledDaysText
40333      * The tooltip to display when the date falls on a disabled day (defaults to 'Disabled')
40334      */
40335     disabledDaysText : "Disabled",
40336     /**
40337      * @cfg {Array} disabledDates
40338      * An array of "dates" to disable, as strings. These strings will be used to build a dynamic regular
40339      * expression so they are very powerful. Some examples:
40340      * <ul>
40341      * <li>["03/08/2003", "09/16/2003"] would disable those exact dates</li>
40342      * <li>["03/08", "09/16"] would disable those days for every year</li>
40343      * <li>["^03/08"] would only match the beginning (useful if you are using short years)</li>
40344      * <li>["03/../2006"] would disable every day in March 2006</li>
40345      * <li>["^03"] would disable every day in every March</li>
40346      * </ul>
40347      * In order to support regular expressions, if you are using a date format that has "." in it, you will have to
40348      * escape the dot when restricting dates. For example: ["03\\.08\\.03"].
40349      */
40350     disabledDates : null,
40351     /**
40352      * @cfg {String} disabledDatesText
40353      * The tooltip text to display when the date falls on a disabled date (defaults to 'Disabled')
40354      */
40355     disabledDatesText : "Disabled",
40356     /**
40357      * @cfg {Date/String} minValue
40358      * The minimum allowed date. Can be either a Javascript date object or a string date in a
40359      * valid format (defaults to null).
40360      */
40361     minValue : null,
40362     /**
40363      * @cfg {Date/String} maxValue
40364      * The maximum allowed date. Can be either a Javascript date object or a string date in a
40365      * valid format (defaults to null).
40366      */
40367     maxValue : null,
40368     /**
40369      * @cfg {String} minText
40370      * The error text to display when the date in the cell is before minValue (defaults to
40371      * 'The date in this field must be after {minValue}').
40372      */
40373     minText : "The date in this field must be equal to or after {0}",
40374     /**
40375      * @cfg {String} maxText
40376      * The error text to display when the date in the cell is after maxValue (defaults to
40377      * 'The date in this field must be before {maxValue}').
40378      */
40379     maxText : "The date in this field must be equal to or before {0}",
40380     /**
40381      * @cfg {String} invalidText
40382      * The error text to display when the date in the field is invalid (defaults to
40383      * '{value} is not a valid date - it must be in the format {format}').
40384      */
40385     invalidText : "{0} is not a valid date - it must be in the format {1}",
40386     /**
40387      * @cfg {String} triggerClass
40388      * An additional CSS class used to style the trigger button.  The trigger will always get the
40389      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-date-trigger'
40390      * which displays a calendar icon).
40391      */
40392     triggerClass : 'x-form-date-trigger',
40393     
40394
40395     /**
40396      * @cfg {Boolean} useIso
40397      * if enabled, then the date field will use a hidden field to store the 
40398      * real value as iso formated date. default (false)
40399      */ 
40400     useIso : false,
40401     /**
40402      * @cfg {String/Object} autoCreate
40403      * A DomHelper element spec, or true for a default element spec (defaults to
40404      * {tag: "input", type: "text", size: "10", autocomplete: "off"})
40405      */ 
40406     // private
40407     defaultAutoCreate : {tag: "input", type: "text", size: "10", autocomplete: "off"},
40408     
40409     // private
40410     hiddenField: false,
40411     
40412     onRender : function(ct, position)
40413     {
40414         Roo.form.DateField.superclass.onRender.call(this, ct, position);
40415         if (this.useIso) {
40416             //this.el.dom.removeAttribute('name'); 
40417             Roo.log("Changing name?");
40418             this.el.dom.setAttribute('name', this.name + '____hidden___' ); 
40419             this.hiddenField = this.el.insertSibling({ tag:'input', type:'hidden', name: this.name },
40420                     'before', true);
40421             this.hiddenField.value = this.value ? this.formatDate(this.value, 'Y-m-d') : '';
40422             // prevent input submission
40423             this.hiddenName = this.name;
40424         }
40425             
40426             
40427     },
40428     
40429     // private
40430     validateValue : function(value)
40431     {
40432         value = this.formatDate(value);
40433         if(!Roo.form.DateField.superclass.validateValue.call(this, value)){
40434             Roo.log('super failed');
40435             return false;
40436         }
40437         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40438              return true;
40439         }
40440         var svalue = value;
40441         value = this.parseDate(value);
40442         if(!value){
40443             Roo.log('parse date failed' + svalue);
40444             this.markInvalid(String.format(this.invalidText, svalue, this.format));
40445             return false;
40446         }
40447         var time = value.getTime();
40448         if(this.minValue && time < this.minValue.getTime()){
40449             this.markInvalid(String.format(this.minText, this.formatDate(this.minValue)));
40450             return false;
40451         }
40452         if(this.maxValue && time > this.maxValue.getTime()){
40453             this.markInvalid(String.format(this.maxText, this.formatDate(this.maxValue)));
40454             return false;
40455         }
40456         if(this.disabledDays){
40457             var day = value.getDay();
40458             for(var i = 0; i < this.disabledDays.length; i++) {
40459                 if(day === this.disabledDays[i]){
40460                     this.markInvalid(this.disabledDaysText);
40461                     return false;
40462                 }
40463             }
40464         }
40465         var fvalue = this.formatDate(value);
40466         if(this.ddMatch && this.ddMatch.test(fvalue)){
40467             this.markInvalid(String.format(this.disabledDatesText, fvalue));
40468             return false;
40469         }
40470         return true;
40471     },
40472
40473     // private
40474     // Provides logic to override the default TriggerField.validateBlur which just returns true
40475     validateBlur : function(){
40476         return !this.menu || !this.menu.isVisible();
40477     },
40478     
40479     getName: function()
40480     {
40481         // returns hidden if it's set..
40482         if (!this.rendered) {return ''};
40483         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
40484         
40485     },
40486
40487     /**
40488      * Returns the current date value of the date field.
40489      * @return {Date} The date value
40490      */
40491     getValue : function(){
40492         
40493         return  this.hiddenField ?
40494                 this.hiddenField.value :
40495                 this.parseDate(Roo.form.DateField.superclass.getValue.call(this)) || "";
40496     },
40497
40498     /**
40499      * Sets the value of the date field.  You can pass a date object or any string that can be parsed into a valid
40500      * date, using DateField.format as the date format, according to the same rules as {@link Date#parseDate}
40501      * (the default format used is "m/d/y").
40502      * <br />Usage:
40503      * <pre><code>
40504 //All of these calls set the same date value (May 4, 2006)
40505
40506 //Pass a date object:
40507 var dt = new Date('5/4/06');
40508 dateField.setValue(dt);
40509
40510 //Pass a date string (default format):
40511 dateField.setValue('5/4/06');
40512
40513 //Pass a date string (custom format):
40514 dateField.format = 'Y-m-d';
40515 dateField.setValue('2006-5-4');
40516 </code></pre>
40517      * @param {String/Date} date The date or valid date string
40518      */
40519     setValue : function(date){
40520         if (this.hiddenField) {
40521             this.hiddenField.value = this.formatDate(this.parseDate(date), 'Y-m-d');
40522         }
40523         Roo.form.DateField.superclass.setValue.call(this, this.formatDate(this.parseDate(date)));
40524         // make sure the value field is always stored as a date..
40525         this.value = this.parseDate(date);
40526         
40527         
40528     },
40529
40530     // private
40531     parseDate : function(value){
40532         if(!value || value instanceof Date){
40533             return value;
40534         }
40535         var v = Date.parseDate(value, this.format);
40536          if (!v && this.useIso) {
40537             v = Date.parseDate(value, 'Y-m-d');
40538         }
40539         if(!v && this.altFormats){
40540             if(!this.altFormatsArray){
40541                 this.altFormatsArray = this.altFormats.split("|");
40542             }
40543             for(var i = 0, len = this.altFormatsArray.length; i < len && !v; i++){
40544                 v = Date.parseDate(value, this.altFormatsArray[i]);
40545             }
40546         }
40547         return v;
40548     },
40549
40550     // private
40551     formatDate : function(date, fmt){
40552         return (!date || !(date instanceof Date)) ?
40553                date : date.dateFormat(fmt || this.format);
40554     },
40555
40556     // private
40557     menuListeners : {
40558         select: function(m, d){
40559             
40560             this.setValue(d);
40561             this.fireEvent('select', this, d);
40562         },
40563         show : function(){ // retain focus styling
40564             this.onFocus();
40565         },
40566         hide : function(){
40567             this.focus.defer(10, this);
40568             var ml = this.menuListeners;
40569             this.menu.un("select", ml.select,  this);
40570             this.menu.un("show", ml.show,  this);
40571             this.menu.un("hide", ml.hide,  this);
40572         }
40573     },
40574
40575     // private
40576     // Implements the default empty TriggerField.onTriggerClick function to display the DatePicker
40577     onTriggerClick : function(){
40578         if(this.disabled){
40579             return;
40580         }
40581         if(this.menu == null){
40582             this.menu = new Roo.menu.DateMenu();
40583         }
40584         Roo.apply(this.menu.picker,  {
40585             showClear: this.allowBlank,
40586             minDate : this.minValue,
40587             maxDate : this.maxValue,
40588             disabledDatesRE : this.ddMatch,
40589             disabledDatesText : this.disabledDatesText,
40590             disabledDays : this.disabledDays,
40591             disabledDaysText : this.disabledDaysText,
40592             format : this.useIso ? 'Y-m-d' : this.format,
40593             minText : String.format(this.minText, this.formatDate(this.minValue)),
40594             maxText : String.format(this.maxText, this.formatDate(this.maxValue))
40595         });
40596         this.menu.on(Roo.apply({}, this.menuListeners, {
40597             scope:this
40598         }));
40599         this.menu.picker.setValue(this.getValue() || new Date());
40600         this.menu.show(this.el, "tl-bl?");
40601     },
40602
40603     beforeBlur : function(){
40604         var v = this.parseDate(this.getRawValue());
40605         if(v){
40606             this.setValue(v);
40607         }
40608     },
40609
40610     /*@
40611      * overide
40612      * 
40613      */
40614     isDirty : function() {
40615         if(this.disabled) {
40616             return false;
40617         }
40618         
40619         if(typeof(this.startValue) === 'undefined'){
40620             return false;
40621         }
40622         
40623         return String(this.getValue()) !== String(this.startValue);
40624         
40625     }
40626 });/*
40627  * Based on:
40628  * Ext JS Library 1.1.1
40629  * Copyright(c) 2006-2007, Ext JS, LLC.
40630  *
40631  * Originally Released Under LGPL - original licence link has changed is not relivant.
40632  *
40633  * Fork - LGPL
40634  * <script type="text/javascript">
40635  */
40636  
40637 /**
40638  * @class Roo.form.MonthField
40639  * @extends Roo.form.TriggerField
40640  * Provides a date input field with a {@link Roo.DatePicker} dropdown and automatic date validation.
40641 * @constructor
40642 * Create a new MonthField
40643 * @param {Object} config
40644  */
40645 Roo.form.MonthField = function(config){
40646     
40647     Roo.form.MonthField.superclass.constructor.call(this, config);
40648     
40649       this.addEvents({
40650          
40651         /**
40652          * @event select
40653          * Fires when a date is selected
40654              * @param {Roo.form.MonthFieeld} combo This combo box
40655              * @param {Date} date The date selected
40656              */
40657         'select' : true
40658          
40659     });
40660     
40661     
40662     if(typeof this.minValue == "string") {
40663         this.minValue = this.parseDate(this.minValue);
40664     }
40665     if(typeof this.maxValue == "string") {
40666         this.maxValue = this.parseDate(this.maxValue);
40667     }
40668     this.ddMatch = null;
40669     if(this.disabledDates){
40670         var dd = this.disabledDates;
40671         var re = "(?:";
40672         for(var i = 0; i < dd.length; i++){
40673             re += dd[i];
40674             if(i != dd.length-1) {
40675                 re += "|";
40676             }
40677         }
40678         this.ddMatch = new RegExp(re + ")");
40679     }
40680 };
40681
40682 Roo.extend(Roo.form.MonthField, Roo.form.TriggerField,  {
40683     /**
40684      * @cfg {String} format
40685      * The default date format string which can be overriden for localization support.  The format must be
40686      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
40687      */
40688     format : "M Y",
40689     /**
40690      * @cfg {String} altFormats
40691      * Multiple date formats separated by "|" to try when parsing a user input value and it doesn't match the defined
40692      * format (defaults to 'm/d/Y|m-d-y|m-d-Y|m/d|m-d|d').
40693      */
40694     altFormats : "M Y|m/Y|m-y|m-Y|my|mY",
40695     /**
40696      * @cfg {Array} disabledDays
40697      * An array of days to disable, 0 based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
40698      */
40699     disabledDays : [0,1,2,3,4,5,6],
40700     /**
40701      * @cfg {String} disabledDaysText
40702      * The tooltip to display when the date falls on a disabled day (defaults to 'Disabled')
40703      */
40704     disabledDaysText : "Disabled",
40705     /**
40706      * @cfg {Array} disabledDates
40707      * An array of "dates" to disable, as strings. These strings will be used to build a dynamic regular
40708      * expression so they are very powerful. Some examples:
40709      * <ul>
40710      * <li>["03/08/2003", "09/16/2003"] would disable those exact dates</li>
40711      * <li>["03/08", "09/16"] would disable those days for every year</li>
40712      * <li>["^03/08"] would only match the beginning (useful if you are using short years)</li>
40713      * <li>["03/../2006"] would disable every day in March 2006</li>
40714      * <li>["^03"] would disable every day in every March</li>
40715      * </ul>
40716      * In order to support regular expressions, if you are using a date format that has "." in it, you will have to
40717      * escape the dot when restricting dates. For example: ["03\\.08\\.03"].
40718      */
40719     disabledDates : null,
40720     /**
40721      * @cfg {String} disabledDatesText
40722      * The tooltip text to display when the date falls on a disabled date (defaults to 'Disabled')
40723      */
40724     disabledDatesText : "Disabled",
40725     /**
40726      * @cfg {Date/String} minValue
40727      * The minimum allowed date. Can be either a Javascript date object or a string date in a
40728      * valid format (defaults to null).
40729      */
40730     minValue : null,
40731     /**
40732      * @cfg {Date/String} maxValue
40733      * The maximum allowed date. Can be either a Javascript date object or a string date in a
40734      * valid format (defaults to null).
40735      */
40736     maxValue : null,
40737     /**
40738      * @cfg {String} minText
40739      * The error text to display when the date in the cell is before minValue (defaults to
40740      * 'The date in this field must be after {minValue}').
40741      */
40742     minText : "The date in this field must be equal to or after {0}",
40743     /**
40744      * @cfg {String} maxTextf
40745      * The error text to display when the date in the cell is after maxValue (defaults to
40746      * 'The date in this field must be before {maxValue}').
40747      */
40748     maxText : "The date in this field must be equal to or before {0}",
40749     /**
40750      * @cfg {String} invalidText
40751      * The error text to display when the date in the field is invalid (defaults to
40752      * '{value} is not a valid date - it must be in the format {format}').
40753      */
40754     invalidText : "{0} is not a valid date - it must be in the format {1}",
40755     /**
40756      * @cfg {String} triggerClass
40757      * An additional CSS class used to style the trigger button.  The trigger will always get the
40758      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-date-trigger'
40759      * which displays a calendar icon).
40760      */
40761     triggerClass : 'x-form-date-trigger',
40762     
40763
40764     /**
40765      * @cfg {Boolean} useIso
40766      * if enabled, then the date field will use a hidden field to store the 
40767      * real value as iso formated date. default (true)
40768      */ 
40769     useIso : true,
40770     /**
40771      * @cfg {String/Object} autoCreate
40772      * A DomHelper element spec, or true for a default element spec (defaults to
40773      * {tag: "input", type: "text", size: "10", autocomplete: "off"})
40774      */ 
40775     // private
40776     defaultAutoCreate : {tag: "input", type: "text", size: "10", autocomplete: "new-password"},
40777     
40778     // private
40779     hiddenField: false,
40780     
40781     hideMonthPicker : false,
40782     
40783     onRender : function(ct, position)
40784     {
40785         Roo.form.MonthField.superclass.onRender.call(this, ct, position);
40786         if (this.useIso) {
40787             this.el.dom.removeAttribute('name'); 
40788             this.hiddenField = this.el.insertSibling({ tag:'input', type:'hidden', name: this.name },
40789                     'before', true);
40790             this.hiddenField.value = this.value ? this.formatDate(this.value, 'Y-m-d') : '';
40791             // prevent input submission
40792             this.hiddenName = this.name;
40793         }
40794             
40795             
40796     },
40797     
40798     // private
40799     validateValue : function(value)
40800     {
40801         value = this.formatDate(value);
40802         if(!Roo.form.MonthField.superclass.validateValue.call(this, value)){
40803             return false;
40804         }
40805         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40806              return true;
40807         }
40808         var svalue = value;
40809         value = this.parseDate(value);
40810         if(!value){
40811             this.markInvalid(String.format(this.invalidText, svalue, this.format));
40812             return false;
40813         }
40814         var time = value.getTime();
40815         if(this.minValue && time < this.minValue.getTime()){
40816             this.markInvalid(String.format(this.minText, this.formatDate(this.minValue)));
40817             return false;
40818         }
40819         if(this.maxValue && time > this.maxValue.getTime()){
40820             this.markInvalid(String.format(this.maxText, this.formatDate(this.maxValue)));
40821             return false;
40822         }
40823         /*if(this.disabledDays){
40824             var day = value.getDay();
40825             for(var i = 0; i < this.disabledDays.length; i++) {
40826                 if(day === this.disabledDays[i]){
40827                     this.markInvalid(this.disabledDaysText);
40828                     return false;
40829                 }
40830             }
40831         }
40832         */
40833         var fvalue = this.formatDate(value);
40834         /*if(this.ddMatch && this.ddMatch.test(fvalue)){
40835             this.markInvalid(String.format(this.disabledDatesText, fvalue));
40836             return false;
40837         }
40838         */
40839         return true;
40840     },
40841
40842     // private
40843     // Provides logic to override the default TriggerField.validateBlur which just returns true
40844     validateBlur : function(){
40845         return !this.menu || !this.menu.isVisible();
40846     },
40847
40848     /**
40849      * Returns the current date value of the date field.
40850      * @return {Date} The date value
40851      */
40852     getValue : function(){
40853         
40854         
40855         
40856         return  this.hiddenField ?
40857                 this.hiddenField.value :
40858                 this.parseDate(Roo.form.MonthField.superclass.getValue.call(this)) || "";
40859     },
40860
40861     /**
40862      * Sets the value of the date field.  You can pass a date object or any string that can be parsed into a valid
40863      * date, using MonthField.format as the date format, according to the same rules as {@link Date#parseDate}
40864      * (the default format used is "m/d/y").
40865      * <br />Usage:
40866      * <pre><code>
40867 //All of these calls set the same date value (May 4, 2006)
40868
40869 //Pass a date object:
40870 var dt = new Date('5/4/06');
40871 monthField.setValue(dt);
40872
40873 //Pass a date string (default format):
40874 monthField.setValue('5/4/06');
40875
40876 //Pass a date string (custom format):
40877 monthField.format = 'Y-m-d';
40878 monthField.setValue('2006-5-4');
40879 </code></pre>
40880      * @param {String/Date} date The date or valid date string
40881      */
40882     setValue : function(date){
40883         Roo.log('month setValue' + date);
40884         // can only be first of month..
40885         
40886         var val = this.parseDate(date);
40887         
40888         if (this.hiddenField) {
40889             this.hiddenField.value = this.formatDate(this.parseDate(date), 'Y-m-d');
40890         }
40891         Roo.form.MonthField.superclass.setValue.call(this, this.formatDate(this.parseDate(date)));
40892         this.value = this.parseDate(date);
40893     },
40894
40895     // private
40896     parseDate : function(value){
40897         if(!value || value instanceof Date){
40898             value = value ? Date.parseDate(value.format('Y-m') + '-01', 'Y-m-d') : null;
40899             return value;
40900         }
40901         var v = Date.parseDate(value, this.format);
40902         if (!v && this.useIso) {
40903             v = Date.parseDate(value, 'Y-m-d');
40904         }
40905         if (v) {
40906             // 
40907             v = Date.parseDate(v.format('Y-m') +'-01', 'Y-m-d');
40908         }
40909         
40910         
40911         if(!v && this.altFormats){
40912             if(!this.altFormatsArray){
40913                 this.altFormatsArray = this.altFormats.split("|");
40914             }
40915             for(var i = 0, len = this.altFormatsArray.length; i < len && !v; i++){
40916                 v = Date.parseDate(value, this.altFormatsArray[i]);
40917             }
40918         }
40919         return v;
40920     },
40921
40922     // private
40923     formatDate : function(date, fmt){
40924         return (!date || !(date instanceof Date)) ?
40925                date : date.dateFormat(fmt || this.format);
40926     },
40927
40928     // private
40929     menuListeners : {
40930         select: function(m, d){
40931             this.setValue(d);
40932             this.fireEvent('select', this, d);
40933         },
40934         show : function(){ // retain focus styling
40935             this.onFocus();
40936         },
40937         hide : function(){
40938             this.focus.defer(10, this);
40939             var ml = this.menuListeners;
40940             this.menu.un("select", ml.select,  this);
40941             this.menu.un("show", ml.show,  this);
40942             this.menu.un("hide", ml.hide,  this);
40943         }
40944     },
40945     // private
40946     // Implements the default empty TriggerField.onTriggerClick function to display the DatePicker
40947     onTriggerClick : function(){
40948         if(this.disabled){
40949             return;
40950         }
40951         if(this.menu == null){
40952             this.menu = new Roo.menu.DateMenu();
40953            
40954         }
40955         
40956         Roo.apply(this.menu.picker,  {
40957             
40958             showClear: this.allowBlank,
40959             minDate : this.minValue,
40960             maxDate : this.maxValue,
40961             disabledDatesRE : this.ddMatch,
40962             disabledDatesText : this.disabledDatesText,
40963             
40964             format : this.useIso ? 'Y-m-d' : this.format,
40965             minText : String.format(this.minText, this.formatDate(this.minValue)),
40966             maxText : String.format(this.maxText, this.formatDate(this.maxValue))
40967             
40968         });
40969          this.menu.on(Roo.apply({}, this.menuListeners, {
40970             scope:this
40971         }));
40972        
40973         
40974         var m = this.menu;
40975         var p = m.picker;
40976         
40977         // hide month picker get's called when we called by 'before hide';
40978         
40979         var ignorehide = true;
40980         p.hideMonthPicker  = function(disableAnim){
40981             if (ignorehide) {
40982                 return;
40983             }
40984              if(this.monthPicker){
40985                 Roo.log("hideMonthPicker called");
40986                 if(disableAnim === true){
40987                     this.monthPicker.hide();
40988                 }else{
40989                     this.monthPicker.slideOut('t', {duration:.2});
40990                     p.setValue(new Date(m.picker.mpSelYear, m.picker.mpSelMonth, 1));
40991                     p.fireEvent("select", this, this.value);
40992                     m.hide();
40993                 }
40994             }
40995         }
40996         
40997         Roo.log('picker set value');
40998         Roo.log(this.getValue());
40999         p.setValue(this.getValue() ? this.parseDate(this.getValue()) : new Date());
41000         m.show(this.el, 'tl-bl?');
41001         ignorehide  = false;
41002         // this will trigger hideMonthPicker..
41003         
41004         
41005         // hidden the day picker
41006         Roo.select('.x-date-picker table', true).first().dom.style.visibility = "hidden";
41007         
41008         
41009         
41010       
41011         
41012         p.showMonthPicker.defer(100, p);
41013     
41014         
41015        
41016     },
41017
41018     beforeBlur : function(){
41019         var v = this.parseDate(this.getRawValue());
41020         if(v){
41021             this.setValue(v);
41022         }
41023     }
41024
41025     /** @cfg {Boolean} grow @hide */
41026     /** @cfg {Number} growMin @hide */
41027     /** @cfg {Number} growMax @hide */
41028     /**
41029      * @hide
41030      * @method autoSize
41031      */
41032 });/*
41033  * Based on:
41034  * Ext JS Library 1.1.1
41035  * Copyright(c) 2006-2007, Ext JS, LLC.
41036  *
41037  * Originally Released Under LGPL - original licence link has changed is not relivant.
41038  *
41039  * Fork - LGPL
41040  * <script type="text/javascript">
41041  */
41042  
41043
41044 /**
41045  * @class Roo.form.ComboBox
41046  * @extends Roo.form.TriggerField
41047  * A combobox control with support for autocomplete, remote-loading, paging and many other features.
41048  * @constructor
41049  * Create a new ComboBox.
41050  * @param {Object} config Configuration options
41051  */
41052 Roo.form.ComboBox = function(config){
41053     Roo.form.ComboBox.superclass.constructor.call(this, config);
41054     this.addEvents({
41055         /**
41056          * @event expand
41057          * Fires when the dropdown list is expanded
41058              * @param {Roo.form.ComboBox} combo This combo box
41059              */
41060         'expand' : true,
41061         /**
41062          * @event collapse
41063          * Fires when the dropdown list is collapsed
41064              * @param {Roo.form.ComboBox} combo This combo box
41065              */
41066         'collapse' : true,
41067         /**
41068          * @event beforeselect
41069          * Fires before a list item is selected. Return false to cancel the selection.
41070              * @param {Roo.form.ComboBox} combo This combo box
41071              * @param {Roo.data.Record} record The data record returned from the underlying store
41072              * @param {Number} index The index of the selected item in the dropdown list
41073              */
41074         'beforeselect' : true,
41075         /**
41076          * @event select
41077          * Fires when a list item is selected
41078              * @param {Roo.form.ComboBox} combo This combo box
41079              * @param {Roo.data.Record} record The data record returned from the underlying store (or false on clear)
41080              * @param {Number} index The index of the selected item in the dropdown list
41081              */
41082         'select' : true,
41083         /**
41084          * @event beforequery
41085          * Fires before all queries are processed. Return false to cancel the query or set cancel to true.
41086          * The event object passed has these properties:
41087              * @param {Roo.form.ComboBox} combo This combo box
41088              * @param {String} query The query
41089              * @param {Boolean} forceAll true to force "all" query
41090              * @param {Boolean} cancel true to cancel the query
41091              * @param {Object} e The query event object
41092              */
41093         'beforequery': true,
41094          /**
41095          * @event add
41096          * Fires when the 'add' icon is pressed (add a listener to enable add button)
41097              * @param {Roo.form.ComboBox} combo This combo box
41098              */
41099         'add' : true,
41100         /**
41101          * @event edit
41102          * Fires when the 'edit' icon is pressed (add a listener to enable add button)
41103              * @param {Roo.form.ComboBox} combo This combo box
41104              * @param {Roo.data.Record|false} record The data record returned from the underlying store (or false on nothing selected)
41105              */
41106         'edit' : true
41107         
41108         
41109     });
41110     if(this.transform){
41111         this.allowDomMove = false;
41112         var s = Roo.getDom(this.transform);
41113         if(!this.hiddenName){
41114             this.hiddenName = s.name;
41115         }
41116         if(!this.store){
41117             this.mode = 'local';
41118             var d = [], opts = s.options;
41119             for(var i = 0, len = opts.length;i < len; i++){
41120                 var o = opts[i];
41121                 var value = (Roo.isIE ? o.getAttributeNode('value').specified : o.hasAttribute('value')) ? o.value : o.text;
41122                 if(o.selected) {
41123                     this.value = value;
41124                 }
41125                 d.push([value, o.text]);
41126             }
41127             this.store = new Roo.data.SimpleStore({
41128                 'id': 0,
41129                 fields: ['value', 'text'],
41130                 data : d
41131             });
41132             this.valueField = 'value';
41133             this.displayField = 'text';
41134         }
41135         s.name = Roo.id(); // wipe out the name in case somewhere else they have a reference
41136         if(!this.lazyRender){
41137             this.target = true;
41138             this.el = Roo.DomHelper.insertBefore(s, this.autoCreate || this.defaultAutoCreate);
41139             s.parentNode.removeChild(s); // remove it
41140             this.render(this.el.parentNode);
41141         }else{
41142             s.parentNode.removeChild(s); // remove it
41143         }
41144
41145     }
41146     if (this.store) {
41147         this.store = Roo.factory(this.store, Roo.data);
41148     }
41149     
41150     this.selectedIndex = -1;
41151     if(this.mode == 'local'){
41152         if(config.queryDelay === undefined){
41153             this.queryDelay = 10;
41154         }
41155         if(config.minChars === undefined){
41156             this.minChars = 0;
41157         }
41158     }
41159 };
41160
41161 Roo.extend(Roo.form.ComboBox, Roo.form.TriggerField, {
41162     /**
41163      * @cfg {String/HTMLElement/Element} transform The id, DOM node or element of an existing select to convert to a ComboBox
41164      */
41165     /**
41166      * @cfg {Boolean} lazyRender True to prevent the ComboBox from rendering until requested (should always be used when
41167      * rendering into an Roo.Editor, defaults to false)
41168      */
41169     /**
41170      * @cfg {Boolean/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to:
41171      * {tag: "input", type: "text", size: "24", autocomplete: "off"})
41172      */
41173     /**
41174      * @cfg {Roo.data.Store} store The data store to which this combo is bound (defaults to undefined)
41175      */
41176     /**
41177      * @cfg {String} title If supplied, a header element is created containing this text and added into the top of
41178      * the dropdown list (defaults to undefined, with no header element)
41179      */
41180
41181      /**
41182      * @cfg {String/Roo.Template} tpl The template to use to render the output
41183      */
41184      
41185     // private
41186     defaultAutoCreate : {tag: "input", type: "text", size: "24", autocomplete: "off"},
41187     /**
41188      * @cfg {Number} listWidth The width in pixels of the dropdown list (defaults to the width of the ComboBox field)
41189      */
41190     listWidth: undefined,
41191     /**
41192      * @cfg {String} displayField The underlying data field name to bind to this CombBox (defaults to undefined if
41193      * mode = 'remote' or 'text' if mode = 'local')
41194      */
41195     displayField: undefined,
41196     /**
41197      * @cfg {String} valueField The underlying data value name to bind to this CombBox (defaults to undefined if
41198      * mode = 'remote' or 'value' if mode = 'local'). 
41199      * Note: use of a valueField requires the user make a selection
41200      * in order for a value to be mapped.
41201      */
41202     valueField: undefined,
41203     
41204     
41205     /**
41206      * @cfg {String} hiddenName If specified, a hidden form field with this name is dynamically generated to store the
41207      * field's data value (defaults to the underlying DOM element's name)
41208      */
41209     hiddenName: undefined,
41210     /**
41211      * @cfg {String} listClass CSS class to apply to the dropdown list element (defaults to '')
41212      */
41213     listClass: '',
41214     /**
41215      * @cfg {String} selectedClass CSS class to apply to the selected item in the dropdown list (defaults to 'x-combo-selected')
41216      */
41217     selectedClass: 'x-combo-selected',
41218     /**
41219      * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
41220      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-arrow-trigger'
41221      * which displays a downward arrow icon).
41222      */
41223     triggerClass : 'x-form-arrow-trigger',
41224     /**
41225      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
41226      */
41227     shadow:'sides',
41228     /**
41229      * @cfg {String} listAlign A valid anchor position value. See {@link Roo.Element#alignTo} for details on supported
41230      * anchor positions (defaults to 'tl-bl')
41231      */
41232     listAlign: 'tl-bl?',
41233     /**
41234      * @cfg {Number} maxHeight The maximum height in pixels of the dropdown list before scrollbars are shown (defaults to 300)
41235      */
41236     maxHeight: 300,
41237     /**
41238      * @cfg {String} triggerAction The action to execute when the trigger field is activated.  Use 'all' to run the
41239      * query specified by the allQuery config option (defaults to 'query')
41240      */
41241     triggerAction: 'query',
41242     /**
41243      * @cfg {Number} minChars The minimum number of characters the user must type before autocomplete and typeahead activate
41244      * (defaults to 4, does not apply if editable = false)
41245      */
41246     minChars : 4,
41247     /**
41248      * @cfg {Boolean} typeAhead True to populate and autoselect the remainder of the text being typed after a configurable
41249      * delay (typeAheadDelay) if it matches a known value (defaults to false)
41250      */
41251     typeAhead: false,
41252     /**
41253      * @cfg {Number} queryDelay The length of time in milliseconds to delay between the start of typing and sending the
41254      * query to filter the dropdown list (defaults to 500 if mode = 'remote' or 10 if mode = 'local')
41255      */
41256     queryDelay: 500,
41257     /**
41258      * @cfg {Number} pageSize If greater than 0, a paging toolbar is displayed in the footer of the dropdown list and the
41259      * filter queries will execute with page start and limit parameters.  Only applies when mode = 'remote' (defaults to 0)
41260      */
41261     pageSize: 0,
41262     /**
41263      * @cfg {Boolean} selectOnFocus True to select any existing text in the field immediately on focus.  Only applies
41264      * when editable = true (defaults to false)
41265      */
41266     selectOnFocus:false,
41267     /**
41268      * @cfg {String} queryParam Name of the query as it will be passed on the querystring (defaults to 'query')
41269      */
41270     queryParam: 'query',
41271     /**
41272      * @cfg {String} loadingText The text to display in the dropdown list while data is loading.  Only applies
41273      * when mode = 'remote' (defaults to 'Loading...')
41274      */
41275     loadingText: 'Loading...',
41276     /**
41277      * @cfg {Boolean} resizable True to add a resize handle to the bottom of the dropdown list (defaults to false)
41278      */
41279     resizable: false,
41280     /**
41281      * @cfg {Number} handleHeight The height in pixels of the dropdown list resize handle if resizable = true (defaults to 8)
41282      */
41283     handleHeight : 8,
41284     /**
41285      * @cfg {Boolean} editable False to prevent the user from typing text directly into the field, just like a
41286      * traditional select (defaults to true)
41287      */
41288     editable: true,
41289     /**
41290      * @cfg {String} allQuery The text query to send to the server to return all records for the list with no filtering (defaults to '')
41291      */
41292     allQuery: '',
41293     /**
41294      * @cfg {String} mode Set to 'local' if the ComboBox loads local data (defaults to 'remote' which loads from the server)
41295      */
41296     mode: 'remote',
41297     /**
41298      * @cfg {Number} minListWidth The minimum width of the dropdown list in pixels (defaults to 70, will be ignored if
41299      * listWidth has a higher value)
41300      */
41301     minListWidth : 70,
41302     /**
41303      * @cfg {Boolean} forceSelection True to restrict the selected value to one of the values in the list, false to
41304      * allow the user to set arbitrary text into the field (defaults to false)
41305      */
41306     forceSelection:false,
41307     /**
41308      * @cfg {Number} typeAheadDelay The length of time in milliseconds to wait until the typeahead text is displayed
41309      * if typeAhead = true (defaults to 250)
41310      */
41311     typeAheadDelay : 250,
41312     /**
41313      * @cfg {String} valueNotFoundText When using a name/value combo, if the value passed to setValue is not found in
41314      * the store, valueNotFoundText will be displayed as the field text if defined (defaults to undefined)
41315      */
41316     valueNotFoundText : undefined,
41317     /**
41318      * @cfg {Boolean} blockFocus Prevents all focus calls, so it can work with things like HTML edtor bar
41319      */
41320     blockFocus : false,
41321     
41322     /**
41323      * @cfg {Boolean} disableClear Disable showing of clear button.
41324      */
41325     disableClear : false,
41326     /**
41327      * @cfg {Boolean} alwaysQuery  Disable caching of results, and always send query
41328      */
41329     alwaysQuery : false,
41330     
41331     //private
41332     addicon : false,
41333     editicon: false,
41334     
41335     // element that contains real text value.. (when hidden is used..)
41336      
41337     // private
41338     onRender : function(ct, position){
41339         Roo.form.ComboBox.superclass.onRender.call(this, ct, position);
41340         if(this.hiddenName){
41341             this.hiddenField = this.el.insertSibling({tag:'input', type:'hidden', name: this.hiddenName, id:  (this.hiddenId||this.hiddenName)},
41342                     'before', true);
41343             this.hiddenField.value =
41344                 this.hiddenValue !== undefined ? this.hiddenValue :
41345                 this.value !== undefined ? this.value : '';
41346
41347             // prevent input submission
41348             this.el.dom.removeAttribute('name');
41349              
41350              
41351         }
41352         if(Roo.isGecko){
41353             this.el.dom.setAttribute('autocomplete', 'off');
41354         }
41355
41356         var cls = 'x-combo-list';
41357
41358         this.list = new Roo.Layer({
41359             shadow: this.shadow, cls: [cls, this.listClass].join(' '), constrain:false
41360         });
41361
41362         var lw = this.listWidth || Math.max(this.wrap.getWidth(), this.minListWidth);
41363         this.list.setWidth(lw);
41364         this.list.swallowEvent('mousewheel');
41365         this.assetHeight = 0;
41366
41367         if(this.title){
41368             this.header = this.list.createChild({cls:cls+'-hd', html: this.title});
41369             this.assetHeight += this.header.getHeight();
41370         }
41371
41372         this.innerList = this.list.createChild({cls:cls+'-inner'});
41373         this.innerList.on('mouseover', this.onViewOver, this);
41374         this.innerList.on('mousemove', this.onViewMove, this);
41375         this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
41376         
41377         if(this.allowBlank && !this.pageSize && !this.disableClear){
41378             this.footer = this.list.createChild({cls:cls+'-ft'});
41379             this.pageTb = new Roo.Toolbar(this.footer);
41380            
41381         }
41382         if(this.pageSize){
41383             this.footer = this.list.createChild({cls:cls+'-ft'});
41384             this.pageTb = new Roo.PagingToolbar(this.footer, this.store,
41385                     {pageSize: this.pageSize});
41386             
41387         }
41388         
41389         if (this.pageTb && this.allowBlank && !this.disableClear) {
41390             var _this = this;
41391             this.pageTb.add(new Roo.Toolbar.Fill(), {
41392                 cls: 'x-btn-icon x-btn-clear',
41393                 text: '&#160;',
41394                 handler: function()
41395                 {
41396                     _this.collapse();
41397                     _this.clearValue();
41398                     _this.onSelect(false, -1);
41399                 }
41400             });
41401         }
41402         if (this.footer) {
41403             this.assetHeight += this.footer.getHeight();
41404         }
41405         
41406
41407         if(!this.tpl){
41408             this.tpl = '<div class="'+cls+'-item">{' + this.displayField + '}</div>';
41409         }
41410
41411         this.view = new Roo.View(this.innerList, this.tpl, {
41412             singleSelect:true, store: this.store, selectedClass: this.selectedClass
41413         });
41414
41415         this.view.on('click', this.onViewClick, this);
41416
41417         this.store.on('beforeload', this.onBeforeLoad, this);
41418         this.store.on('load', this.onLoad, this);
41419         this.store.on('loadexception', this.onLoadException, this);
41420
41421         if(this.resizable){
41422             this.resizer = new Roo.Resizable(this.list,  {
41423                pinned:true, handles:'se'
41424             });
41425             this.resizer.on('resize', function(r, w, h){
41426                 this.maxHeight = h-this.handleHeight-this.list.getFrameWidth('tb')-this.assetHeight;
41427                 this.listWidth = w;
41428                 this.innerList.setWidth(w - this.list.getFrameWidth('lr'));
41429                 this.restrictHeight();
41430             }, this);
41431             this[this.pageSize?'footer':'innerList'].setStyle('margin-bottom', this.handleHeight+'px');
41432         }
41433         if(!this.editable){
41434             this.editable = true;
41435             this.setEditable(false);
41436         }  
41437         
41438         
41439         if (typeof(this.events.add.listeners) != 'undefined') {
41440             
41441             this.addicon = this.wrap.createChild(
41442                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-combo-add' });  
41443        
41444             this.addicon.on('click', function(e) {
41445                 this.fireEvent('add', this);
41446             }, this);
41447         }
41448         if (typeof(this.events.edit.listeners) != 'undefined') {
41449             
41450             this.editicon = this.wrap.createChild(
41451                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-combo-edit' });  
41452             if (this.addicon) {
41453                 this.editicon.setStyle('margin-left', '40px');
41454             }
41455             this.editicon.on('click', function(e) {
41456                 
41457                 // we fire even  if inothing is selected..
41458                 this.fireEvent('edit', this, this.lastData );
41459                 
41460             }, this);
41461         }
41462         
41463         
41464         
41465     },
41466
41467     // private
41468     initEvents : function(){
41469         Roo.form.ComboBox.superclass.initEvents.call(this);
41470
41471         this.keyNav = new Roo.KeyNav(this.el, {
41472             "up" : function(e){
41473                 this.inKeyMode = true;
41474                 this.selectPrev();
41475             },
41476
41477             "down" : function(e){
41478                 if(!this.isExpanded()){
41479                     this.onTriggerClick();
41480                 }else{
41481                     this.inKeyMode = true;
41482                     this.selectNext();
41483                 }
41484             },
41485
41486             "enter" : function(e){
41487                 this.onViewClick();
41488                 //return true;
41489             },
41490
41491             "esc" : function(e){
41492                 this.collapse();
41493             },
41494
41495             "tab" : function(e){
41496                 this.onViewClick(false);
41497                 this.fireEvent("specialkey", this, e);
41498                 return true;
41499             },
41500
41501             scope : this,
41502
41503             doRelay : function(foo, bar, hname){
41504                 if(hname == 'down' || this.scope.isExpanded()){
41505                    return Roo.KeyNav.prototype.doRelay.apply(this, arguments);
41506                 }
41507                 return true;
41508             },
41509
41510             forceKeyDown: true
41511         });
41512         this.queryDelay = Math.max(this.queryDelay || 10,
41513                 this.mode == 'local' ? 10 : 250);
41514         this.dqTask = new Roo.util.DelayedTask(this.initQuery, this);
41515         if(this.typeAhead){
41516             this.taTask = new Roo.util.DelayedTask(this.onTypeAhead, this);
41517         }
41518         if(this.editable !== false){
41519             this.el.on("keyup", this.onKeyUp, this);
41520         }
41521         if(this.forceSelection){
41522             this.on('blur', this.doForce, this);
41523         }
41524     },
41525
41526     onDestroy : function(){
41527         if(this.view){
41528             this.view.setStore(null);
41529             this.view.el.removeAllListeners();
41530             this.view.el.remove();
41531             this.view.purgeListeners();
41532         }
41533         if(this.list){
41534             this.list.destroy();
41535         }
41536         if(this.store){
41537             this.store.un('beforeload', this.onBeforeLoad, this);
41538             this.store.un('load', this.onLoad, this);
41539             this.store.un('loadexception', this.onLoadException, this);
41540         }
41541         Roo.form.ComboBox.superclass.onDestroy.call(this);
41542     },
41543
41544     // private
41545     fireKey : function(e){
41546         if(e.isNavKeyPress() && !this.list.isVisible()){
41547             this.fireEvent("specialkey", this, e);
41548         }
41549     },
41550
41551     // private
41552     onResize: function(w, h){
41553         Roo.form.ComboBox.superclass.onResize.apply(this, arguments);
41554         
41555         if(typeof w != 'number'){
41556             // we do not handle it!?!?
41557             return;
41558         }
41559         var tw = this.trigger.getWidth();
41560         tw += this.addicon ? this.addicon.getWidth() : 0;
41561         tw += this.editicon ? this.editicon.getWidth() : 0;
41562         var x = w - tw;
41563         this.el.setWidth( this.adjustWidth('input', x));
41564             
41565         this.trigger.setStyle('left', x+'px');
41566         
41567         if(this.list && this.listWidth === undefined){
41568             var lw = Math.max(x + this.trigger.getWidth(), this.minListWidth);
41569             this.list.setWidth(lw);
41570             this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
41571         }
41572         
41573     
41574         
41575     },
41576
41577     /**
41578      * Allow or prevent the user from directly editing the field text.  If false is passed,
41579      * the user will only be able to select from the items defined in the dropdown list.  This method
41580      * is the runtime equivalent of setting the 'editable' config option at config time.
41581      * @param {Boolean} value True to allow the user to directly edit the field text
41582      */
41583     setEditable : function(value){
41584         if(value == this.editable){
41585             return;
41586         }
41587         this.editable = value;
41588         if(!value){
41589             this.el.dom.setAttribute('readOnly', true);
41590             this.el.on('mousedown', this.onTriggerClick,  this);
41591             this.el.addClass('x-combo-noedit');
41592         }else{
41593             this.el.dom.setAttribute('readOnly', false);
41594             this.el.un('mousedown', this.onTriggerClick,  this);
41595             this.el.removeClass('x-combo-noedit');
41596         }
41597     },
41598
41599     // private
41600     onBeforeLoad : function(){
41601         if(!this.hasFocus){
41602             return;
41603         }
41604         this.innerList.update(this.loadingText ?
41605                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
41606         this.restrictHeight();
41607         this.selectedIndex = -1;
41608     },
41609
41610     // private
41611     onLoad : function(){
41612         if(!this.hasFocus){
41613             return;
41614         }
41615         if(this.store.getCount() > 0){
41616             this.expand();
41617             this.restrictHeight();
41618             if(this.lastQuery == this.allQuery){
41619                 if(this.editable){
41620                     this.el.dom.select();
41621                 }
41622                 if(!this.selectByValue(this.value, true)){
41623                     this.select(0, true);
41624                 }
41625             }else{
41626                 this.selectNext();
41627                 if(this.typeAhead && this.lastKey != Roo.EventObject.BACKSPACE && this.lastKey != Roo.EventObject.DELETE){
41628                     this.taTask.delay(this.typeAheadDelay);
41629                 }
41630             }
41631         }else{
41632             this.onEmptyResults();
41633         }
41634         //this.el.focus();
41635     },
41636     // private
41637     onLoadException : function()
41638     {
41639         this.collapse();
41640         Roo.log(this.store.reader.jsonData);
41641         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
41642             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
41643         }
41644         
41645         
41646     },
41647     // private
41648     onTypeAhead : function(){
41649         if(this.store.getCount() > 0){
41650             var r = this.store.getAt(0);
41651             var newValue = r.data[this.displayField];
41652             var len = newValue.length;
41653             var selStart = this.getRawValue().length;
41654             if(selStart != len){
41655                 this.setRawValue(newValue);
41656                 this.selectText(selStart, newValue.length);
41657             }
41658         }
41659     },
41660
41661     // private
41662     onSelect : function(record, index){
41663         if(this.fireEvent('beforeselect', this, record, index) !== false){
41664             this.setFromData(index > -1 ? record.data : false);
41665             this.collapse();
41666             this.fireEvent('select', this, record, index);
41667         }
41668     },
41669
41670     /**
41671      * Returns the currently selected field value or empty string if no value is set.
41672      * @return {String} value The selected value
41673      */
41674     getValue : function(){
41675         if(this.valueField){
41676             return typeof this.value != 'undefined' ? this.value : '';
41677         }
41678         return Roo.form.ComboBox.superclass.getValue.call(this);
41679     },
41680
41681     /**
41682      * Clears any text/value currently set in the field
41683      */
41684     clearValue : function(){
41685         if(this.hiddenField){
41686             this.hiddenField.value = '';
41687         }
41688         this.value = '';
41689         this.setRawValue('');
41690         this.lastSelectionText = '';
41691         
41692     },
41693
41694     /**
41695      * Sets the specified value into the field.  If the value finds a match, the corresponding record text
41696      * will be displayed in the field.  If the value does not match the data value of an existing item,
41697      * and the valueNotFoundText config option is defined, it will be displayed as the default field text.
41698      * Otherwise the field will be blank (although the value will still be set).
41699      * @param {String} value The value to match
41700      */
41701     setValue : function(v){
41702         var text = v;
41703         if(this.valueField){
41704             var r = this.findRecord(this.valueField, v);
41705             if(r){
41706                 text = r.data[this.displayField];
41707             }else if(this.valueNotFoundText !== undefined){
41708                 text = this.valueNotFoundText;
41709             }
41710         }
41711         this.lastSelectionText = text;
41712         if(this.hiddenField){
41713             this.hiddenField.value = v;
41714         }
41715         Roo.form.ComboBox.superclass.setValue.call(this, text);
41716         this.value = v;
41717     },
41718     /**
41719      * @property {Object} the last set data for the element
41720      */
41721     
41722     lastData : false,
41723     /**
41724      * Sets the value of the field based on a object which is related to the record format for the store.
41725      * @param {Object} value the value to set as. or false on reset?
41726      */
41727     setFromData : function(o){
41728         var dv = ''; // display value
41729         var vv = ''; // value value..
41730         this.lastData = o;
41731         if (this.displayField) {
41732             dv = !o || typeof(o[this.displayField]) == 'undefined' ? '' : o[this.displayField];
41733         } else {
41734             // this is an error condition!!!
41735             Roo.log('no  displayField value set for '+ (this.name ? this.name : this.id));
41736         }
41737         
41738         if(this.valueField){
41739             vv = !o || typeof(o[this.valueField]) == 'undefined' ? dv : o[this.valueField];
41740         }
41741         if(this.hiddenField){
41742             this.hiddenField.value = vv;
41743             
41744             this.lastSelectionText = dv;
41745             Roo.form.ComboBox.superclass.setValue.call(this, dv);
41746             this.value = vv;
41747             return;
41748         }
41749         // no hidden field.. - we store the value in 'value', but still display
41750         // display field!!!!
41751         this.lastSelectionText = dv;
41752         Roo.form.ComboBox.superclass.setValue.call(this, dv);
41753         this.value = vv;
41754         
41755         
41756     },
41757     // private
41758     reset : function(){
41759         // overridden so that last data is reset..
41760         this.setValue(this.resetValue);
41761         this.originalValue = this.getValue();
41762         this.clearInvalid();
41763         this.lastData = false;
41764         if (this.view) {
41765             this.view.clearSelections();
41766         }
41767     },
41768     // private
41769     findRecord : function(prop, value){
41770         var record;
41771         if(this.store.getCount() > 0){
41772             this.store.each(function(r){
41773                 if(r.data[prop] == value){
41774                     record = r;
41775                     return false;
41776                 }
41777                 return true;
41778             });
41779         }
41780         return record;
41781     },
41782     
41783     getName: function()
41784     {
41785         // returns hidden if it's set..
41786         if (!this.rendered) {return ''};
41787         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
41788         
41789     },
41790     // private
41791     onViewMove : function(e, t){
41792         this.inKeyMode = false;
41793     },
41794
41795     // private
41796     onViewOver : function(e, t){
41797         if(this.inKeyMode){ // prevent key nav and mouse over conflicts
41798             return;
41799         }
41800         var item = this.view.findItemFromChild(t);
41801         if(item){
41802             var index = this.view.indexOf(item);
41803             this.select(index, false);
41804         }
41805     },
41806
41807     // private
41808     onViewClick : function(doFocus)
41809     {
41810         var index = this.view.getSelectedIndexes()[0];
41811         var r = this.store.getAt(index);
41812         if(r){
41813             this.onSelect(r, index);
41814         }
41815         if(doFocus !== false && !this.blockFocus){
41816             this.el.focus();
41817         }
41818     },
41819
41820     // private
41821     restrictHeight : function(){
41822         this.innerList.dom.style.height = '';
41823         var inner = this.innerList.dom;
41824         var h = Math.max(inner.clientHeight, inner.offsetHeight, inner.scrollHeight);
41825         this.innerList.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
41826         this.list.beginUpdate();
41827         this.list.setHeight(this.innerList.getHeight()+this.list.getFrameWidth('tb')+(this.resizable?this.handleHeight:0)+this.assetHeight);
41828         this.list.alignTo(this.el, this.listAlign);
41829         this.list.endUpdate();
41830     },
41831
41832     // private
41833     onEmptyResults : function(){
41834         this.collapse();
41835     },
41836
41837     /**
41838      * Returns true if the dropdown list is expanded, else false.
41839      */
41840     isExpanded : function(){
41841         return this.list.isVisible();
41842     },
41843
41844     /**
41845      * Select an item in the dropdown list by its data value. This function does NOT cause the select event to fire.
41846      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
41847      * @param {String} value The data value of the item to select
41848      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
41849      * selected item if it is not currently in view (defaults to true)
41850      * @return {Boolean} True if the value matched an item in the list, else false
41851      */
41852     selectByValue : function(v, scrollIntoView){
41853         if(v !== undefined && v !== null){
41854             var r = this.findRecord(this.valueField || this.displayField, v);
41855             if(r){
41856                 this.select(this.store.indexOf(r), scrollIntoView);
41857                 return true;
41858             }
41859         }
41860         return false;
41861     },
41862
41863     /**
41864      * Select an item in the dropdown list by its numeric index in the list. This function does NOT cause the select event to fire.
41865      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
41866      * @param {Number} index The zero-based index of the list item to select
41867      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
41868      * selected item if it is not currently in view (defaults to true)
41869      */
41870     select : function(index, scrollIntoView){
41871         this.selectedIndex = index;
41872         this.view.select(index);
41873         if(scrollIntoView !== false){
41874             var el = this.view.getNode(index);
41875             if(el){
41876                 this.innerList.scrollChildIntoView(el, false);
41877             }
41878         }
41879     },
41880
41881     // private
41882     selectNext : function(){
41883         var ct = this.store.getCount();
41884         if(ct > 0){
41885             if(this.selectedIndex == -1){
41886                 this.select(0);
41887             }else if(this.selectedIndex < ct-1){
41888                 this.select(this.selectedIndex+1);
41889             }
41890         }
41891     },
41892
41893     // private
41894     selectPrev : function(){
41895         var ct = this.store.getCount();
41896         if(ct > 0){
41897             if(this.selectedIndex == -1){
41898                 this.select(0);
41899             }else if(this.selectedIndex != 0){
41900                 this.select(this.selectedIndex-1);
41901             }
41902         }
41903     },
41904
41905     // private
41906     onKeyUp : function(e){
41907         if(this.editable !== false && !e.isSpecialKey()){
41908             this.lastKey = e.getKey();
41909             this.dqTask.delay(this.queryDelay);
41910         }
41911     },
41912
41913     // private
41914     validateBlur : function(){
41915         return !this.list || !this.list.isVisible();   
41916     },
41917
41918     // private
41919     initQuery : function(){
41920         this.doQuery(this.getRawValue());
41921     },
41922
41923     // private
41924     doForce : function(){
41925         if(this.el.dom.value.length > 0){
41926             this.el.dom.value =
41927                 this.lastSelectionText === undefined ? '' : this.lastSelectionText;
41928              
41929         }
41930     },
41931
41932     /**
41933      * Execute a query to filter the dropdown list.  Fires the beforequery event prior to performing the
41934      * query allowing the query action to be canceled if needed.
41935      * @param {String} query The SQL query to execute
41936      * @param {Boolean} forceAll True to force the query to execute even if there are currently fewer characters
41937      * in the field than the minimum specified by the minChars config option.  It also clears any filter previously
41938      * saved in the current store (defaults to false)
41939      */
41940     doQuery : function(q, forceAll){
41941         if(q === undefined || q === null){
41942             q = '';
41943         }
41944         var qe = {
41945             query: q,
41946             forceAll: forceAll,
41947             combo: this,
41948             cancel:false
41949         };
41950         if(this.fireEvent('beforequery', qe)===false || qe.cancel){
41951             return false;
41952         }
41953         q = qe.query;
41954         forceAll = qe.forceAll;
41955         if(forceAll === true || (q.length >= this.minChars)){
41956             if(this.lastQuery != q || this.alwaysQuery){
41957                 this.lastQuery = q;
41958                 if(this.mode == 'local'){
41959                     this.selectedIndex = -1;
41960                     if(forceAll){
41961                         this.store.clearFilter();
41962                     }else{
41963                         this.store.filter(this.displayField, q);
41964                     }
41965                     this.onLoad();
41966                 }else{
41967                     this.store.baseParams[this.queryParam] = q;
41968                     this.store.load({
41969                         params: this.getParams(q)
41970                     });
41971                     this.expand();
41972                 }
41973             }else{
41974                 this.selectedIndex = -1;
41975                 this.onLoad();   
41976             }
41977         }
41978     },
41979
41980     // private
41981     getParams : function(q){
41982         var p = {};
41983         //p[this.queryParam] = q;
41984         if(this.pageSize){
41985             p.start = 0;
41986             p.limit = this.pageSize;
41987         }
41988         return p;
41989     },
41990
41991     /**
41992      * Hides the dropdown list if it is currently expanded. Fires the 'collapse' event on completion.
41993      */
41994     collapse : function(){
41995         if(!this.isExpanded()){
41996             return;
41997         }
41998         this.list.hide();
41999         Roo.get(document).un('mousedown', this.collapseIf, this);
42000         Roo.get(document).un('mousewheel', this.collapseIf, this);
42001         if (!this.editable) {
42002             Roo.get(document).un('keydown', this.listKeyPress, this);
42003         }
42004         this.fireEvent('collapse', this);
42005     },
42006
42007     // private
42008     collapseIf : function(e){
42009         if(!e.within(this.wrap) && !e.within(this.list)){
42010             this.collapse();
42011         }
42012     },
42013
42014     /**
42015      * Expands the dropdown list if it is currently hidden. Fires the 'expand' event on completion.
42016      */
42017     expand : function(){
42018         if(this.isExpanded() || !this.hasFocus){
42019             return;
42020         }
42021         this.list.alignTo(this.el, this.listAlign);
42022         this.list.show();
42023         Roo.get(document).on('mousedown', this.collapseIf, this);
42024         Roo.get(document).on('mousewheel', this.collapseIf, this);
42025         if (!this.editable) {
42026             Roo.get(document).on('keydown', this.listKeyPress, this);
42027         }
42028         
42029         this.fireEvent('expand', this);
42030     },
42031
42032     // private
42033     // Implements the default empty TriggerField.onTriggerClick function
42034     onTriggerClick : function(){
42035         if(this.disabled){
42036             return;
42037         }
42038         if(this.isExpanded()){
42039             this.collapse();
42040             if (!this.blockFocus) {
42041                 this.el.focus();
42042             }
42043             
42044         }else {
42045             this.hasFocus = true;
42046             if(this.triggerAction == 'all') {
42047                 this.doQuery(this.allQuery, true);
42048             } else {
42049                 this.doQuery(this.getRawValue());
42050             }
42051             if (!this.blockFocus) {
42052                 this.el.focus();
42053             }
42054         }
42055     },
42056     listKeyPress : function(e)
42057     {
42058         //Roo.log('listkeypress');
42059         // scroll to first matching element based on key pres..
42060         if (e.isSpecialKey()) {
42061             return false;
42062         }
42063         var k = String.fromCharCode(e.getKey()).toUpperCase();
42064         //Roo.log(k);
42065         var match  = false;
42066         var csel = this.view.getSelectedNodes();
42067         var cselitem = false;
42068         if (csel.length) {
42069             var ix = this.view.indexOf(csel[0]);
42070             cselitem  = this.store.getAt(ix);
42071             if (!cselitem.get(this.displayField) || cselitem.get(this.displayField).substring(0,1).toUpperCase() != k) {
42072                 cselitem = false;
42073             }
42074             
42075         }
42076         
42077         this.store.each(function(v) { 
42078             if (cselitem) {
42079                 // start at existing selection.
42080                 if (cselitem.id == v.id) {
42081                     cselitem = false;
42082                 }
42083                 return;
42084             }
42085                 
42086             if (v.get(this.displayField) && v.get(this.displayField).substring(0,1).toUpperCase() == k) {
42087                 match = this.store.indexOf(v);
42088                 return false;
42089             }
42090         }, this);
42091         
42092         if (match === false) {
42093             return true; // no more action?
42094         }
42095         // scroll to?
42096         this.view.select(match);
42097         var sn = Roo.get(this.view.getSelectedNodes()[0]);
42098         sn.scrollIntoView(sn.dom.parentNode, false);
42099     }
42100
42101     /** 
42102     * @cfg {Boolean} grow 
42103     * @hide 
42104     */
42105     /** 
42106     * @cfg {Number} growMin 
42107     * @hide 
42108     */
42109     /** 
42110     * @cfg {Number} growMax 
42111     * @hide 
42112     */
42113     /**
42114      * @hide
42115      * @method autoSize
42116      */
42117 });/*
42118  * Copyright(c) 2010-2012, Roo J Solutions Limited
42119  *
42120  * Licence LGPL
42121  *
42122  */
42123
42124 /**
42125  * @class Roo.form.ComboBoxArray
42126  * @extends Roo.form.TextField
42127  * A facebook style adder... for lists of email / people / countries  etc...
42128  * pick multiple items from a combo box, and shows each one.
42129  *
42130  *  Fred [x]  Brian [x]  [Pick another |v]
42131  *
42132  *
42133  *  For this to work: it needs various extra information
42134  *    - normal combo problay has
42135  *      name, hiddenName
42136  *    + displayField, valueField
42137  *
42138  *    For our purpose...
42139  *
42140  *
42141  *   If we change from 'extends' to wrapping...
42142  *   
42143  *  
42144  *
42145  
42146  
42147  * @constructor
42148  * Create a new ComboBoxArray.
42149  * @param {Object} config Configuration options
42150  */
42151  
42152
42153 Roo.form.ComboBoxArray = function(config)
42154 {
42155     this.addEvents({
42156         /**
42157          * @event beforeremove
42158          * Fires before remove the value from the list
42159              * @param {Roo.form.ComboBoxArray} _self This combo box array
42160              * @param {Roo.form.ComboBoxArray.Item} item removed item
42161              */
42162         'beforeremove' : true,
42163         /**
42164          * @event remove
42165          * Fires when remove the value from the list
42166              * @param {Roo.form.ComboBoxArray} _self This combo box array
42167              * @param {Roo.form.ComboBoxArray.Item} item removed item
42168              */
42169         'remove' : true
42170         
42171         
42172     });
42173     
42174     Roo.form.ComboBoxArray.superclass.constructor.call(this, config);
42175     
42176     this.items = new Roo.util.MixedCollection(false);
42177     
42178     // construct the child combo...
42179     
42180     
42181     
42182     
42183    
42184     
42185 }
42186
42187  
42188 Roo.extend(Roo.form.ComboBoxArray, Roo.form.TextField,
42189
42190     /**
42191      * @cfg {Roo.form.Combo} combo The combo box that is wrapped
42192      */
42193     
42194     lastData : false,
42195     
42196     // behavies liek a hiddne field
42197     inputType:      'hidden',
42198     /**
42199      * @cfg {Number} width The width of the box that displays the selected element
42200      */ 
42201     width:          300,
42202
42203     
42204     
42205     /**
42206      * @cfg {String} name    The name of the visable items on this form (eg. titles not ids)
42207      */
42208     name : false,
42209     /**
42210      * @cfg {String} hiddenName    The hidden name of the field, often contains an comma seperated list of names
42211      */
42212     hiddenName : false,
42213     
42214     
42215     // private the array of items that are displayed..
42216     items  : false,
42217     // private - the hidden field el.
42218     hiddenEl : false,
42219     // private - the filed el..
42220     el : false,
42221     
42222     //validateValue : function() { return true; }, // all values are ok!
42223     //onAddClick: function() { },
42224     
42225     onRender : function(ct, position) 
42226     {
42227         
42228         // create the standard hidden element
42229         //Roo.form.ComboBoxArray.superclass.onRender.call(this, ct, position);
42230         
42231         
42232         // give fake names to child combo;
42233         this.combo.hiddenName = this.hiddenName ? (this.hiddenName+'-subcombo') : this.hiddenName;
42234         this.combo.name = this.name? (this.name+'-subcombo') : this.name;
42235         
42236         this.combo = Roo.factory(this.combo, Roo.form);
42237         this.combo.onRender(ct, position);
42238         if (typeof(this.combo.width) != 'undefined') {
42239             this.combo.onResize(this.combo.width,0);
42240         }
42241         
42242         this.combo.initEvents();
42243         
42244         // assigned so form know we need to do this..
42245         this.store          = this.combo.store;
42246         this.valueField     = this.combo.valueField;
42247         this.displayField   = this.combo.displayField ;
42248         
42249         
42250         this.combo.wrap.addClass('x-cbarray-grp');
42251         
42252         var cbwrap = this.combo.wrap.createChild(
42253             {tag: 'div', cls: 'x-cbarray-cb'},
42254             this.combo.el.dom
42255         );
42256         
42257              
42258         this.hiddenEl = this.combo.wrap.createChild({
42259             tag: 'input',  type:'hidden' , name: this.hiddenName, value : ''
42260         });
42261         this.el = this.combo.wrap.createChild({
42262             tag: 'input',  type:'hidden' , name: this.name, value : ''
42263         });
42264          //   this.el.dom.removeAttribute("name");
42265         
42266         
42267         this.outerWrap = this.combo.wrap;
42268         this.wrap = cbwrap;
42269         
42270         this.outerWrap.setWidth(this.width);
42271         this.outerWrap.dom.removeChild(this.el.dom);
42272         
42273         this.wrap.dom.appendChild(this.el.dom);
42274         this.outerWrap.dom.removeChild(this.combo.trigger.dom);
42275         this.combo.wrap.dom.appendChild(this.combo.trigger.dom);
42276         
42277         this.combo.trigger.setStyle('position','relative');
42278         this.combo.trigger.setStyle('left', '0px');
42279         this.combo.trigger.setStyle('top', '2px');
42280         
42281         this.combo.el.setStyle('vertical-align', 'text-bottom');
42282         
42283         //this.trigger.setStyle('vertical-align', 'top');
42284         
42285         // this should use the code from combo really... on('add' ....)
42286         if (this.adder) {
42287             
42288         
42289             this.adder = this.outerWrap.createChild(
42290                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-adder', style: 'margin-left:2px'});  
42291             var _t = this;
42292             this.adder.on('click', function(e) {
42293                 _t.fireEvent('adderclick', this, e);
42294             }, _t);
42295         }
42296         //var _t = this;
42297         //this.adder.on('click', this.onAddClick, _t);
42298         
42299         
42300         this.combo.on('select', function(cb, rec, ix) {
42301             this.addItem(rec.data);
42302             
42303             cb.setValue('');
42304             cb.el.dom.value = '';
42305             //cb.lastData = rec.data;
42306             // add to list
42307             
42308         }, this);
42309         
42310         
42311     },
42312     
42313     
42314     getName: function()
42315     {
42316         // returns hidden if it's set..
42317         if (!this.rendered) {return ''};
42318         return  this.hiddenName ? this.hiddenName : this.name;
42319         
42320     },
42321     
42322     
42323     onResize: function(w, h){
42324         
42325         return;
42326         // not sure if this is needed..
42327         //this.combo.onResize(w,h);
42328         
42329         if(typeof w != 'number'){
42330             // we do not handle it!?!?
42331             return;
42332         }
42333         var tw = this.combo.trigger.getWidth();
42334         tw += this.addicon ? this.addicon.getWidth() : 0;
42335         tw += this.editicon ? this.editicon.getWidth() : 0;
42336         var x = w - tw;
42337         this.combo.el.setWidth( this.combo.adjustWidth('input', x));
42338             
42339         this.combo.trigger.setStyle('left', '0px');
42340         
42341         if(this.list && this.listWidth === undefined){
42342             var lw = Math.max(x + this.combo.trigger.getWidth(), this.combo.minListWidth);
42343             this.list.setWidth(lw);
42344             this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
42345         }
42346         
42347     
42348         
42349     },
42350     
42351     addItem: function(rec)
42352     {
42353         var valueField = this.combo.valueField;
42354         var displayField = this.combo.displayField;
42355         if (this.items.indexOfKey(rec[valueField]) > -1) {
42356             //console.log("GOT " + rec.data.id);
42357             return;
42358         }
42359         
42360         var x = new Roo.form.ComboBoxArray.Item({
42361             //id : rec[this.idField],
42362             data : rec,
42363             displayField : displayField ,
42364             tipField : displayField ,
42365             cb : this
42366         });
42367         // use the 
42368         this.items.add(rec[valueField],x);
42369         // add it before the element..
42370         this.updateHiddenEl();
42371         x.render(this.outerWrap, this.wrap.dom);
42372         // add the image handler..
42373     },
42374     
42375     updateHiddenEl : function()
42376     {
42377         this.validate();
42378         if (!this.hiddenEl) {
42379             return;
42380         }
42381         var ar = [];
42382         var idField = this.combo.valueField;
42383         
42384         this.items.each(function(f) {
42385             ar.push(f.data[idField]);
42386            
42387         });
42388         this.hiddenEl.dom.value = ar.join(',');
42389         this.validate();
42390     },
42391     
42392     reset : function()
42393     {
42394         this.items.clear();
42395         
42396         Roo.each(this.outerWrap.select('.x-cbarray-item', true).elements, function(el){
42397            el.remove();
42398         });
42399         
42400         this.el.dom.value = '';
42401         if (this.hiddenEl) {
42402             this.hiddenEl.dom.value = '';
42403         }
42404         
42405     },
42406     getValue: function()
42407     {
42408         return this.hiddenEl ? this.hiddenEl.dom.value : '';
42409     },
42410     setValue: function(v) // not a valid action - must use addItems..
42411     {
42412          
42413         this.reset();
42414         
42415         
42416         
42417         if (this.store.isLocal && (typeof(v) == 'string')) {
42418             // then we can use the store to find the values..
42419             // comma seperated at present.. this needs to allow JSON based encoding..
42420             this.hiddenEl.value  = v;
42421             var v_ar = [];
42422             Roo.each(v.split(','), function(k) {
42423                 Roo.log("CHECK " + this.valueField + ',' + k);
42424                 var li = this.store.query(this.valueField, k);
42425                 if (!li.length) {
42426                     return;
42427                 }
42428                 var add = {};
42429                 add[this.valueField] = k;
42430                 add[this.displayField] = li.item(0).data[this.displayField];
42431                 
42432                 this.addItem(add);
42433             }, this) 
42434              
42435         }
42436         if (typeof(v) == 'object' ) {
42437             // then let's assume it's an array of objects..
42438             Roo.each(v, function(l) {
42439                 this.addItem(l);
42440             }, this);
42441              
42442         }
42443         
42444         
42445     },
42446     setFromData: function(v)
42447     {
42448         // this recieves an object, if setValues is called.
42449         this.reset();
42450         this.el.dom.value = v[this.displayField];
42451         this.hiddenEl.dom.value = v[this.valueField];
42452         if (typeof(v[this.valueField]) != 'string' || !v[this.valueField].length) {
42453             return;
42454         }
42455         var kv = v[this.valueField];
42456         var dv = v[this.displayField];
42457         kv = typeof(kv) != 'string' ? '' : kv;
42458         dv = typeof(dv) != 'string' ? '' : dv;
42459         
42460         
42461         var keys = kv.split(',');
42462         var display = dv.split(',');
42463         for (var i = 0 ; i < keys.length; i++) {
42464             
42465             add = {};
42466             add[this.valueField] = keys[i];
42467             add[this.displayField] = display[i];
42468             this.addItem(add);
42469         }
42470       
42471         
42472     },
42473     
42474     /**
42475      * Validates the combox array value
42476      * @return {Boolean} True if the value is valid, else false
42477      */
42478     validate : function(){
42479         if(this.disabled || this.validateValue(this.processValue(this.getValue()))){
42480             this.clearInvalid();
42481             return true;
42482         }
42483         return false;
42484     },
42485     
42486     validateValue : function(value){
42487         return Roo.form.ComboBoxArray.superclass.validateValue.call(this, this.getValue());
42488         
42489     },
42490     
42491     /*@
42492      * overide
42493      * 
42494      */
42495     isDirty : function() {
42496         if(this.disabled) {
42497             return false;
42498         }
42499         
42500         try {
42501             var d = Roo.decode(String(this.originalValue));
42502         } catch (e) {
42503             return String(this.getValue()) !== String(this.originalValue);
42504         }
42505         
42506         var originalValue = [];
42507         
42508         for (var i = 0; i < d.length; i++){
42509             originalValue.push(d[i][this.valueField]);
42510         }
42511         
42512         return String(this.getValue()) !== String(originalValue.join(','));
42513         
42514     }
42515     
42516 });
42517
42518
42519
42520 /**
42521  * @class Roo.form.ComboBoxArray.Item
42522  * @extends Roo.BoxComponent
42523  * A selected item in the list
42524  *  Fred [x]  Brian [x]  [Pick another |v]
42525  * 
42526  * @constructor
42527  * Create a new item.
42528  * @param {Object} config Configuration options
42529  */
42530  
42531 Roo.form.ComboBoxArray.Item = function(config) {
42532     config.id = Roo.id();
42533     Roo.form.ComboBoxArray.Item.superclass.constructor.call(this, config);
42534 }
42535
42536 Roo.extend(Roo.form.ComboBoxArray.Item, Roo.BoxComponent, {
42537     data : {},
42538     cb: false,
42539     displayField : false,
42540     tipField : false,
42541     
42542     
42543     defaultAutoCreate : {
42544         tag: 'div',
42545         cls: 'x-cbarray-item',
42546         cn : [ 
42547             { tag: 'div' },
42548             {
42549                 tag: 'img',
42550                 width:16,
42551                 height : 16,
42552                 src : Roo.BLANK_IMAGE_URL ,
42553                 align: 'center'
42554             }
42555         ]
42556         
42557     },
42558     
42559  
42560     onRender : function(ct, position)
42561     {
42562         Roo.form.Field.superclass.onRender.call(this, ct, position);
42563         
42564         if(!this.el){
42565             var cfg = this.getAutoCreate();
42566             this.el = ct.createChild(cfg, position);
42567         }
42568         
42569         this.el.child('img').dom.setAttribute('src', Roo.BLANK_IMAGE_URL);
42570         
42571         this.el.child('div').dom.innerHTML = this.cb.renderer ? 
42572             this.cb.renderer(this.data) :
42573             String.format('{0}',this.data[this.displayField]);
42574         
42575             
42576         this.el.child('div').dom.setAttribute('qtip',
42577                         String.format('{0}',this.data[this.tipField])
42578         );
42579         
42580         this.el.child('img').on('click', this.remove, this);
42581         
42582     },
42583    
42584     remove : function()
42585     {
42586         if(this.cb.disabled){
42587             return;
42588         }
42589         
42590         if(false !== this.cb.fireEvent('beforeremove', this.cb, this)){
42591             this.cb.items.remove(this);
42592             this.el.child('img').un('click', this.remove, this);
42593             this.el.remove();
42594             this.cb.updateHiddenEl();
42595
42596             this.cb.fireEvent('remove', this.cb, this);
42597         }
42598         
42599     }
42600 });/*
42601  * Based on:
42602  * Ext JS Library 1.1.1
42603  * Copyright(c) 2006-2007, Ext JS, LLC.
42604  *
42605  * Originally Released Under LGPL - original licence link has changed is not relivant.
42606  *
42607  * Fork - LGPL
42608  * <script type="text/javascript">
42609  */
42610 /**
42611  * @class Roo.form.Checkbox
42612  * @extends Roo.form.Field
42613  * Single checkbox field.  Can be used as a direct replacement for traditional checkbox fields.
42614  * @constructor
42615  * Creates a new Checkbox
42616  * @param {Object} config Configuration options
42617  */
42618 Roo.form.Checkbox = function(config){
42619     Roo.form.Checkbox.superclass.constructor.call(this, config);
42620     this.addEvents({
42621         /**
42622          * @event check
42623          * Fires when the checkbox is checked or unchecked.
42624              * @param {Roo.form.Checkbox} this This checkbox
42625              * @param {Boolean} checked The new checked value
42626              */
42627         check : true
42628     });
42629 };
42630
42631 Roo.extend(Roo.form.Checkbox, Roo.form.Field,  {
42632     /**
42633      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
42634      */
42635     focusClass : undefined,
42636     /**
42637      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
42638      */
42639     fieldClass: "x-form-field",
42640     /**
42641      * @cfg {Boolean} checked True if the the checkbox should render already checked (defaults to false)
42642      */
42643     checked: false,
42644     /**
42645      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
42646      * {tag: "input", type: "checkbox", autocomplete: "off"})
42647      */
42648     defaultAutoCreate : { tag: "input", type: 'hidden', autocomplete: "off"},
42649     /**
42650      * @cfg {String} boxLabel The text that appears beside the checkbox
42651      */
42652     boxLabel : "",
42653     /**
42654      * @cfg {String} inputValue The value that should go into the generated input element's value attribute
42655      */  
42656     inputValue : '1',
42657     /**
42658      * @cfg {String} valueOff The value that should go into the generated input element's value when unchecked.
42659      */
42660      valueOff: '0', // value when not checked..
42661
42662     actionMode : 'viewEl', 
42663     //
42664     // private
42665     itemCls : 'x-menu-check-item x-form-item',
42666     groupClass : 'x-menu-group-item',
42667     inputType : 'hidden',
42668     
42669     
42670     inSetChecked: false, // check that we are not calling self...
42671     
42672     inputElement: false, // real input element?
42673     basedOn: false, // ????
42674     
42675     isFormField: true, // not sure where this is needed!!!!
42676
42677     onResize : function(){
42678         Roo.form.Checkbox.superclass.onResize.apply(this, arguments);
42679         if(!this.boxLabel){
42680             this.el.alignTo(this.wrap, 'c-c');
42681         }
42682     },
42683
42684     initEvents : function(){
42685         Roo.form.Checkbox.superclass.initEvents.call(this);
42686         this.el.on("click", this.onClick,  this);
42687         this.el.on("change", this.onClick,  this);
42688     },
42689
42690
42691     getResizeEl : function(){
42692         return this.wrap;
42693     },
42694
42695     getPositionEl : function(){
42696         return this.wrap;
42697     },
42698
42699     // private
42700     onRender : function(ct, position){
42701         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
42702         /*
42703         if(this.inputValue !== undefined){
42704             this.el.dom.value = this.inputValue;
42705         }
42706         */
42707         //this.wrap = this.el.wrap({cls: "x-form-check-wrap"});
42708         this.wrap = this.el.wrap({cls: 'x-menu-check-item '});
42709         var viewEl = this.wrap.createChild({ 
42710             tag: 'img', cls: 'x-menu-item-icon', style: 'margin: 0px;' ,src : Roo.BLANK_IMAGE_URL });
42711         this.viewEl = viewEl;   
42712         this.wrap.on('click', this.onClick,  this); 
42713         
42714         this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
42715         this.el.on('propertychange', this.setFromHidden,  this);  //ie
42716         
42717         
42718         
42719         if(this.boxLabel){
42720             this.wrap.createChild({tag: 'label', htmlFor: this.el.id, cls: 'x-form-cb-label', html: this.boxLabel});
42721         //    viewEl.on('click', this.onClick,  this); 
42722         }
42723         //if(this.checked){
42724             this.setChecked(this.checked);
42725         //}else{
42726             //this.checked = this.el.dom;
42727         //}
42728
42729     },
42730
42731     // private
42732     initValue : Roo.emptyFn,
42733
42734     /**
42735      * Returns the checked state of the checkbox.
42736      * @return {Boolean} True if checked, else false
42737      */
42738     getValue : function(){
42739         if(this.el){
42740             return String(this.el.dom.value) == String(this.inputValue ) ? this.inputValue : this.valueOff;
42741         }
42742         return this.valueOff;
42743         
42744     },
42745
42746         // private
42747     onClick : function(){ 
42748         if (this.disabled) {
42749             return;
42750         }
42751         this.setChecked(!this.checked);
42752
42753         //if(this.el.dom.checked != this.checked){
42754         //    this.setValue(this.el.dom.checked);
42755        // }
42756     },
42757
42758     /**
42759      * Sets the checked state of the checkbox.
42760      * On is always based on a string comparison between inputValue and the param.
42761      * @param {Boolean/String} value - the value to set 
42762      * @param {Boolean/String} suppressEvent - whether to suppress the checkchange event.
42763      */
42764     setValue : function(v,suppressEvent){
42765         
42766         
42767         //this.checked = (v === true || v === 'true' || v == '1' || String(v).toLowerCase() == 'on');
42768         //if(this.el && this.el.dom){
42769         //    this.el.dom.checked = this.checked;
42770         //    this.el.dom.defaultChecked = this.checked;
42771         //}
42772         this.setChecked(String(v) === String(this.inputValue), suppressEvent);
42773         //this.fireEvent("check", this, this.checked);
42774     },
42775     // private..
42776     setChecked : function(state,suppressEvent)
42777     {
42778         if (this.inSetChecked) {
42779             this.checked = state;
42780             return;
42781         }
42782         
42783     
42784         if(this.wrap){
42785             this.wrap[state ? 'addClass' : 'removeClass']('x-menu-item-checked');
42786         }
42787         this.checked = state;
42788         if(suppressEvent !== true){
42789             this.fireEvent('check', this, state);
42790         }
42791         this.inSetChecked = true;
42792         this.el.dom.value = state ? this.inputValue : this.valueOff;
42793         this.inSetChecked = false;
42794         
42795     },
42796     // handle setting of hidden value by some other method!!?!?
42797     setFromHidden: function()
42798     {
42799         if(!this.el){
42800             return;
42801         }
42802         //console.log("SET FROM HIDDEN");
42803         //alert('setFrom hidden');
42804         this.setValue(this.el.dom.value);
42805     },
42806     
42807     onDestroy : function()
42808     {
42809         if(this.viewEl){
42810             Roo.get(this.viewEl).remove();
42811         }
42812          
42813         Roo.form.Checkbox.superclass.onDestroy.call(this);
42814     },
42815     
42816     setBoxLabel : function(str)
42817     {
42818         this.wrap.select('.x-form-cb-label', true).first().dom.innerHTML = str;
42819     }
42820
42821 });/*
42822  * Based on:
42823  * Ext JS Library 1.1.1
42824  * Copyright(c) 2006-2007, Ext JS, LLC.
42825  *
42826  * Originally Released Under LGPL - original licence link has changed is not relivant.
42827  *
42828  * Fork - LGPL
42829  * <script type="text/javascript">
42830  */
42831  
42832 /**
42833  * @class Roo.form.Radio
42834  * @extends Roo.form.Checkbox
42835  * Single radio field.  Same as Checkbox, but provided as a convenience for automatically setting the input type.
42836  * Radio grouping is handled automatically by the browser if you give each radio in a group the same name.
42837  * @constructor
42838  * Creates a new Radio
42839  * @param {Object} config Configuration options
42840  */
42841 Roo.form.Radio = function(){
42842     Roo.form.Radio.superclass.constructor.apply(this, arguments);
42843 };
42844 Roo.extend(Roo.form.Radio, Roo.form.Checkbox, {
42845     inputType: 'radio',
42846
42847     /**
42848      * If this radio is part of a group, it will return the selected value
42849      * @return {String}
42850      */
42851     getGroupValue : function(){
42852         return this.el.up('form').child('input[name='+this.el.dom.name+']:checked', true).value;
42853     },
42854     
42855     
42856     onRender : function(ct, position){
42857         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
42858         
42859         if(this.inputValue !== undefined){
42860             this.el.dom.value = this.inputValue;
42861         }
42862          
42863         this.wrap = this.el.wrap({cls: "x-form-check-wrap"});
42864         //this.wrap = this.el.wrap({cls: 'x-menu-check-item '});
42865         //var viewEl = this.wrap.createChild({ 
42866         //    tag: 'img', cls: 'x-menu-item-icon', style: 'margin: 0px;' ,src : Roo.BLANK_IMAGE_URL });
42867         //this.viewEl = viewEl;   
42868         //this.wrap.on('click', this.onClick,  this); 
42869         
42870         //this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
42871         //this.el.on('propertychange', this.setFromHidden,  this);  //ie
42872         
42873         
42874         
42875         if(this.boxLabel){
42876             this.wrap.createChild({tag: 'label', htmlFor: this.el.id, cls: 'x-form-cb-label', html: this.boxLabel});
42877         //    viewEl.on('click', this.onClick,  this); 
42878         }
42879          if(this.checked){
42880             this.el.dom.checked =   'checked' ;
42881         }
42882          
42883     } 
42884     
42885     
42886 });//<script type="text/javascript">
42887
42888 /*
42889  * Based  Ext JS Library 1.1.1
42890  * Copyright(c) 2006-2007, Ext JS, LLC.
42891  * LGPL
42892  *
42893  */
42894  
42895 /**
42896  * @class Roo.HtmlEditorCore
42897  * @extends Roo.Component
42898  * Provides a the editing component for the HTML editors in Roo. (bootstrap and Roo.form)
42899  *
42900  * any element that has display set to 'none' can cause problems in Safari and Firefox.<br/><br/>
42901  */
42902
42903 Roo.HtmlEditorCore = function(config){
42904     
42905     
42906     Roo.HtmlEditorCore.superclass.constructor.call(this, config);
42907     
42908     
42909     this.addEvents({
42910         /**
42911          * @event initialize
42912          * Fires when the editor is fully initialized (including the iframe)
42913          * @param {Roo.HtmlEditorCore} this
42914          */
42915         initialize: true,
42916         /**
42917          * @event activate
42918          * Fires when the editor is first receives the focus. Any insertion must wait
42919          * until after this event.
42920          * @param {Roo.HtmlEditorCore} this
42921          */
42922         activate: true,
42923          /**
42924          * @event beforesync
42925          * Fires before the textarea is updated with content from the editor iframe. Return false
42926          * to cancel the sync.
42927          * @param {Roo.HtmlEditorCore} this
42928          * @param {String} html
42929          */
42930         beforesync: true,
42931          /**
42932          * @event beforepush
42933          * Fires before the iframe editor is updated with content from the textarea. Return false
42934          * to cancel the push.
42935          * @param {Roo.HtmlEditorCore} this
42936          * @param {String} html
42937          */
42938         beforepush: true,
42939          /**
42940          * @event sync
42941          * Fires when the textarea is updated with content from the editor iframe.
42942          * @param {Roo.HtmlEditorCore} this
42943          * @param {String} html
42944          */
42945         sync: true,
42946          /**
42947          * @event push
42948          * Fires when the iframe editor is updated with content from the textarea.
42949          * @param {Roo.HtmlEditorCore} this
42950          * @param {String} html
42951          */
42952         push: true,
42953         
42954         /**
42955          * @event editorevent
42956          * Fires when on any editor (mouse up/down cursor movement etc.) - used for toolbar hooks.
42957          * @param {Roo.HtmlEditorCore} this
42958          */
42959         editorevent: true
42960         
42961     });
42962     
42963     // at this point this.owner is set, so we can start working out the whitelisted / blacklisted elements
42964     
42965     // defaults : white / black...
42966     this.applyBlacklists();
42967     
42968     
42969     
42970 };
42971
42972
42973 Roo.extend(Roo.HtmlEditorCore, Roo.Component,  {
42974
42975
42976      /**
42977      * @cfg {Roo.form.HtmlEditor|Roo.bootstrap.HtmlEditor} the owner field 
42978      */
42979     
42980     owner : false,
42981     
42982      /**
42983      * @cfg {String} resizable  's' or 'se' or 'e' - wrapps the element in a
42984      *                        Roo.resizable.
42985      */
42986     resizable : false,
42987      /**
42988      * @cfg {Number} height (in pixels)
42989      */   
42990     height: 300,
42991    /**
42992      * @cfg {Number} width (in pixels)
42993      */   
42994     width: 500,
42995     
42996     /**
42997      * @cfg {Array} stylesheets url of stylesheets. set to [] to disable stylesheets.
42998      * 
42999      */
43000     stylesheets: false,
43001     
43002     // id of frame..
43003     frameId: false,
43004     
43005     // private properties
43006     validationEvent : false,
43007     deferHeight: true,
43008     initialized : false,
43009     activated : false,
43010     sourceEditMode : false,
43011     onFocus : Roo.emptyFn,
43012     iframePad:3,
43013     hideMode:'offsets',
43014     
43015     clearUp: true,
43016     
43017     // blacklist + whitelisted elements..
43018     black: false,
43019     white: false,
43020      
43021     bodyCls : '',
43022
43023     /**
43024      * Protected method that will not generally be called directly. It
43025      * is called when the editor initializes the iframe with HTML contents. Override this method if you
43026      * want to change the initialization markup of the iframe (e.g. to add stylesheets).
43027      */
43028     getDocMarkup : function(){
43029         // body styles..
43030         var st = '';
43031         
43032         // inherit styels from page...?? 
43033         if (this.stylesheets === false) {
43034             
43035             Roo.get(document.head).select('style').each(function(node) {
43036                 st += node.dom.outerHTML || new XMLSerializer().serializeToString(node.dom);
43037             });
43038             
43039             Roo.get(document.head).select('link').each(function(node) { 
43040                 st += node.dom.outerHTML || new XMLSerializer().serializeToString(node.dom);
43041             });
43042             
43043         } else if (!this.stylesheets.length) {
43044                 // simple..
43045                 st = '<style type="text/css">' +
43046                     'body{border:0;margin:0;padding:3px;height:98%;cursor:text;}' +
43047                    '</style>';
43048         } else { 
43049             st = '<style type="text/css">' +
43050                     this.stylesheets +
43051                 '</style>';
43052         }
43053         
43054         st +=  '<style type="text/css">' +
43055             'IMG { cursor: pointer } ' +
43056         '</style>';
43057
43058         var cls = 'roo-htmleditor-body';
43059         
43060         if(this.bodyCls.length){
43061             cls += ' ' + this.bodyCls;
43062         }
43063         
43064         return '<html><head>' + st  +
43065             //<style type="text/css">' +
43066             //'body{border:0;margin:0;padding:3px;height:98%;cursor:text;}' +
43067             //'</style>' +
43068             ' </head><body class="' +  cls + '"></body></html>';
43069     },
43070
43071     // private
43072     onRender : function(ct, position)
43073     {
43074         var _t = this;
43075         //Roo.HtmlEditorCore.superclass.onRender.call(this, ct, position);
43076         this.el = this.owner.inputEl ? this.owner.inputEl() : this.owner.el;
43077         
43078         
43079         this.el.dom.style.border = '0 none';
43080         this.el.dom.setAttribute('tabIndex', -1);
43081         this.el.addClass('x-hidden hide');
43082         
43083         
43084         
43085         if(Roo.isIE){ // fix IE 1px bogus margin
43086             this.el.applyStyles('margin-top:-1px;margin-bottom:-1px;')
43087         }
43088        
43089         
43090         this.frameId = Roo.id();
43091         
43092          
43093         
43094         var iframe = this.owner.wrap.createChild({
43095             tag: 'iframe',
43096             cls: 'form-control', // bootstrap..
43097             id: this.frameId,
43098             name: this.frameId,
43099             frameBorder : 'no',
43100             'src' : Roo.SSL_SECURE_URL ? Roo.SSL_SECURE_URL  :  "javascript:false"
43101         }, this.el
43102         );
43103         
43104         
43105         this.iframe = iframe.dom;
43106
43107          this.assignDocWin();
43108         
43109         this.doc.designMode = 'on';
43110        
43111         this.doc.open();
43112         this.doc.write(this.getDocMarkup());
43113         this.doc.close();
43114
43115         
43116         var task = { // must defer to wait for browser to be ready
43117             run : function(){
43118                 //console.log("run task?" + this.doc.readyState);
43119                 this.assignDocWin();
43120                 if(this.doc.body || this.doc.readyState == 'complete'){
43121                     try {
43122                         this.doc.designMode="on";
43123                     } catch (e) {
43124                         return;
43125                     }
43126                     Roo.TaskMgr.stop(task);
43127                     this.initEditor.defer(10, this);
43128                 }
43129             },
43130             interval : 10,
43131             duration: 10000,
43132             scope: this
43133         };
43134         Roo.TaskMgr.start(task);
43135
43136     },
43137
43138     // private
43139     onResize : function(w, h)
43140     {
43141          Roo.log('resize: ' +w + ',' + h );
43142         //Roo.HtmlEditorCore.superclass.onResize.apply(this, arguments);
43143         if(!this.iframe){
43144             return;
43145         }
43146         if(typeof w == 'number'){
43147             
43148             this.iframe.style.width = w + 'px';
43149         }
43150         if(typeof h == 'number'){
43151             
43152             this.iframe.style.height = h + 'px';
43153             if(this.doc){
43154                 (this.doc.body || this.doc.documentElement).style.height = (h - (this.iframePad*2)) + 'px';
43155             }
43156         }
43157         
43158     },
43159
43160     /**
43161      * Toggles the editor between standard and source edit mode.
43162      * @param {Boolean} sourceEdit (optional) True for source edit, false for standard
43163      */
43164     toggleSourceEdit : function(sourceEditMode){
43165         
43166         this.sourceEditMode = sourceEditMode === true;
43167         
43168         if(this.sourceEditMode){
43169  
43170             Roo.get(this.iframe).addClass(['x-hidden','hide']);     //FIXME - what's the BS styles for these
43171             
43172         }else{
43173             Roo.get(this.iframe).removeClass(['x-hidden','hide']);
43174             //this.iframe.className = '';
43175             this.deferFocus();
43176         }
43177         //this.setSize(this.owner.wrap.getSize());
43178         //this.fireEvent('editmodechange', this, this.sourceEditMode);
43179     },
43180
43181     
43182   
43183
43184     /**
43185      * Protected method that will not generally be called directly. If you need/want
43186      * custom HTML cleanup, this is the method you should override.
43187      * @param {String} html The HTML to be cleaned
43188      * return {String} The cleaned HTML
43189      */
43190     cleanHtml : function(html){
43191         html = String(html);
43192         if(html.length > 5){
43193             if(Roo.isSafari){ // strip safari nonsense
43194                 html = html.replace(/\sclass="(?:Apple-style-span|khtml-block-placeholder)"/gi, '');
43195             }
43196         }
43197         if(html == '&nbsp;'){
43198             html = '';
43199         }
43200         return html;
43201     },
43202
43203     /**
43204      * HTML Editor -> Textarea
43205      * Protected method that will not generally be called directly. Syncs the contents
43206      * of the editor iframe with the textarea.
43207      */
43208     syncValue : function(){
43209         if(this.initialized){
43210             var bd = (this.doc.body || this.doc.documentElement);
43211             //this.cleanUpPaste(); -- this is done else where and causes havoc..
43212             var html = bd.innerHTML;
43213             if(Roo.isSafari){
43214                 var bs = bd.getAttribute('style'); // Safari puts text-align styles on the body element!
43215                 var m = bs ? bs.match(/text-align:(.*?);/i) : false;
43216                 if(m && m[1]){
43217                     html = '<div style="'+m[0]+'">' + html + '</div>';
43218                 }
43219             }
43220             html = this.cleanHtml(html);
43221             // fix up the special chars.. normaly like back quotes in word...
43222             // however we do not want to do this with chinese..
43223             html = html.replace(/([\x80-\uffff])/g, function (a, b) {
43224                 var cc = b.charCodeAt();
43225                 if (
43226                     (cc >= 0x4E00 && cc < 0xA000 ) ||
43227                     (cc >= 0x3400 && cc < 0x4E00 ) ||
43228                     (cc >= 0xf900 && cc < 0xfb00 )
43229                 ) {
43230                         return b;
43231                 }
43232                 return "&#"+cc+";" 
43233             });
43234             if(this.owner.fireEvent('beforesync', this, html) !== false){
43235                 this.el.dom.value = html;
43236                 this.owner.fireEvent('sync', this, html);
43237             }
43238         }
43239     },
43240
43241     /**
43242      * Protected method that will not generally be called directly. Pushes the value of the textarea
43243      * into the iframe editor.
43244      */
43245     pushValue : function(){
43246         if(this.initialized){
43247             var v = this.el.dom.value.trim();
43248             
43249 //            if(v.length < 1){
43250 //                v = '&#160;';
43251 //            }
43252             
43253             if(this.owner.fireEvent('beforepush', this, v) !== false){
43254                 var d = (this.doc.body || this.doc.documentElement);
43255                 d.innerHTML = v;
43256                 this.cleanUpPaste();
43257                 this.el.dom.value = d.innerHTML;
43258                 this.owner.fireEvent('push', this, v);
43259             }
43260         }
43261     },
43262
43263     // private
43264     deferFocus : function(){
43265         this.focus.defer(10, this);
43266     },
43267
43268     // doc'ed in Field
43269     focus : function(){
43270         if(this.win && !this.sourceEditMode){
43271             this.win.focus();
43272         }else{
43273             this.el.focus();
43274         }
43275     },
43276     
43277     assignDocWin: function()
43278     {
43279         var iframe = this.iframe;
43280         
43281          if(Roo.isIE){
43282             this.doc = iframe.contentWindow.document;
43283             this.win = iframe.contentWindow;
43284         } else {
43285 //            if (!Roo.get(this.frameId)) {
43286 //                return;
43287 //            }
43288 //            this.doc = (iframe.contentDocument || Roo.get(this.frameId).dom.document);
43289 //            this.win = Roo.get(this.frameId).dom.contentWindow;
43290             
43291             if (!Roo.get(this.frameId) && !iframe.contentDocument) {
43292                 return;
43293             }
43294             
43295             this.doc = (iframe.contentDocument || Roo.get(this.frameId).dom.document);
43296             this.win = (iframe.contentWindow || Roo.get(this.frameId).dom.contentWindow);
43297         }
43298     },
43299     
43300     // private
43301     initEditor : function(){
43302         //console.log("INIT EDITOR");
43303         this.assignDocWin();
43304         
43305         
43306         
43307         this.doc.designMode="on";
43308         this.doc.open();
43309         this.doc.write(this.getDocMarkup());
43310         this.doc.close();
43311         
43312         var dbody = (this.doc.body || this.doc.documentElement);
43313         //var ss = this.el.getStyles('font-size', 'font-family', 'background-image', 'background-repeat');
43314         // this copies styles from the containing element into thsi one..
43315         // not sure why we need all of this..
43316         //var ss = this.el.getStyles('font-size', 'background-image', 'background-repeat');
43317         
43318         //var ss = this.el.getStyles( 'background-image', 'background-repeat');
43319         //ss['background-attachment'] = 'fixed'; // w3c
43320         dbody.bgProperties = 'fixed'; // ie
43321         //Roo.DomHelper.applyStyles(dbody, ss);
43322         Roo.EventManager.on(this.doc, {
43323             //'mousedown': this.onEditorEvent,
43324             'mouseup': this.onEditorEvent,
43325             'dblclick': this.onEditorEvent,
43326             'click': this.onEditorEvent,
43327             'keyup': this.onEditorEvent,
43328             buffer:100,
43329             scope: this
43330         });
43331         if(Roo.isGecko){
43332             Roo.EventManager.on(this.doc, 'keypress', this.mozKeyPress, this);
43333         }
43334         if(Roo.isIE || Roo.isSafari || Roo.isOpera){
43335             Roo.EventManager.on(this.doc, 'keydown', this.fixKeys, this);
43336         }
43337         this.initialized = true;
43338
43339         this.owner.fireEvent('initialize', this);
43340         this.pushValue();
43341     },
43342
43343     // private
43344     onDestroy : function(){
43345         
43346         
43347         
43348         if(this.rendered){
43349             
43350             //for (var i =0; i < this.toolbars.length;i++) {
43351             //    // fixme - ask toolbars for heights?
43352             //    this.toolbars[i].onDestroy();
43353            // }
43354             
43355             //this.wrap.dom.innerHTML = '';
43356             //this.wrap.remove();
43357         }
43358     },
43359
43360     // private
43361     onFirstFocus : function(){
43362         
43363         this.assignDocWin();
43364         
43365         
43366         this.activated = true;
43367          
43368     
43369         if(Roo.isGecko){ // prevent silly gecko errors
43370             this.win.focus();
43371             var s = this.win.getSelection();
43372             if(!s.focusNode || s.focusNode.nodeType != 3){
43373                 var r = s.getRangeAt(0);
43374                 r.selectNodeContents((this.doc.body || this.doc.documentElement));
43375                 r.collapse(true);
43376                 this.deferFocus();
43377             }
43378             try{
43379                 this.execCmd('useCSS', true);
43380                 this.execCmd('styleWithCSS', false);
43381             }catch(e){}
43382         }
43383         this.owner.fireEvent('activate', this);
43384     },
43385
43386     // private
43387     adjustFont: function(btn){
43388         var adjust = btn.cmd == 'increasefontsize' ? 1 : -1;
43389         //if(Roo.isSafari){ // safari
43390         //    adjust *= 2;
43391        // }
43392         var v = parseInt(this.doc.queryCommandValue('FontSize')|| 3, 10);
43393         if(Roo.isSafari){ // safari
43394             var sm = { 10 : 1, 13: 2, 16:3, 18:4, 24: 5, 32:6, 48: 7 };
43395             v =  (v < 10) ? 10 : v;
43396             v =  (v > 48) ? 48 : v;
43397             v = typeof(sm[v]) == 'undefined' ? 1 : sm[v];
43398             
43399         }
43400         
43401         
43402         v = Math.max(1, v+adjust);
43403         
43404         this.execCmd('FontSize', v  );
43405     },
43406
43407     onEditorEvent : function(e)
43408     {
43409         this.owner.fireEvent('editorevent', this, e);
43410       //  this.updateToolbar();
43411         this.syncValue(); //we can not sync so often.. sync cleans, so this breaks stuff
43412     },
43413
43414     insertTag : function(tg)
43415     {
43416         // could be a bit smarter... -> wrap the current selected tRoo..
43417         if (tg.toLowerCase() == 'span' || tg.toLowerCase() == 'code') {
43418             
43419             range = this.createRange(this.getSelection());
43420             var wrappingNode = this.doc.createElement(tg.toLowerCase());
43421             wrappingNode.appendChild(range.extractContents());
43422             range.insertNode(wrappingNode);
43423
43424             return;
43425             
43426             
43427             
43428         }
43429         this.execCmd("formatblock",   tg);
43430         
43431     },
43432     
43433     insertText : function(txt)
43434     {
43435         
43436         
43437         var range = this.createRange();
43438         range.deleteContents();
43439                //alert(Sender.getAttribute('label'));
43440                
43441         range.insertNode(this.doc.createTextNode(txt));
43442     } ,
43443     
43444      
43445
43446     /**
43447      * Executes a Midas editor command on the editor document and performs necessary focus and
43448      * toolbar updates. <b>This should only be called after the editor is initialized.</b>
43449      * @param {String} cmd The Midas command
43450      * @param {String/Boolean} value (optional) The value to pass to the command (defaults to null)
43451      */
43452     relayCmd : function(cmd, value){
43453         this.win.focus();
43454         this.execCmd(cmd, value);
43455         this.owner.fireEvent('editorevent', this);
43456         //this.updateToolbar();
43457         this.owner.deferFocus();
43458     },
43459
43460     /**
43461      * Executes a Midas editor command directly on the editor document.
43462      * For visual commands, you should use {@link #relayCmd} instead.
43463      * <b>This should only be called after the editor is initialized.</b>
43464      * @param {String} cmd The Midas command
43465      * @param {String/Boolean} value (optional) The value to pass to the command (defaults to null)
43466      */
43467     execCmd : function(cmd, value){
43468         this.doc.execCommand(cmd, false, value === undefined ? null : value);
43469         this.syncValue();
43470     },
43471  
43472  
43473    
43474     /**
43475      * Inserts the passed text at the current cursor position. Note: the editor must be initialized and activated
43476      * to insert tRoo.
43477      * @param {String} text | dom node.. 
43478      */
43479     insertAtCursor : function(text)
43480     {
43481         
43482         if(!this.activated){
43483             return;
43484         }
43485         /*
43486         if(Roo.isIE){
43487             this.win.focus();
43488             var r = this.doc.selection.createRange();
43489             if(r){
43490                 r.collapse(true);
43491                 r.pasteHTML(text);
43492                 this.syncValue();
43493                 this.deferFocus();
43494             
43495             }
43496             return;
43497         }
43498         */
43499         if(Roo.isGecko || Roo.isOpera || Roo.isSafari){
43500             this.win.focus();
43501             
43502             
43503             // from jquery ui (MIT licenced)
43504             var range, node;
43505             var win = this.win;
43506             
43507             if (win.getSelection && win.getSelection().getRangeAt) {
43508                 range = win.getSelection().getRangeAt(0);
43509                 node = typeof(text) == 'string' ? range.createContextualFragment(text) : text;
43510                 range.insertNode(node);
43511             } else if (win.document.selection && win.document.selection.createRange) {
43512                 // no firefox support
43513                 var txt = typeof(text) == 'string' ? text : text.outerHTML;
43514                 win.document.selection.createRange().pasteHTML(txt);
43515             } else {
43516                 // no firefox support
43517                 var txt = typeof(text) == 'string' ? text : text.outerHTML;
43518                 this.execCmd('InsertHTML', txt);
43519             } 
43520             
43521             this.syncValue();
43522             
43523             this.deferFocus();
43524         }
43525     },
43526  // private
43527     mozKeyPress : function(e){
43528         if(e.ctrlKey){
43529             var c = e.getCharCode(), cmd;
43530           
43531             if(c > 0){
43532                 c = String.fromCharCode(c).toLowerCase();
43533                 switch(c){
43534                     case 'b':
43535                         cmd = 'bold';
43536                         break;
43537                     case 'i':
43538                         cmd = 'italic';
43539                         break;
43540                     
43541                     case 'u':
43542                         cmd = 'underline';
43543                         break;
43544                     
43545                     case 'v':
43546                         this.cleanUpPaste.defer(100, this);
43547                         return;
43548                         
43549                 }
43550                 if(cmd){
43551                     this.win.focus();
43552                     this.execCmd(cmd);
43553                     this.deferFocus();
43554                     e.preventDefault();
43555                 }
43556                 
43557             }
43558         }
43559     },
43560
43561     // private
43562     fixKeys : function(){ // load time branching for fastest keydown performance
43563         if(Roo.isIE){
43564             return function(e){
43565                 var k = e.getKey(), r;
43566                 if(k == e.TAB){
43567                     e.stopEvent();
43568                     r = this.doc.selection.createRange();
43569                     if(r){
43570                         r.collapse(true);
43571                         r.pasteHTML('&#160;&#160;&#160;&#160;');
43572                         this.deferFocus();
43573                     }
43574                     return;
43575                 }
43576                 
43577                 if(k == e.ENTER){
43578                     r = this.doc.selection.createRange();
43579                     if(r){
43580                         var target = r.parentElement();
43581                         if(!target || target.tagName.toLowerCase() != 'li'){
43582                             e.stopEvent();
43583                             r.pasteHTML('<br />');
43584                             r.collapse(false);
43585                             r.select();
43586                         }
43587                     }
43588                 }
43589                 if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
43590                     this.cleanUpPaste.defer(100, this);
43591                     return;
43592                 }
43593                 
43594                 
43595             };
43596         }else if(Roo.isOpera){
43597             return function(e){
43598                 var k = e.getKey();
43599                 if(k == e.TAB){
43600                     e.stopEvent();
43601                     this.win.focus();
43602                     this.execCmd('InsertHTML','&#160;&#160;&#160;&#160;');
43603                     this.deferFocus();
43604                 }
43605                 if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
43606                     this.cleanUpPaste.defer(100, this);
43607                     return;
43608                 }
43609                 
43610             };
43611         }else if(Roo.isSafari){
43612             return function(e){
43613                 var k = e.getKey();
43614                 
43615                 if(k == e.TAB){
43616                     e.stopEvent();
43617                     this.execCmd('InsertText','\t');
43618                     this.deferFocus();
43619                     return;
43620                 }
43621                if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
43622                     this.cleanUpPaste.defer(100, this);
43623                     return;
43624                 }
43625                 
43626              };
43627         }
43628     }(),
43629     
43630     getAllAncestors: function()
43631     {
43632         var p = this.getSelectedNode();
43633         var a = [];
43634         if (!p) {
43635             a.push(p); // push blank onto stack..
43636             p = this.getParentElement();
43637         }
43638         
43639         
43640         while (p && (p.nodeType == 1) && (p.tagName.toLowerCase() != 'body')) {
43641             a.push(p);
43642             p = p.parentNode;
43643         }
43644         a.push(this.doc.body);
43645         return a;
43646     },
43647     lastSel : false,
43648     lastSelNode : false,
43649     
43650     
43651     getSelection : function() 
43652     {
43653         this.assignDocWin();
43654         return Roo.isIE ? this.doc.selection : this.win.getSelection();
43655     },
43656     
43657     getSelectedNode: function() 
43658     {
43659         // this may only work on Gecko!!!
43660         
43661         // should we cache this!!!!
43662         
43663         
43664         
43665          
43666         var range = this.createRange(this.getSelection()).cloneRange();
43667         
43668         if (Roo.isIE) {
43669             var parent = range.parentElement();
43670             while (true) {
43671                 var testRange = range.duplicate();
43672                 testRange.moveToElementText(parent);
43673                 if (testRange.inRange(range)) {
43674                     break;
43675                 }
43676                 if ((parent.nodeType != 1) || (parent.tagName.toLowerCase() == 'body')) {
43677                     break;
43678                 }
43679                 parent = parent.parentElement;
43680             }
43681             return parent;
43682         }
43683         
43684         // is ancestor a text element.
43685         var ac =  range.commonAncestorContainer;
43686         if (ac.nodeType == 3) {
43687             ac = ac.parentNode;
43688         }
43689         
43690         var ar = ac.childNodes;
43691          
43692         var nodes = [];
43693         var other_nodes = [];
43694         var has_other_nodes = false;
43695         for (var i=0;i<ar.length;i++) {
43696             if ((ar[i].nodeType == 3) && (!ar[i].data.length)) { // empty text ? 
43697                 continue;
43698             }
43699             // fullly contained node.
43700             
43701             if (this.rangeIntersectsNode(range,ar[i]) && this.rangeCompareNode(range,ar[i]) == 3) {
43702                 nodes.push(ar[i]);
43703                 continue;
43704             }
43705             
43706             // probably selected..
43707             if ((ar[i].nodeType == 1) && this.rangeIntersectsNode(range,ar[i]) && (this.rangeCompareNode(range,ar[i]) > 0)) {
43708                 other_nodes.push(ar[i]);
43709                 continue;
43710             }
43711             // outer..
43712             if (!this.rangeIntersectsNode(range,ar[i])|| (this.rangeCompareNode(range,ar[i]) == 0))  {
43713                 continue;
43714             }
43715             
43716             
43717             has_other_nodes = true;
43718         }
43719         if (!nodes.length && other_nodes.length) {
43720             nodes= other_nodes;
43721         }
43722         if (has_other_nodes || !nodes.length || (nodes.length > 1)) {
43723             return false;
43724         }
43725         
43726         return nodes[0];
43727     },
43728     createRange: function(sel)
43729     {
43730         // this has strange effects when using with 
43731         // top toolbar - not sure if it's a great idea.
43732         //this.editor.contentWindow.focus();
43733         if (typeof sel != "undefined") {
43734             try {
43735                 return sel.getRangeAt ? sel.getRangeAt(0) : sel.createRange();
43736             } catch(e) {
43737                 return this.doc.createRange();
43738             }
43739         } else {
43740             return this.doc.createRange();
43741         }
43742     },
43743     getParentElement: function()
43744     {
43745         
43746         this.assignDocWin();
43747         var sel = Roo.isIE ? this.doc.selection : this.win.getSelection();
43748         
43749         var range = this.createRange(sel);
43750          
43751         try {
43752             var p = range.commonAncestorContainer;
43753             while (p.nodeType == 3) { // text node
43754                 p = p.parentNode;
43755             }
43756             return p;
43757         } catch (e) {
43758             return null;
43759         }
43760     
43761     },
43762     /***
43763      *
43764      * Range intersection.. the hard stuff...
43765      *  '-1' = before
43766      *  '0' = hits..
43767      *  '1' = after.
43768      *         [ -- selected range --- ]
43769      *   [fail]                        [fail]
43770      *
43771      *    basically..
43772      *      if end is before start or  hits it. fail.
43773      *      if start is after end or hits it fail.
43774      *
43775      *   if either hits (but other is outside. - then it's not 
43776      *   
43777      *    
43778      **/
43779     
43780     
43781     // @see http://www.thismuchiknow.co.uk/?p=64.
43782     rangeIntersectsNode : function(range, node)
43783     {
43784         var nodeRange = node.ownerDocument.createRange();
43785         try {
43786             nodeRange.selectNode(node);
43787         } catch (e) {
43788             nodeRange.selectNodeContents(node);
43789         }
43790     
43791         var rangeStartRange = range.cloneRange();
43792         rangeStartRange.collapse(true);
43793     
43794         var rangeEndRange = range.cloneRange();
43795         rangeEndRange.collapse(false);
43796     
43797         var nodeStartRange = nodeRange.cloneRange();
43798         nodeStartRange.collapse(true);
43799     
43800         var nodeEndRange = nodeRange.cloneRange();
43801         nodeEndRange.collapse(false);
43802     
43803         return rangeStartRange.compareBoundaryPoints(
43804                  Range.START_TO_START, nodeEndRange) == -1 &&
43805                rangeEndRange.compareBoundaryPoints(
43806                  Range.START_TO_START, nodeStartRange) == 1;
43807         
43808          
43809     },
43810     rangeCompareNode : function(range, node)
43811     {
43812         var nodeRange = node.ownerDocument.createRange();
43813         try {
43814             nodeRange.selectNode(node);
43815         } catch (e) {
43816             nodeRange.selectNodeContents(node);
43817         }
43818         
43819         
43820         range.collapse(true);
43821     
43822         nodeRange.collapse(true);
43823      
43824         var ss = range.compareBoundaryPoints( Range.START_TO_START, nodeRange);
43825         var ee = range.compareBoundaryPoints(  Range.END_TO_END, nodeRange);
43826          
43827         //Roo.log(node.tagName + ': ss='+ss +', ee='+ee)
43828         
43829         var nodeIsBefore   =  ss == 1;
43830         var nodeIsAfter    = ee == -1;
43831         
43832         if (nodeIsBefore && nodeIsAfter) {
43833             return 0; // outer
43834         }
43835         if (!nodeIsBefore && nodeIsAfter) {
43836             return 1; //right trailed.
43837         }
43838         
43839         if (nodeIsBefore && !nodeIsAfter) {
43840             return 2;  // left trailed.
43841         }
43842         // fully contined.
43843         return 3;
43844     },
43845
43846     // private? - in a new class?
43847     cleanUpPaste :  function()
43848     {
43849         // cleans up the whole document..
43850         Roo.log('cleanuppaste');
43851         
43852         this.cleanUpChildren(this.doc.body);
43853         var clean = this.cleanWordChars(this.doc.body.innerHTML);
43854         if (clean != this.doc.body.innerHTML) {
43855             this.doc.body.innerHTML = clean;
43856         }
43857         
43858     },
43859     
43860     cleanWordChars : function(input) {// change the chars to hex code
43861         var he = Roo.HtmlEditorCore;
43862         
43863         var output = input;
43864         Roo.each(he.swapCodes, function(sw) { 
43865             var swapper = new RegExp("\\u" + sw[0].toString(16), "g"); // hex codes
43866             
43867             output = output.replace(swapper, sw[1]);
43868         });
43869         
43870         return output;
43871     },
43872     
43873     
43874     cleanUpChildren : function (n)
43875     {
43876         if (!n.childNodes.length) {
43877             return;
43878         }
43879         for (var i = n.childNodes.length-1; i > -1 ; i--) {
43880            this.cleanUpChild(n.childNodes[i]);
43881         }
43882     },
43883     
43884     
43885         
43886     
43887     cleanUpChild : function (node)
43888     {
43889         var ed = this;
43890         //console.log(node);
43891         if (node.nodeName == "#text") {
43892             // clean up silly Windows -- stuff?
43893             return; 
43894         }
43895         if (node.nodeName == "#comment") {
43896             node.parentNode.removeChild(node);
43897             // clean up silly Windows -- stuff?
43898             return; 
43899         }
43900         var lcname = node.tagName.toLowerCase();
43901         // we ignore whitelists... ?? = not really the way to go, but we probably have not got a full
43902         // whitelist of tags..
43903         
43904         if (this.black.indexOf(lcname) > -1 && this.clearUp ) {
43905             // remove node.
43906             node.parentNode.removeChild(node);
43907             return;
43908             
43909         }
43910         
43911         var remove_keep_children= Roo.HtmlEditorCore.remove.indexOf(node.tagName.toLowerCase()) > -1;
43912         
43913         // remove <a name=....> as rendering on yahoo mailer is borked with this.
43914         // this will have to be flaged elsewhere - perhaps ablack=name... on the mailer..
43915         
43916         //if (node.tagName.toLowerCase() == 'a' && !node.hasAttribute('href')) {
43917         //    remove_keep_children = true;
43918         //}
43919         
43920         if (remove_keep_children) {
43921             this.cleanUpChildren(node);
43922             // inserts everything just before this node...
43923             while (node.childNodes.length) {
43924                 var cn = node.childNodes[0];
43925                 node.removeChild(cn);
43926                 node.parentNode.insertBefore(cn, node);
43927             }
43928             node.parentNode.removeChild(node);
43929             return;
43930         }
43931         
43932         if (!node.attributes || !node.attributes.length) {
43933             this.cleanUpChildren(node);
43934             return;
43935         }
43936         
43937         function cleanAttr(n,v)
43938         {
43939             
43940             if (v.match(/^\./) || v.match(/^\//)) {
43941                 return;
43942             }
43943             if (v.match(/^(http|https):\/\//) || v.match(/^mailto:/) || v.match(/^ftp:/)) {
43944                 return;
43945             }
43946             if (v.match(/^#/)) {
43947                 return;
43948             }
43949 //            Roo.log("(REMOVE TAG)"+ node.tagName +'.' + n + '=' + v);
43950             node.removeAttribute(n);
43951             
43952         }
43953         
43954         var cwhite = this.cwhite;
43955         var cblack = this.cblack;
43956             
43957         function cleanStyle(n,v)
43958         {
43959             if (v.match(/expression/)) { //XSS?? should we even bother..
43960                 node.removeAttribute(n);
43961                 return;
43962             }
43963             
43964             var parts = v.split(/;/);
43965             var clean = [];
43966             
43967             Roo.each(parts, function(p) {
43968                 p = p.replace(/^\s+/g,'').replace(/\s+$/g,'');
43969                 if (!p.length) {
43970                     return true;
43971                 }
43972                 var l = p.split(':').shift().replace(/\s+/g,'');
43973                 l = l.replace(/^\s+/g,'').replace(/\s+$/g,'');
43974                 
43975                 if ( cwhite.length && cblack.indexOf(l) > -1) {
43976 //                    Roo.log('(REMOVE CSS)' + node.tagName +'.' + n + ':'+l + '=' + v);
43977                     //node.removeAttribute(n);
43978                     return true;
43979                 }
43980                 //Roo.log()
43981                 // only allow 'c whitelisted system attributes'
43982                 if ( cwhite.length &&  cwhite.indexOf(l) < 0) {
43983 //                    Roo.log('(REMOVE CSS)' + node.tagName +'.' + n + ':'+l + '=' + v);
43984                     //node.removeAttribute(n);
43985                     return true;
43986                 }
43987                 
43988                 
43989                  
43990                 
43991                 clean.push(p);
43992                 return true;
43993             });
43994             if (clean.length) { 
43995                 node.setAttribute(n, clean.join(';'));
43996             } else {
43997                 node.removeAttribute(n);
43998             }
43999             
44000         }
44001         
44002         
44003         for (var i = node.attributes.length-1; i > -1 ; i--) {
44004             var a = node.attributes[i];
44005             //console.log(a);
44006             
44007             if (a.name.toLowerCase().substr(0,2)=='on')  {
44008                 node.removeAttribute(a.name);
44009                 continue;
44010             }
44011             if (Roo.HtmlEditorCore.ablack.indexOf(a.name.toLowerCase()) > -1) {
44012                 node.removeAttribute(a.name);
44013                 continue;
44014             }
44015             if (Roo.HtmlEditorCore.aclean.indexOf(a.name.toLowerCase()) > -1) {
44016                 cleanAttr(a.name,a.value); // fixme..
44017                 continue;
44018             }
44019             if (a.name == 'style') {
44020                 cleanStyle(a.name,a.value);
44021                 continue;
44022             }
44023             /// clean up MS crap..
44024             // tecnically this should be a list of valid class'es..
44025             
44026             
44027             if (a.name == 'class') {
44028                 if (a.value.match(/^Mso/)) {
44029                     node.className = '';
44030                 }
44031                 
44032                 if (a.value.match(/^body$/)) {
44033                     node.className = '';
44034                 }
44035                 continue;
44036             }
44037             
44038             // style cleanup!?
44039             // class cleanup?
44040             
44041         }
44042         
44043         
44044         this.cleanUpChildren(node);
44045         
44046         
44047     },
44048     
44049     /**
44050      * Clean up MS wordisms...
44051      */
44052     cleanWord : function(node)
44053     {
44054         
44055         
44056         if (!node) {
44057             this.cleanWord(this.doc.body);
44058             return;
44059         }
44060         if (node.nodeName == "#text") {
44061             // clean up silly Windows -- stuff?
44062             return; 
44063         }
44064         if (node.nodeName == "#comment") {
44065             node.parentNode.removeChild(node);
44066             // clean up silly Windows -- stuff?
44067             return; 
44068         }
44069         
44070         if (node.tagName.toLowerCase().match(/^(style|script|applet|embed|noframes|noscript)$/)) {
44071             node.parentNode.removeChild(node);
44072             return;
44073         }
44074         
44075         // remove - but keep children..
44076         if (node.tagName.toLowerCase().match(/^(meta|link|\\?xml:|st1:|o:|font)/)) {
44077             while (node.childNodes.length) {
44078                 var cn = node.childNodes[0];
44079                 node.removeChild(cn);
44080                 node.parentNode.insertBefore(cn, node);
44081             }
44082             node.parentNode.removeChild(node);
44083             this.iterateChildren(node, this.cleanWord);
44084             return;
44085         }
44086         // clean styles
44087         if (node.className.length) {
44088             
44089             var cn = node.className.split(/\W+/);
44090             var cna = [];
44091             Roo.each(cn, function(cls) {
44092                 if (cls.match(/Mso[a-zA-Z]+/)) {
44093                     return;
44094                 }
44095                 cna.push(cls);
44096             });
44097             node.className = cna.length ? cna.join(' ') : '';
44098             if (!cna.length) {
44099                 node.removeAttribute("class");
44100             }
44101         }
44102         
44103         if (node.hasAttribute("lang")) {
44104             node.removeAttribute("lang");
44105         }
44106         
44107         if (node.hasAttribute("style")) {
44108             
44109             var styles = node.getAttribute("style").split(";");
44110             var nstyle = [];
44111             Roo.each(styles, function(s) {
44112                 if (!s.match(/:/)) {
44113                     return;
44114                 }
44115                 var kv = s.split(":");
44116                 if (kv[0].match(/^(mso-|line|font|background|margin|padding|color)/)) {
44117                     return;
44118                 }
44119                 // what ever is left... we allow.
44120                 nstyle.push(s);
44121             });
44122             node.setAttribute("style", nstyle.length ? nstyle.join(';') : '');
44123             if (!nstyle.length) {
44124                 node.removeAttribute('style');
44125             }
44126         }
44127         this.iterateChildren(node, this.cleanWord);
44128         
44129         
44130         
44131     },
44132     /**
44133      * iterateChildren of a Node, calling fn each time, using this as the scole..
44134      * @param {DomNode} node node to iterate children of.
44135      * @param {Function} fn method of this class to call on each item.
44136      */
44137     iterateChildren : function(node, fn)
44138     {
44139         if (!node.childNodes.length) {
44140                 return;
44141         }
44142         for (var i = node.childNodes.length-1; i > -1 ; i--) {
44143            fn.call(this, node.childNodes[i])
44144         }
44145     },
44146     
44147     
44148     /**
44149      * cleanTableWidths.
44150      *
44151      * Quite often pasting from word etc.. results in tables with column and widths.
44152      * This does not work well on fluid HTML layouts - like emails. - so this code should hunt an destroy them..
44153      *
44154      */
44155     cleanTableWidths : function(node)
44156     {
44157          
44158          
44159         if (!node) {
44160             this.cleanTableWidths(this.doc.body);
44161             return;
44162         }
44163         
44164         // ignore list...
44165         if (node.nodeName == "#text" || node.nodeName == "#comment") {
44166             return; 
44167         }
44168         Roo.log(node.tagName);
44169         if (!node.tagName.toLowerCase().match(/^(table|td|tr)$/)) {
44170             this.iterateChildren(node, this.cleanTableWidths);
44171             return;
44172         }
44173         if (node.hasAttribute('width')) {
44174             node.removeAttribute('width');
44175         }
44176         
44177          
44178         if (node.hasAttribute("style")) {
44179             // pretty basic...
44180             
44181             var styles = node.getAttribute("style").split(";");
44182             var nstyle = [];
44183             Roo.each(styles, function(s) {
44184                 if (!s.match(/:/)) {
44185                     return;
44186                 }
44187                 var kv = s.split(":");
44188                 if (kv[0].match(/^\s*(width|min-width)\s*$/)) {
44189                     return;
44190                 }
44191                 // what ever is left... we allow.
44192                 nstyle.push(s);
44193             });
44194             node.setAttribute("style", nstyle.length ? nstyle.join(';') : '');
44195             if (!nstyle.length) {
44196                 node.removeAttribute('style');
44197             }
44198         }
44199         
44200         this.iterateChildren(node, this.cleanTableWidths);
44201         
44202         
44203     },
44204     
44205     
44206     
44207     
44208     domToHTML : function(currentElement, depth, nopadtext) {
44209         
44210         depth = depth || 0;
44211         nopadtext = nopadtext || false;
44212     
44213         if (!currentElement) {
44214             return this.domToHTML(this.doc.body);
44215         }
44216         
44217         //Roo.log(currentElement);
44218         var j;
44219         var allText = false;
44220         var nodeName = currentElement.nodeName;
44221         var tagName = Roo.util.Format.htmlEncode(currentElement.tagName);
44222         
44223         if  (nodeName == '#text') {
44224             
44225             return nopadtext ? currentElement.nodeValue : currentElement.nodeValue.trim();
44226         }
44227         
44228         
44229         var ret = '';
44230         if (nodeName != 'BODY') {
44231              
44232             var i = 0;
44233             // Prints the node tagName, such as <A>, <IMG>, etc
44234             if (tagName) {
44235                 var attr = [];
44236                 for(i = 0; i < currentElement.attributes.length;i++) {
44237                     // quoting?
44238                     var aname = currentElement.attributes.item(i).name;
44239                     if (!currentElement.attributes.item(i).value.length) {
44240                         continue;
44241                     }
44242                     attr.push(aname + '="' + Roo.util.Format.htmlEncode(currentElement.attributes.item(i).value) + '"' );
44243                 }
44244                 
44245                 ret = "<"+currentElement.tagName+ ( attr.length ? (' ' + attr.join(' ') ) : '') + ">";
44246             } 
44247             else {
44248                 
44249                 // eack
44250             }
44251         } else {
44252             tagName = false;
44253         }
44254         if (['IMG', 'BR', 'HR', 'INPUT'].indexOf(tagName) > -1) {
44255             return ret;
44256         }
44257         if (['PRE', 'TEXTAREA', 'TD', 'A', 'SPAN'].indexOf(tagName) > -1) { // or code?
44258             nopadtext = true;
44259         }
44260         
44261         
44262         // Traverse the tree
44263         i = 0;
44264         var currentElementChild = currentElement.childNodes.item(i);
44265         var allText = true;
44266         var innerHTML  = '';
44267         lastnode = '';
44268         while (currentElementChild) {
44269             // Formatting code (indent the tree so it looks nice on the screen)
44270             var nopad = nopadtext;
44271             if (lastnode == 'SPAN') {
44272                 nopad  = true;
44273             }
44274             // text
44275             if  (currentElementChild.nodeName == '#text') {
44276                 var toadd = Roo.util.Format.htmlEncode(currentElementChild.nodeValue);
44277                 toadd = nopadtext ? toadd : toadd.trim();
44278                 if (!nopad && toadd.length > 80) {
44279                     innerHTML  += "\n" + (new Array( depth + 1 )).join( "  "  );
44280                 }
44281                 innerHTML  += toadd;
44282                 
44283                 i++;
44284                 currentElementChild = currentElement.childNodes.item(i);
44285                 lastNode = '';
44286                 continue;
44287             }
44288             allText = false;
44289             
44290             innerHTML  += nopad ? '' : "\n" + (new Array( depth + 1 )).join( "  "  );
44291                 
44292             // Recursively traverse the tree structure of the child node
44293             innerHTML   += this.domToHTML(currentElementChild, depth+1, nopadtext);
44294             lastnode = currentElementChild.nodeName;
44295             i++;
44296             currentElementChild=currentElement.childNodes.item(i);
44297         }
44298         
44299         ret += innerHTML;
44300         
44301         if (!allText) {
44302                 // The remaining code is mostly for formatting the tree
44303             ret+= nopadtext ? '' : "\n" + (new Array( depth  )).join( "  "  );
44304         }
44305         
44306         
44307         if (tagName) {
44308             ret+= "</"+tagName+">";
44309         }
44310         return ret;
44311         
44312     },
44313         
44314     applyBlacklists : function()
44315     {
44316         var w = typeof(this.owner.white) != 'undefined' && this.owner.white ? this.owner.white  : [];
44317         var b = typeof(this.owner.black) != 'undefined' && this.owner.black ? this.owner.black :  [];
44318         
44319         this.white = [];
44320         this.black = [];
44321         Roo.each(Roo.HtmlEditorCore.white, function(tag) {
44322             if (b.indexOf(tag) > -1) {
44323                 return;
44324             }
44325             this.white.push(tag);
44326             
44327         }, this);
44328         
44329         Roo.each(w, function(tag) {
44330             if (b.indexOf(tag) > -1) {
44331                 return;
44332             }
44333             if (this.white.indexOf(tag) > -1) {
44334                 return;
44335             }
44336             this.white.push(tag);
44337             
44338         }, this);
44339         
44340         
44341         Roo.each(Roo.HtmlEditorCore.black, function(tag) {
44342             if (w.indexOf(tag) > -1) {
44343                 return;
44344             }
44345             this.black.push(tag);
44346             
44347         }, this);
44348         
44349         Roo.each(b, function(tag) {
44350             if (w.indexOf(tag) > -1) {
44351                 return;
44352             }
44353             if (this.black.indexOf(tag) > -1) {
44354                 return;
44355             }
44356             this.black.push(tag);
44357             
44358         }, this);
44359         
44360         
44361         w = typeof(this.owner.cwhite) != 'undefined' && this.owner.cwhite ? this.owner.cwhite  : [];
44362         b = typeof(this.owner.cblack) != 'undefined' && this.owner.cblack ? this.owner.cblack :  [];
44363         
44364         this.cwhite = [];
44365         this.cblack = [];
44366         Roo.each(Roo.HtmlEditorCore.cwhite, function(tag) {
44367             if (b.indexOf(tag) > -1) {
44368                 return;
44369             }
44370             this.cwhite.push(tag);
44371             
44372         }, this);
44373         
44374         Roo.each(w, function(tag) {
44375             if (b.indexOf(tag) > -1) {
44376                 return;
44377             }
44378             if (this.cwhite.indexOf(tag) > -1) {
44379                 return;
44380             }
44381             this.cwhite.push(tag);
44382             
44383         }, this);
44384         
44385         
44386         Roo.each(Roo.HtmlEditorCore.cblack, function(tag) {
44387             if (w.indexOf(tag) > -1) {
44388                 return;
44389             }
44390             this.cblack.push(tag);
44391             
44392         }, this);
44393         
44394         Roo.each(b, function(tag) {
44395             if (w.indexOf(tag) > -1) {
44396                 return;
44397             }
44398             if (this.cblack.indexOf(tag) > -1) {
44399                 return;
44400             }
44401             this.cblack.push(tag);
44402             
44403         }, this);
44404     },
44405     
44406     setStylesheets : function(stylesheets)
44407     {
44408         if(typeof(stylesheets) == 'string'){
44409             Roo.get(this.iframe.contentDocument.head).createChild({
44410                 tag : 'link',
44411                 rel : 'stylesheet',
44412                 type : 'text/css',
44413                 href : stylesheets
44414             });
44415             
44416             return;
44417         }
44418         var _this = this;
44419      
44420         Roo.each(stylesheets, function(s) {
44421             if(!s.length){
44422                 return;
44423             }
44424             
44425             Roo.get(_this.iframe.contentDocument.head).createChild({
44426                 tag : 'link',
44427                 rel : 'stylesheet',
44428                 type : 'text/css',
44429                 href : s
44430             });
44431         });
44432
44433         
44434     },
44435     
44436     removeStylesheets : function()
44437     {
44438         var _this = this;
44439         
44440         Roo.each(Roo.get(_this.iframe.contentDocument.head).select('link[rel=stylesheet]', true).elements, function(s){
44441             s.remove();
44442         });
44443     },
44444     
44445     setStyle : function(style)
44446     {
44447         Roo.get(this.iframe.contentDocument.head).createChild({
44448             tag : 'style',
44449             type : 'text/css',
44450             html : style
44451         });
44452
44453         return;
44454     }
44455     
44456     // hide stuff that is not compatible
44457     /**
44458      * @event blur
44459      * @hide
44460      */
44461     /**
44462      * @event change
44463      * @hide
44464      */
44465     /**
44466      * @event focus
44467      * @hide
44468      */
44469     /**
44470      * @event specialkey
44471      * @hide
44472      */
44473     /**
44474      * @cfg {String} fieldClass @hide
44475      */
44476     /**
44477      * @cfg {String} focusClass @hide
44478      */
44479     /**
44480      * @cfg {String} autoCreate @hide
44481      */
44482     /**
44483      * @cfg {String} inputType @hide
44484      */
44485     /**
44486      * @cfg {String} invalidClass @hide
44487      */
44488     /**
44489      * @cfg {String} invalidText @hide
44490      */
44491     /**
44492      * @cfg {String} msgFx @hide
44493      */
44494     /**
44495      * @cfg {String} validateOnBlur @hide
44496      */
44497 });
44498
44499 Roo.HtmlEditorCore.white = [
44500         'area', 'br', 'img', 'input', 'hr', 'wbr',
44501         
44502        'address', 'blockquote', 'center', 'dd',      'dir',       'div', 
44503        'dl',      'dt',         'h1',     'h2',      'h3',        'h4', 
44504        'h5',      'h6',         'hr',     'isindex', 'listing',   'marquee', 
44505        'menu',    'multicol',   'ol',     'p',       'plaintext', 'pre', 
44506        'table',   'ul',         'xmp', 
44507        
44508        'caption', 'col', 'colgroup', 'tbody', 'td', 'tfoot', 'th', 
44509       'thead',   'tr', 
44510      
44511       'dir', 'menu', 'ol', 'ul', 'dl',
44512        
44513       'embed',  'object'
44514 ];
44515
44516
44517 Roo.HtmlEditorCore.black = [
44518     //    'embed',  'object', // enable - backend responsiblity to clean thiese
44519         'applet', // 
44520         'base',   'basefont', 'bgsound', 'blink',  'body', 
44521         'frame',  'frameset', 'head',    'html',   'ilayer', 
44522         'iframe', 'layer',  'link',     'meta',    'object',   
44523         'script', 'style' ,'title',  'xml' // clean later..
44524 ];
44525 Roo.HtmlEditorCore.clean = [
44526     'script', 'style', 'title', 'xml'
44527 ];
44528 Roo.HtmlEditorCore.remove = [
44529     'font'
44530 ];
44531 // attributes..
44532
44533 Roo.HtmlEditorCore.ablack = [
44534     'on'
44535 ];
44536     
44537 Roo.HtmlEditorCore.aclean = [ 
44538     'action', 'background', 'codebase', 'dynsrc', 'href', 'lowsrc' 
44539 ];
44540
44541 // protocols..
44542 Roo.HtmlEditorCore.pwhite= [
44543         'http',  'https',  'mailto'
44544 ];
44545
44546 // white listed style attributes.
44547 Roo.HtmlEditorCore.cwhite= [
44548       //  'text-align', /// default is to allow most things..
44549       
44550          
44551 //        'font-size'//??
44552 ];
44553
44554 // black listed style attributes.
44555 Roo.HtmlEditorCore.cblack= [
44556       //  'font-size' -- this can be set by the project 
44557 ];
44558
44559
44560 Roo.HtmlEditorCore.swapCodes   =[ 
44561     [    8211, "--" ], 
44562     [    8212, "--" ], 
44563     [    8216,  "'" ],  
44564     [    8217, "'" ],  
44565     [    8220, '"' ],  
44566     [    8221, '"' ],  
44567     [    8226, "*" ],  
44568     [    8230, "..." ]
44569 ]; 
44570
44571     //<script type="text/javascript">
44572
44573 /*
44574  * Ext JS Library 1.1.1
44575  * Copyright(c) 2006-2007, Ext JS, LLC.
44576  * Licence LGPL
44577  * 
44578  */
44579  
44580  
44581 Roo.form.HtmlEditor = function(config){
44582     
44583     
44584     
44585     Roo.form.HtmlEditor.superclass.constructor.call(this, config);
44586     
44587     if (!this.toolbars) {
44588         this.toolbars = [];
44589     }
44590     this.editorcore = new Roo.HtmlEditorCore(Roo.apply({ owner : this} , config));
44591     
44592     
44593 };
44594
44595 /**
44596  * @class Roo.form.HtmlEditor
44597  * @extends Roo.form.Field
44598  * Provides a lightweight HTML Editor component.
44599  *
44600  * This has been tested on Fireforx / Chrome.. IE may not be so great..
44601  * 
44602  * <br><br><b>Note: The focus/blur and validation marking functionality inherited from Ext.form.Field is NOT
44603  * supported by this editor.</b><br/><br/>
44604  * An Editor is a sensitive component that can't be used in all spots standard fields can be used. Putting an Editor within
44605  * any element that has display set to 'none' can cause problems in Safari and Firefox.<br/><br/>
44606  */
44607 Roo.extend(Roo.form.HtmlEditor, Roo.form.Field, {
44608     /**
44609      * @cfg {Boolean} clearUp
44610      */
44611     clearUp : true,
44612       /**
44613      * @cfg {Array} toolbars Array of toolbars. - defaults to just the Standard one
44614      */
44615     toolbars : false,
44616    
44617      /**
44618      * @cfg {String} resizable  's' or 'se' or 'e' - wrapps the element in a
44619      *                        Roo.resizable.
44620      */
44621     resizable : false,
44622      /**
44623      * @cfg {Number} height (in pixels)
44624      */   
44625     height: 300,
44626    /**
44627      * @cfg {Number} width (in pixels)
44628      */   
44629     width: 500,
44630     
44631     /**
44632      * @cfg {Array} stylesheets url of stylesheets. set to [] to disable stylesheets.
44633      * 
44634      */
44635     stylesheets: false,
44636     
44637     
44638      /**
44639      * @cfg {Array} blacklist of css styles style attributes (blacklist overrides whitelist)
44640      * 
44641      */
44642     cblack: false,
44643     /**
44644      * @cfg {Array} whitelist of css styles style attributes (blacklist overrides whitelist)
44645      * 
44646      */
44647     cwhite: false,
44648     
44649      /**
44650      * @cfg {Array} blacklist of html tags - in addition to standard blacklist.
44651      * 
44652      */
44653     black: false,
44654     /**
44655      * @cfg {Array} whitelist of html tags - in addition to statndard whitelist
44656      * 
44657      */
44658     white: false,
44659     
44660     // id of frame..
44661     frameId: false,
44662     
44663     // private properties
44664     validationEvent : false,
44665     deferHeight: true,
44666     initialized : false,
44667     activated : false,
44668     
44669     onFocus : Roo.emptyFn,
44670     iframePad:3,
44671     hideMode:'offsets',
44672     
44673     actionMode : 'container', // defaults to hiding it...
44674     
44675     defaultAutoCreate : { // modified by initCompnoent..
44676         tag: "textarea",
44677         style:"width:500px;height:300px;",
44678         autocomplete: "new-password"
44679     },
44680
44681     // private
44682     initComponent : function(){
44683         this.addEvents({
44684             /**
44685              * @event initialize
44686              * Fires when the editor is fully initialized (including the iframe)
44687              * @param {HtmlEditor} this
44688              */
44689             initialize: true,
44690             /**
44691              * @event activate
44692              * Fires when the editor is first receives the focus. Any insertion must wait
44693              * until after this event.
44694              * @param {HtmlEditor} this
44695              */
44696             activate: true,
44697              /**
44698              * @event beforesync
44699              * Fires before the textarea is updated with content from the editor iframe. Return false
44700              * to cancel the sync.
44701              * @param {HtmlEditor} this
44702              * @param {String} html
44703              */
44704             beforesync: true,
44705              /**
44706              * @event beforepush
44707              * Fires before the iframe editor is updated with content from the textarea. Return false
44708              * to cancel the push.
44709              * @param {HtmlEditor} this
44710              * @param {String} html
44711              */
44712             beforepush: true,
44713              /**
44714              * @event sync
44715              * Fires when the textarea is updated with content from the editor iframe.
44716              * @param {HtmlEditor} this
44717              * @param {String} html
44718              */
44719             sync: true,
44720              /**
44721              * @event push
44722              * Fires when the iframe editor is updated with content from the textarea.
44723              * @param {HtmlEditor} this
44724              * @param {String} html
44725              */
44726             push: true,
44727              /**
44728              * @event editmodechange
44729              * Fires when the editor switches edit modes
44730              * @param {HtmlEditor} this
44731              * @param {Boolean} sourceEdit True if source edit, false if standard editing.
44732              */
44733             editmodechange: true,
44734             /**
44735              * @event editorevent
44736              * Fires when on any editor (mouse up/down cursor movement etc.) - used for toolbar hooks.
44737              * @param {HtmlEditor} this
44738              */
44739             editorevent: true,
44740             /**
44741              * @event firstfocus
44742              * Fires when on first focus - needed by toolbars..
44743              * @param {HtmlEditor} this
44744              */
44745             firstfocus: true,
44746             /**
44747              * @event autosave
44748              * Auto save the htmlEditor value as a file into Events
44749              * @param {HtmlEditor} this
44750              */
44751             autosave: true,
44752             /**
44753              * @event savedpreview
44754              * preview the saved version of htmlEditor
44755              * @param {HtmlEditor} this
44756              */
44757             savedpreview: true,
44758             
44759             /**
44760             * @event stylesheetsclick
44761             * Fires when press the Sytlesheets button
44762             * @param {Roo.HtmlEditorCore} this
44763             */
44764             stylesheetsclick: true
44765         });
44766         this.defaultAutoCreate =  {
44767             tag: "textarea",
44768             style:'width: ' + this.width + 'px;height: ' + this.height + 'px;',
44769             autocomplete: "new-password"
44770         };
44771     },
44772
44773     /**
44774      * Protected method that will not generally be called directly. It
44775      * is called when the editor creates its toolbar. Override this method if you need to
44776      * add custom toolbar buttons.
44777      * @param {HtmlEditor} editor
44778      */
44779     createToolbar : function(editor){
44780         Roo.log("create toolbars");
44781         if (!editor.toolbars || !editor.toolbars.length) {
44782             editor.toolbars = [ new Roo.form.HtmlEditor.ToolbarStandard() ]; // can be empty?
44783         }
44784         
44785         for (var i =0 ; i < editor.toolbars.length;i++) {
44786             editor.toolbars[i] = Roo.factory(
44787                     typeof(editor.toolbars[i]) == 'string' ?
44788                         { xtype: editor.toolbars[i]} : editor.toolbars[i],
44789                 Roo.form.HtmlEditor);
44790             editor.toolbars[i].init(editor);
44791         }
44792          
44793         
44794     },
44795
44796      
44797     // private
44798     onRender : function(ct, position)
44799     {
44800         var _t = this;
44801         Roo.form.HtmlEditor.superclass.onRender.call(this, ct, position);
44802         
44803         this.wrap = this.el.wrap({
44804             cls:'x-html-editor-wrap', cn:{cls:'x-html-editor-tb'}
44805         });
44806         
44807         this.editorcore.onRender(ct, position);
44808          
44809         if (this.resizable) {
44810             this.resizeEl = new Roo.Resizable(this.wrap, {
44811                 pinned : true,
44812                 wrap: true,
44813                 dynamic : true,
44814                 minHeight : this.height,
44815                 height: this.height,
44816                 handles : this.resizable,
44817                 width: this.width,
44818                 listeners : {
44819                     resize : function(r, w, h) {
44820                         _t.onResize(w,h); // -something
44821                     }
44822                 }
44823             });
44824             
44825         }
44826         this.createToolbar(this);
44827        
44828         
44829         if(!this.width){
44830             this.setSize(this.wrap.getSize());
44831         }
44832         if (this.resizeEl) {
44833             this.resizeEl.resizeTo.defer(100, this.resizeEl,[ this.width,this.height ] );
44834             // should trigger onReize..
44835         }
44836         
44837         this.keyNav = new Roo.KeyNav(this.el, {
44838             
44839             "tab" : function(e){
44840                 e.preventDefault();
44841                 
44842                 var value = this.getValue();
44843                 
44844                 var start = this.el.dom.selectionStart;
44845                 var end = this.el.dom.selectionEnd;
44846                 
44847                 if(!e.shiftKey){
44848                     
44849                     this.setValue(value.substring(0, start) + "\t" + value.substring(end));
44850                     this.el.dom.setSelectionRange(end + 1, end + 1);
44851                     return;
44852                 }
44853                 
44854                 var f = value.substring(0, start).split("\t");
44855                 
44856                 if(f.pop().length != 0){
44857                     return;
44858                 }
44859                 
44860                 this.setValue(f.join("\t") + value.substring(end));
44861                 this.el.dom.setSelectionRange(start - 1, start - 1);
44862                 
44863             },
44864             
44865             "home" : function(e){
44866                 e.preventDefault();
44867                 
44868                 var curr = this.el.dom.selectionStart;
44869                 var lines = this.getValue().split("\n");
44870                 
44871                 if(!lines.length){
44872                     return;
44873                 }
44874                 
44875                 if(e.ctrlKey){
44876                     this.el.dom.setSelectionRange(0, 0);
44877                     return;
44878                 }
44879                 
44880                 var pos = 0;
44881                 
44882                 for (var i = 0; i < lines.length;i++) {
44883                     pos += lines[i].length;
44884                     
44885                     if(i != 0){
44886                         pos += 1;
44887                     }
44888                     
44889                     if(pos < curr){
44890                         continue;
44891                     }
44892                     
44893                     pos -= lines[i].length;
44894                     
44895                     break;
44896                 }
44897                 
44898                 if(!e.shiftKey){
44899                     this.el.dom.setSelectionRange(pos, pos);
44900                     return;
44901                 }
44902                 
44903                 this.el.dom.selectionStart = pos;
44904                 this.el.dom.selectionEnd = curr;
44905             },
44906             
44907             "end" : function(e){
44908                 e.preventDefault();
44909                 
44910                 var curr = this.el.dom.selectionStart;
44911                 var lines = this.getValue().split("\n");
44912                 
44913                 if(!lines.length){
44914                     return;
44915                 }
44916                 
44917                 if(e.ctrlKey){
44918                     this.el.dom.setSelectionRange(this.getValue().length, this.getValue().length);
44919                     return;
44920                 }
44921                 
44922                 var pos = 0;
44923                 
44924                 for (var i = 0; i < lines.length;i++) {
44925                     
44926                     pos += lines[i].length;
44927                     
44928                     if(i != 0){
44929                         pos += 1;
44930                     }
44931                     
44932                     if(pos < curr){
44933                         continue;
44934                     }
44935                     
44936                     break;
44937                 }
44938                 
44939                 if(!e.shiftKey){
44940                     this.el.dom.setSelectionRange(pos, pos);
44941                     return;
44942                 }
44943                 
44944                 this.el.dom.selectionStart = curr;
44945                 this.el.dom.selectionEnd = pos;
44946             },
44947
44948             scope : this,
44949
44950             doRelay : function(foo, bar, hname){
44951                 return Roo.KeyNav.prototype.doRelay.apply(this, arguments);
44952             },
44953
44954             forceKeyDown: true
44955         });
44956         
44957 //        if(this.autosave && this.w){
44958 //            this.autoSaveFn = setInterval(this.autosave, 1000);
44959 //        }
44960     },
44961
44962     // private
44963     onResize : function(w, h)
44964     {
44965         Roo.form.HtmlEditor.superclass.onResize.apply(this, arguments);
44966         var ew = false;
44967         var eh = false;
44968         
44969         if(this.el ){
44970             if(typeof w == 'number'){
44971                 var aw = w - this.wrap.getFrameWidth('lr');
44972                 this.el.setWidth(this.adjustWidth('textarea', aw));
44973                 ew = aw;
44974             }
44975             if(typeof h == 'number'){
44976                 var tbh = 0;
44977                 for (var i =0; i < this.toolbars.length;i++) {
44978                     // fixme - ask toolbars for heights?
44979                     tbh += this.toolbars[i].tb.el.getHeight();
44980                     if (this.toolbars[i].footer) {
44981                         tbh += this.toolbars[i].footer.el.getHeight();
44982                     }
44983                 }
44984                 
44985                 
44986                 
44987                 
44988                 var ah = h - this.wrap.getFrameWidth('tb') - tbh;// this.tb.el.getHeight();
44989                 ah -= 5; // knock a few pixes off for look..
44990 //                Roo.log(ah);
44991                 this.el.setHeight(this.adjustWidth('textarea', ah));
44992                 var eh = ah;
44993             }
44994         }
44995         Roo.log('onResize:' + [w,h,ew,eh].join(',') );
44996         this.editorcore.onResize(ew,eh);
44997         
44998     },
44999
45000     /**
45001      * Toggles the editor between standard and source edit mode.
45002      * @param {Boolean} sourceEdit (optional) True for source edit, false for standard
45003      */
45004     toggleSourceEdit : function(sourceEditMode)
45005     {
45006         this.editorcore.toggleSourceEdit(sourceEditMode);
45007         
45008         if(this.editorcore.sourceEditMode){
45009             Roo.log('editor - showing textarea');
45010             
45011 //            Roo.log('in');
45012 //            Roo.log(this.syncValue());
45013             this.editorcore.syncValue();
45014             this.el.removeClass('x-hidden');
45015             this.el.dom.removeAttribute('tabIndex');
45016             this.el.focus();
45017             
45018             for (var i = 0; i < this.toolbars.length; i++) {
45019                 if(this.toolbars[i] instanceof Roo.form.HtmlEditor.ToolbarContext){
45020                     this.toolbars[i].tb.hide();
45021                     this.toolbars[i].footer.hide();
45022                 }
45023             }
45024             
45025         }else{
45026             Roo.log('editor - hiding textarea');
45027 //            Roo.log('out')
45028 //            Roo.log(this.pushValue()); 
45029             this.editorcore.pushValue();
45030             
45031             this.el.addClass('x-hidden');
45032             this.el.dom.setAttribute('tabIndex', -1);
45033             
45034             for (var i = 0; i < this.toolbars.length; i++) {
45035                 if(this.toolbars[i] instanceof Roo.form.HtmlEditor.ToolbarContext){
45036                     this.toolbars[i].tb.show();
45037                     this.toolbars[i].footer.show();
45038                 }
45039             }
45040             
45041             //this.deferFocus();
45042         }
45043         
45044         this.setSize(this.wrap.getSize());
45045         this.onResize(this.wrap.getSize().width, this.wrap.getSize().height);
45046         
45047         this.fireEvent('editmodechange', this, this.editorcore.sourceEditMode);
45048     },
45049  
45050     // private (for BoxComponent)
45051     adjustSize : Roo.BoxComponent.prototype.adjustSize,
45052
45053     // private (for BoxComponent)
45054     getResizeEl : function(){
45055         return this.wrap;
45056     },
45057
45058     // private (for BoxComponent)
45059     getPositionEl : function(){
45060         return this.wrap;
45061     },
45062
45063     // private
45064     initEvents : function(){
45065         this.originalValue = this.getValue();
45066     },
45067
45068     /**
45069      * Overridden and disabled. The editor element does not support standard valid/invalid marking. @hide
45070      * @method
45071      */
45072     markInvalid : Roo.emptyFn,
45073     /**
45074      * Overridden and disabled. The editor element does not support standard valid/invalid marking. @hide
45075      * @method
45076      */
45077     clearInvalid : Roo.emptyFn,
45078
45079     setValue : function(v){
45080         Roo.form.HtmlEditor.superclass.setValue.call(this, v);
45081         this.editorcore.pushValue();
45082     },
45083
45084      
45085     // private
45086     deferFocus : function(){
45087         this.focus.defer(10, this);
45088     },
45089
45090     // doc'ed in Field
45091     focus : function(){
45092         this.editorcore.focus();
45093         
45094     },
45095       
45096
45097     // private
45098     onDestroy : function(){
45099         
45100         
45101         
45102         if(this.rendered){
45103             
45104             for (var i =0; i < this.toolbars.length;i++) {
45105                 // fixme - ask toolbars for heights?
45106                 this.toolbars[i].onDestroy();
45107             }
45108             
45109             this.wrap.dom.innerHTML = '';
45110             this.wrap.remove();
45111         }
45112     },
45113
45114     // private
45115     onFirstFocus : function(){
45116         //Roo.log("onFirstFocus");
45117         this.editorcore.onFirstFocus();
45118          for (var i =0; i < this.toolbars.length;i++) {
45119             this.toolbars[i].onFirstFocus();
45120         }
45121         
45122     },
45123     
45124     // private
45125     syncValue : function()
45126     {
45127         this.editorcore.syncValue();
45128     },
45129     
45130     pushValue : function()
45131     {
45132         this.editorcore.pushValue();
45133     },
45134     
45135     setStylesheets : function(stylesheets)
45136     {
45137         this.editorcore.setStylesheets(stylesheets);
45138     },
45139     
45140     removeStylesheets : function()
45141     {
45142         this.editorcore.removeStylesheets();
45143     }
45144      
45145     
45146     // hide stuff that is not compatible
45147     /**
45148      * @event blur
45149      * @hide
45150      */
45151     /**
45152      * @event change
45153      * @hide
45154      */
45155     /**
45156      * @event focus
45157      * @hide
45158      */
45159     /**
45160      * @event specialkey
45161      * @hide
45162      */
45163     /**
45164      * @cfg {String} fieldClass @hide
45165      */
45166     /**
45167      * @cfg {String} focusClass @hide
45168      */
45169     /**
45170      * @cfg {String} autoCreate @hide
45171      */
45172     /**
45173      * @cfg {String} inputType @hide
45174      */
45175     /**
45176      * @cfg {String} invalidClass @hide
45177      */
45178     /**
45179      * @cfg {String} invalidText @hide
45180      */
45181     /**
45182      * @cfg {String} msgFx @hide
45183      */
45184     /**
45185      * @cfg {String} validateOnBlur @hide
45186      */
45187 });
45188  
45189     // <script type="text/javascript">
45190 /*
45191  * Based on
45192  * Ext JS Library 1.1.1
45193  * Copyright(c) 2006-2007, Ext JS, LLC.
45194  *  
45195  
45196  */
45197
45198 /**
45199  * @class Roo.form.HtmlEditorToolbar1
45200  * Basic Toolbar
45201  * 
45202  * Usage:
45203  *
45204  new Roo.form.HtmlEditor({
45205     ....
45206     toolbars : [
45207         new Roo.form.HtmlEditorToolbar1({
45208             disable : { fonts: 1 , format: 1, ..., ... , ...],
45209             btns : [ .... ]
45210         })
45211     }
45212      
45213  * 
45214  * @cfg {Object} disable List of elements to disable..
45215  * @cfg {Array} btns List of additional buttons.
45216  * 
45217  * 
45218  * NEEDS Extra CSS? 
45219  * .x-html-editor-tb .x-edit-none .x-btn-text { background: none; }
45220  */
45221  
45222 Roo.form.HtmlEditor.ToolbarStandard = function(config)
45223 {
45224     
45225     Roo.apply(this, config);
45226     
45227     // default disabled, based on 'good practice'..
45228     this.disable = this.disable || {};
45229     Roo.applyIf(this.disable, {
45230         fontSize : true,
45231         colors : true,
45232         specialElements : true
45233     });
45234     
45235     
45236     //Roo.form.HtmlEditorToolbar1.superclass.constructor.call(this, editor.wrap.dom.firstChild, [], config);
45237     // dont call parent... till later.
45238 }
45239
45240 Roo.apply(Roo.form.HtmlEditor.ToolbarStandard.prototype,  {
45241     
45242     tb: false,
45243     
45244     rendered: false,
45245     
45246     editor : false,
45247     editorcore : false,
45248     /**
45249      * @cfg {Object} disable  List of toolbar elements to disable
45250          
45251      */
45252     disable : false,
45253     
45254     
45255      /**
45256      * @cfg {String} createLinkText The default text for the create link prompt
45257      */
45258     createLinkText : 'Please enter the URL for the link:',
45259     /**
45260      * @cfg {String} defaultLinkValue The default value for the create link prompt (defaults to http:/ /)
45261      */
45262     defaultLinkValue : 'http:/'+'/',
45263    
45264     
45265       /**
45266      * @cfg {Array} fontFamilies An array of available font families
45267      */
45268     fontFamilies : [
45269         'Arial',
45270         'Courier New',
45271         'Tahoma',
45272         'Times New Roman',
45273         'Verdana'
45274     ],
45275     
45276     specialChars : [
45277            "&#169;",
45278           "&#174;",     
45279           "&#8482;",    
45280           "&#163;" ,    
45281          // "&#8212;",    
45282           "&#8230;",    
45283           "&#247;" ,    
45284         //  "&#225;" ,     ?? a acute?
45285            "&#8364;"    , //Euro
45286        //   "&#8220;"    ,
45287         //  "&#8221;"    ,
45288         //  "&#8226;"    ,
45289           "&#176;"  //   , // degrees
45290
45291          // "&#233;"     , // e ecute
45292          // "&#250;"     , // u ecute?
45293     ],
45294     
45295     specialElements : [
45296         {
45297             text: "Insert Table",
45298             xtype: 'MenuItem',
45299             xns : Roo.Menu,
45300             ihtml :  '<table><tr><td>Cell</td></tr></table>' 
45301                 
45302         },
45303         {    
45304             text: "Insert Image",
45305             xtype: 'MenuItem',
45306             xns : Roo.Menu,
45307             ihtml : '<img src="about:blank"/>'
45308             
45309         }
45310         
45311          
45312     ],
45313     
45314     
45315     inputElements : [ 
45316             "form", "input:text", "input:hidden", "input:checkbox", "input:radio", "input:password", 
45317             "input:submit", "input:button", "select", "textarea", "label" ],
45318     formats : [
45319         ["p"] ,  
45320         ["h1"],["h2"],["h3"],["h4"],["h5"],["h6"], 
45321         ["pre"],[ "code"], 
45322         ["abbr"],[ "acronym"],[ "address"],[ "cite"],[ "samp"],[ "var"],
45323         ['div'],['span']
45324     ],
45325     
45326     cleanStyles : [
45327         "font-size"
45328     ],
45329      /**
45330      * @cfg {String} defaultFont default font to use.
45331      */
45332     defaultFont: 'tahoma',
45333    
45334     fontSelect : false,
45335     
45336     
45337     formatCombo : false,
45338     
45339     init : function(editor)
45340     {
45341         this.editor = editor;
45342         this.editorcore = editor.editorcore ? editor.editorcore : editor;
45343         var editorcore = this.editorcore;
45344         
45345         var _t = this;
45346         
45347         var fid = editorcore.frameId;
45348         var etb = this;
45349         function btn(id, toggle, handler){
45350             var xid = fid + '-'+ id ;
45351             return {
45352                 id : xid,
45353                 cmd : id,
45354                 cls : 'x-btn-icon x-edit-'+id,
45355                 enableToggle:toggle !== false,
45356                 scope: _t, // was editor...
45357                 handler:handler||_t.relayBtnCmd,
45358                 clickEvent:'mousedown',
45359                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
45360                 tabIndex:-1
45361             };
45362         }
45363         
45364         
45365         
45366         var tb = new Roo.Toolbar(editor.wrap.dom.firstChild);
45367         this.tb = tb;
45368          // stop form submits
45369         tb.el.on('click', function(e){
45370             e.preventDefault(); // what does this do?
45371         });
45372
45373         if(!this.disable.font) { // && !Roo.isSafari){
45374             /* why no safari for fonts 
45375             editor.fontSelect = tb.el.createChild({
45376                 tag:'select',
45377                 tabIndex: -1,
45378                 cls:'x-font-select',
45379                 html: this.createFontOptions()
45380             });
45381             
45382             editor.fontSelect.on('change', function(){
45383                 var font = editor.fontSelect.dom.value;
45384                 editor.relayCmd('fontname', font);
45385                 editor.deferFocus();
45386             }, editor);
45387             
45388             tb.add(
45389                 editor.fontSelect.dom,
45390                 '-'
45391             );
45392             */
45393             
45394         };
45395         if(!this.disable.formats){
45396             this.formatCombo = new Roo.form.ComboBox({
45397                 store: new Roo.data.SimpleStore({
45398                     id : 'tag',
45399                     fields: ['tag'],
45400                     data : this.formats // from states.js
45401                 }),
45402                 blockFocus : true,
45403                 name : '',
45404                 //autoCreate : {tag: "div",  size: "20"},
45405                 displayField:'tag',
45406                 typeAhead: false,
45407                 mode: 'local',
45408                 editable : false,
45409                 triggerAction: 'all',
45410                 emptyText:'Add tag',
45411                 selectOnFocus:true,
45412                 width:135,
45413                 listeners : {
45414                     'select': function(c, r, i) {
45415                         editorcore.insertTag(r.get('tag'));
45416                         editor.focus();
45417                     }
45418                 }
45419
45420             });
45421             tb.addField(this.formatCombo);
45422             
45423         }
45424         
45425         if(!this.disable.format){
45426             tb.add(
45427                 btn('bold'),
45428                 btn('italic'),
45429                 btn('underline'),
45430                 btn('strikethrough')
45431             );
45432         };
45433         if(!this.disable.fontSize){
45434             tb.add(
45435                 '-',
45436                 
45437                 
45438                 btn('increasefontsize', false, editorcore.adjustFont),
45439                 btn('decreasefontsize', false, editorcore.adjustFont)
45440             );
45441         };
45442         
45443         
45444         if(!this.disable.colors){
45445             tb.add(
45446                 '-', {
45447                     id:editorcore.frameId +'-forecolor',
45448                     cls:'x-btn-icon x-edit-forecolor',
45449                     clickEvent:'mousedown',
45450                     tooltip: this.buttonTips['forecolor'] || undefined,
45451                     tabIndex:-1,
45452                     menu : new Roo.menu.ColorMenu({
45453                         allowReselect: true,
45454                         focus: Roo.emptyFn,
45455                         value:'000000',
45456                         plain:true,
45457                         selectHandler: function(cp, color){
45458                             editorcore.execCmd('forecolor', Roo.isSafari || Roo.isIE ? '#'+color : color);
45459                             editor.deferFocus();
45460                         },
45461                         scope: editorcore,
45462                         clickEvent:'mousedown'
45463                     })
45464                 }, {
45465                     id:editorcore.frameId +'backcolor',
45466                     cls:'x-btn-icon x-edit-backcolor',
45467                     clickEvent:'mousedown',
45468                     tooltip: this.buttonTips['backcolor'] || undefined,
45469                     tabIndex:-1,
45470                     menu : new Roo.menu.ColorMenu({
45471                         focus: Roo.emptyFn,
45472                         value:'FFFFFF',
45473                         plain:true,
45474                         allowReselect: true,
45475                         selectHandler: function(cp, color){
45476                             if(Roo.isGecko){
45477                                 editorcore.execCmd('useCSS', false);
45478                                 editorcore.execCmd('hilitecolor', color);
45479                                 editorcore.execCmd('useCSS', true);
45480                                 editor.deferFocus();
45481                             }else{
45482                                 editorcore.execCmd(Roo.isOpera ? 'hilitecolor' : 'backcolor', 
45483                                     Roo.isSafari || Roo.isIE ? '#'+color : color);
45484                                 editor.deferFocus();
45485                             }
45486                         },
45487                         scope:editorcore,
45488                         clickEvent:'mousedown'
45489                     })
45490                 }
45491             );
45492         };
45493         // now add all the items...
45494         
45495
45496         if(!this.disable.alignments){
45497             tb.add(
45498                 '-',
45499                 btn('justifyleft'),
45500                 btn('justifycenter'),
45501                 btn('justifyright')
45502             );
45503         };
45504
45505         //if(!Roo.isSafari){
45506             if(!this.disable.links){
45507                 tb.add(
45508                     '-',
45509                     btn('createlink', false, this.createLink)    /// MOVE TO HERE?!!?!?!?!
45510                 );
45511             };
45512
45513             if(!this.disable.lists){
45514                 tb.add(
45515                     '-',
45516                     btn('insertorderedlist'),
45517                     btn('insertunorderedlist')
45518                 );
45519             }
45520             if(!this.disable.sourceEdit){
45521                 tb.add(
45522                     '-',
45523                     btn('sourceedit', true, function(btn){
45524                         this.toggleSourceEdit(btn.pressed);
45525                     })
45526                 );
45527             }
45528         //}
45529         
45530         var smenu = { };
45531         // special menu.. - needs to be tidied up..
45532         if (!this.disable.special) {
45533             smenu = {
45534                 text: "&#169;",
45535                 cls: 'x-edit-none',
45536                 
45537                 menu : {
45538                     items : []
45539                 }
45540             };
45541             for (var i =0; i < this.specialChars.length; i++) {
45542                 smenu.menu.items.push({
45543                     
45544                     html: this.specialChars[i],
45545                     handler: function(a,b) {
45546                         editorcore.insertAtCursor(String.fromCharCode(a.html.replace('&#','').replace(';', '')));
45547                         //editor.insertAtCursor(a.html);
45548                         
45549                     },
45550                     tabIndex:-1
45551                 });
45552             }
45553             
45554             
45555             tb.add(smenu);
45556             
45557             
45558         }
45559         
45560         var cmenu = { };
45561         if (!this.disable.cleanStyles) {
45562             cmenu = {
45563                 cls: 'x-btn-icon x-btn-clear',
45564                 
45565                 menu : {
45566                     items : []
45567                 }
45568             };
45569             for (var i =0; i < this.cleanStyles.length; i++) {
45570                 cmenu.menu.items.push({
45571                     actiontype : this.cleanStyles[i],
45572                     html: 'Remove ' + this.cleanStyles[i],
45573                     handler: function(a,b) {
45574 //                        Roo.log(a);
45575 //                        Roo.log(b);
45576                         var c = Roo.get(editorcore.doc.body);
45577                         c.select('[style]').each(function(s) {
45578                             s.dom.style.removeProperty(a.actiontype);
45579                         });
45580                         editorcore.syncValue();
45581                     },
45582                     tabIndex:-1
45583                 });
45584             }
45585              cmenu.menu.items.push({
45586                 actiontype : 'tablewidths',
45587                 html: 'Remove Table Widths',
45588                 handler: function(a,b) {
45589                     editorcore.cleanTableWidths();
45590                     editorcore.syncValue();
45591                 },
45592                 tabIndex:-1
45593             });
45594             cmenu.menu.items.push({
45595                 actiontype : 'word',
45596                 html: 'Remove MS Word Formating',
45597                 handler: function(a,b) {
45598                     editorcore.cleanWord();
45599                     editorcore.syncValue();
45600                 },
45601                 tabIndex:-1
45602             });
45603             
45604             cmenu.menu.items.push({
45605                 actiontype : 'all',
45606                 html: 'Remove All Styles',
45607                 handler: function(a,b) {
45608                     
45609                     var c = Roo.get(editorcore.doc.body);
45610                     c.select('[style]').each(function(s) {
45611                         s.dom.removeAttribute('style');
45612                     });
45613                     editorcore.syncValue();
45614                 },
45615                 tabIndex:-1
45616             });
45617             
45618             cmenu.menu.items.push({
45619                 actiontype : 'all',
45620                 html: 'Remove All CSS Classes',
45621                 handler: function(a,b) {
45622                     
45623                     var c = Roo.get(editorcore.doc.body);
45624                     c.select('[class]').each(function(s) {
45625                         s.dom.className = '';
45626                     });
45627                     editorcore.syncValue();
45628                 },
45629                 tabIndex:-1
45630             });
45631             
45632              cmenu.menu.items.push({
45633                 actiontype : 'tidy',
45634                 html: 'Tidy HTML Source',
45635                 handler: function(a,b) {
45636                     editorcore.doc.body.innerHTML = editorcore.domToHTML();
45637                     editorcore.syncValue();
45638                 },
45639                 tabIndex:-1
45640             });
45641             
45642             
45643             tb.add(cmenu);
45644         }
45645          
45646         if (!this.disable.specialElements) {
45647             var semenu = {
45648                 text: "Other;",
45649                 cls: 'x-edit-none',
45650                 menu : {
45651                     items : []
45652                 }
45653             };
45654             for (var i =0; i < this.specialElements.length; i++) {
45655                 semenu.menu.items.push(
45656                     Roo.apply({ 
45657                         handler: function(a,b) {
45658                             editor.insertAtCursor(this.ihtml);
45659                         }
45660                     }, this.specialElements[i])
45661                 );
45662                     
45663             }
45664             
45665             tb.add(semenu);
45666             
45667             
45668         }
45669          
45670         
45671         if (this.btns) {
45672             for(var i =0; i< this.btns.length;i++) {
45673                 var b = Roo.factory(this.btns[i],Roo.form);
45674                 b.cls =  'x-edit-none';
45675                 
45676                 if(typeof(this.btns[i].cls) != 'undefined' && this.btns[i].cls.indexOf('x-init-enable') !== -1){
45677                     b.cls += ' x-init-enable';
45678                 }
45679                 
45680                 b.scope = editorcore;
45681                 tb.add(b);
45682             }
45683         
45684         }
45685         
45686         
45687         
45688         // disable everything...
45689         
45690         this.tb.items.each(function(item){
45691             
45692            if(
45693                 item.id != editorcore.frameId+ '-sourceedit' && 
45694                 (typeof(item.cls) != 'undefined' && item.cls.indexOf('x-init-enable') === -1)
45695             ){
45696                 
45697                 item.disable();
45698             }
45699         });
45700         this.rendered = true;
45701         
45702         // the all the btns;
45703         editor.on('editorevent', this.updateToolbar, this);
45704         // other toolbars need to implement this..
45705         //editor.on('editmodechange', this.updateToolbar, this);
45706     },
45707     
45708     
45709     relayBtnCmd : function(btn) {
45710         this.editorcore.relayCmd(btn.cmd);
45711     },
45712     // private used internally
45713     createLink : function(){
45714         Roo.log("create link?");
45715         var url = prompt(this.createLinkText, this.defaultLinkValue);
45716         if(url && url != 'http:/'+'/'){
45717             this.editorcore.relayCmd('createlink', url);
45718         }
45719     },
45720
45721     
45722     /**
45723      * Protected method that will not generally be called directly. It triggers
45724      * a toolbar update by reading the markup state of the current selection in the editor.
45725      */
45726     updateToolbar: function(){
45727
45728         if(!this.editorcore.activated){
45729             this.editor.onFirstFocus();
45730             return;
45731         }
45732
45733         var btns = this.tb.items.map, 
45734             doc = this.editorcore.doc,
45735             frameId = this.editorcore.frameId;
45736
45737         if(!this.disable.font && !Roo.isSafari){
45738             /*
45739             var name = (doc.queryCommandValue('FontName')||this.editor.defaultFont).toLowerCase();
45740             if(name != this.fontSelect.dom.value){
45741                 this.fontSelect.dom.value = name;
45742             }
45743             */
45744         }
45745         if(!this.disable.format){
45746             btns[frameId + '-bold'].toggle(doc.queryCommandState('bold'));
45747             btns[frameId + '-italic'].toggle(doc.queryCommandState('italic'));
45748             btns[frameId + '-underline'].toggle(doc.queryCommandState('underline'));
45749             btns[frameId + '-strikethrough'].toggle(doc.queryCommandState('strikethrough'));
45750         }
45751         if(!this.disable.alignments){
45752             btns[frameId + '-justifyleft'].toggle(doc.queryCommandState('justifyleft'));
45753             btns[frameId + '-justifycenter'].toggle(doc.queryCommandState('justifycenter'));
45754             btns[frameId + '-justifyright'].toggle(doc.queryCommandState('justifyright'));
45755         }
45756         if(!Roo.isSafari && !this.disable.lists){
45757             btns[frameId + '-insertorderedlist'].toggle(doc.queryCommandState('insertorderedlist'));
45758             btns[frameId + '-insertunorderedlist'].toggle(doc.queryCommandState('insertunorderedlist'));
45759         }
45760         
45761         var ans = this.editorcore.getAllAncestors();
45762         if (this.formatCombo) {
45763             
45764             
45765             var store = this.formatCombo.store;
45766             this.formatCombo.setValue("");
45767             for (var i =0; i < ans.length;i++) {
45768                 if (ans[i] && store.query('tag',ans[i].tagName.toLowerCase(), false).length) {
45769                     // select it..
45770                     this.formatCombo.setValue(ans[i].tagName.toLowerCase());
45771                     break;
45772                 }
45773             }
45774         }
45775         
45776         
45777         
45778         // hides menus... - so this cant be on a menu...
45779         Roo.menu.MenuMgr.hideAll();
45780
45781         //this.editorsyncValue();
45782     },
45783    
45784     
45785     createFontOptions : function(){
45786         var buf = [], fs = this.fontFamilies, ff, lc;
45787         
45788         
45789         
45790         for(var i = 0, len = fs.length; i< len; i++){
45791             ff = fs[i];
45792             lc = ff.toLowerCase();
45793             buf.push(
45794                 '<option value="',lc,'" style="font-family:',ff,';"',
45795                     (this.defaultFont == lc ? ' selected="true">' : '>'),
45796                     ff,
45797                 '</option>'
45798             );
45799         }
45800         return buf.join('');
45801     },
45802     
45803     toggleSourceEdit : function(sourceEditMode){
45804         
45805         Roo.log("toolbar toogle");
45806         if(sourceEditMode === undefined){
45807             sourceEditMode = !this.sourceEditMode;
45808         }
45809         this.sourceEditMode = sourceEditMode === true;
45810         var btn = this.tb.items.get(this.editorcore.frameId +'-sourceedit');
45811         // just toggle the button?
45812         if(btn.pressed !== this.sourceEditMode){
45813             btn.toggle(this.sourceEditMode);
45814             return;
45815         }
45816         
45817         if(sourceEditMode){
45818             Roo.log("disabling buttons");
45819             this.tb.items.each(function(item){
45820                 if(item.cmd != 'sourceedit' && (typeof(item.cls) != 'undefined' && item.cls.indexOf('x-init-enable') === -1)){
45821                     item.disable();
45822                 }
45823             });
45824           
45825         }else{
45826             Roo.log("enabling buttons");
45827             if(this.editorcore.initialized){
45828                 this.tb.items.each(function(item){
45829                     item.enable();
45830                 });
45831             }
45832             
45833         }
45834         Roo.log("calling toggole on editor");
45835         // tell the editor that it's been pressed..
45836         this.editor.toggleSourceEdit(sourceEditMode);
45837        
45838     },
45839      /**
45840      * Object collection of toolbar tooltips for the buttons in the editor. The key
45841      * is the command id associated with that button and the value is a valid QuickTips object.
45842      * For example:
45843 <pre><code>
45844 {
45845     bold : {
45846         title: 'Bold (Ctrl+B)',
45847         text: 'Make the selected text bold.',
45848         cls: 'x-html-editor-tip'
45849     },
45850     italic : {
45851         title: 'Italic (Ctrl+I)',
45852         text: 'Make the selected text italic.',
45853         cls: 'x-html-editor-tip'
45854     },
45855     ...
45856 </code></pre>
45857     * @type Object
45858      */
45859     buttonTips : {
45860         bold : {
45861             title: 'Bold (Ctrl+B)',
45862             text: 'Make the selected text bold.',
45863             cls: 'x-html-editor-tip'
45864         },
45865         italic : {
45866             title: 'Italic (Ctrl+I)',
45867             text: 'Make the selected text italic.',
45868             cls: 'x-html-editor-tip'
45869         },
45870         underline : {
45871             title: 'Underline (Ctrl+U)',
45872             text: 'Underline the selected text.',
45873             cls: 'x-html-editor-tip'
45874         },
45875         strikethrough : {
45876             title: 'Strikethrough',
45877             text: 'Strikethrough the selected text.',
45878             cls: 'x-html-editor-tip'
45879         },
45880         increasefontsize : {
45881             title: 'Grow Text',
45882             text: 'Increase the font size.',
45883             cls: 'x-html-editor-tip'
45884         },
45885         decreasefontsize : {
45886             title: 'Shrink Text',
45887             text: 'Decrease the font size.',
45888             cls: 'x-html-editor-tip'
45889         },
45890         backcolor : {
45891             title: 'Text Highlight Color',
45892             text: 'Change the background color of the selected text.',
45893             cls: 'x-html-editor-tip'
45894         },
45895         forecolor : {
45896             title: 'Font Color',
45897             text: 'Change the color of the selected text.',
45898             cls: 'x-html-editor-tip'
45899         },
45900         justifyleft : {
45901             title: 'Align Text Left',
45902             text: 'Align text to the left.',
45903             cls: 'x-html-editor-tip'
45904         },
45905         justifycenter : {
45906             title: 'Center Text',
45907             text: 'Center text in the editor.',
45908             cls: 'x-html-editor-tip'
45909         },
45910         justifyright : {
45911             title: 'Align Text Right',
45912             text: 'Align text to the right.',
45913             cls: 'x-html-editor-tip'
45914         },
45915         insertunorderedlist : {
45916             title: 'Bullet List',
45917             text: 'Start a bulleted list.',
45918             cls: 'x-html-editor-tip'
45919         },
45920         insertorderedlist : {
45921             title: 'Numbered List',
45922             text: 'Start a numbered list.',
45923             cls: 'x-html-editor-tip'
45924         },
45925         createlink : {
45926             title: 'Hyperlink',
45927             text: 'Make the selected text a hyperlink.',
45928             cls: 'x-html-editor-tip'
45929         },
45930         sourceedit : {
45931             title: 'Source Edit',
45932             text: 'Switch to source editing mode.',
45933             cls: 'x-html-editor-tip'
45934         }
45935     },
45936     // private
45937     onDestroy : function(){
45938         if(this.rendered){
45939             
45940             this.tb.items.each(function(item){
45941                 if(item.menu){
45942                     item.menu.removeAll();
45943                     if(item.menu.el){
45944                         item.menu.el.destroy();
45945                     }
45946                 }
45947                 item.destroy();
45948             });
45949              
45950         }
45951     },
45952     onFirstFocus: function() {
45953         this.tb.items.each(function(item){
45954            item.enable();
45955         });
45956     }
45957 });
45958
45959
45960
45961
45962 // <script type="text/javascript">
45963 /*
45964  * Based on
45965  * Ext JS Library 1.1.1
45966  * Copyright(c) 2006-2007, Ext JS, LLC.
45967  *  
45968  
45969  */
45970
45971  
45972 /**
45973  * @class Roo.form.HtmlEditor.ToolbarContext
45974  * Context Toolbar
45975  * 
45976  * Usage:
45977  *
45978  new Roo.form.HtmlEditor({
45979     ....
45980     toolbars : [
45981         { xtype: 'ToolbarStandard', styles : {} }
45982         { xtype: 'ToolbarContext', disable : {} }
45983     ]
45984 })
45985
45986      
45987  * 
45988  * @config : {Object} disable List of elements to disable.. (not done yet.)
45989  * @config : {Object} styles  Map of styles available.
45990  * 
45991  */
45992
45993 Roo.form.HtmlEditor.ToolbarContext = function(config)
45994 {
45995     
45996     Roo.apply(this, config);
45997     //Roo.form.HtmlEditorToolbar1.superclass.constructor.call(this, editor.wrap.dom.firstChild, [], config);
45998     // dont call parent... till later.
45999     this.styles = this.styles || {};
46000 }
46001
46002  
46003
46004 Roo.form.HtmlEditor.ToolbarContext.types = {
46005     'IMG' : {
46006         width : {
46007             title: "Width",
46008             width: 40
46009         },
46010         height:  {
46011             title: "Height",
46012             width: 40
46013         },
46014         align: {
46015             title: "Align",
46016             opts : [ [""],[ "left"],[ "right"],[ "center"],[ "top"]],
46017             width : 80
46018             
46019         },
46020         border: {
46021             title: "Border",
46022             width: 40
46023         },
46024         alt: {
46025             title: "Alt",
46026             width: 120
46027         },
46028         src : {
46029             title: "Src",
46030             width: 220
46031         }
46032         
46033     },
46034     'A' : {
46035         name : {
46036             title: "Name",
46037             width: 50
46038         },
46039         target:  {
46040             title: "Target",
46041             width: 120
46042         },
46043         href:  {
46044             title: "Href",
46045             width: 220
46046         } // border?
46047         
46048     },
46049     'TABLE' : {
46050         rows : {
46051             title: "Rows",
46052             width: 20
46053         },
46054         cols : {
46055             title: "Cols",
46056             width: 20
46057         },
46058         width : {
46059             title: "Width",
46060             width: 40
46061         },
46062         height : {
46063             title: "Height",
46064             width: 40
46065         },
46066         border : {
46067             title: "Border",
46068             width: 20
46069         }
46070     },
46071     'TD' : {
46072         width : {
46073             title: "Width",
46074             width: 40
46075         },
46076         height : {
46077             title: "Height",
46078             width: 40
46079         },   
46080         align: {
46081             title: "Align",
46082             opts : [[""],[ "left"],[ "center"],[ "right"],[ "justify"],[ "char"]],
46083             width: 80
46084         },
46085         valign: {
46086             title: "Valign",
46087             opts : [[""],[ "top"],[ "middle"],[ "bottom"],[ "baseline"]],
46088             width: 80
46089         },
46090         colspan: {
46091             title: "Colspan",
46092             width: 20
46093             
46094         },
46095          'font-family'  : {
46096             title : "Font",
46097             style : 'fontFamily',
46098             displayField: 'display',
46099             optname : 'font-family',
46100             width: 140
46101         }
46102     },
46103     'INPUT' : {
46104         name : {
46105             title: "name",
46106             width: 120
46107         },
46108         value : {
46109             title: "Value",
46110             width: 120
46111         },
46112         width : {
46113             title: "Width",
46114             width: 40
46115         }
46116     },
46117     'LABEL' : {
46118         'for' : {
46119             title: "For",
46120             width: 120
46121         }
46122     },
46123     'TEXTAREA' : {
46124           name : {
46125             title: "name",
46126             width: 120
46127         },
46128         rows : {
46129             title: "Rows",
46130             width: 20
46131         },
46132         cols : {
46133             title: "Cols",
46134             width: 20
46135         }
46136     },
46137     'SELECT' : {
46138         name : {
46139             title: "name",
46140             width: 120
46141         },
46142         selectoptions : {
46143             title: "Options",
46144             width: 200
46145         }
46146     },
46147     
46148     // should we really allow this??
46149     // should this just be 
46150     'BODY' : {
46151         title : {
46152             title: "Title",
46153             width: 200,
46154             disabled : true
46155         }
46156     },
46157     'SPAN' : {
46158         'font-family'  : {
46159             title : "Font",
46160             style : 'fontFamily',
46161             displayField: 'display',
46162             optname : 'font-family',
46163             width: 140
46164         }
46165     },
46166     'DIV' : {
46167         'font-family'  : {
46168             title : "Font",
46169             style : 'fontFamily',
46170             displayField: 'display',
46171             optname : 'font-family',
46172             width: 140
46173         }
46174     },
46175      'P' : {
46176         'font-family'  : {
46177             title : "Font",
46178             style : 'fontFamily',
46179             displayField: 'display',
46180             optname : 'font-family',
46181             width: 140
46182         }
46183     },
46184     
46185     '*' : {
46186         // empty..
46187     }
46188
46189 };
46190
46191 // this should be configurable.. - you can either set it up using stores, or modify options somehwere..
46192 Roo.form.HtmlEditor.ToolbarContext.stores = false;
46193
46194 Roo.form.HtmlEditor.ToolbarContext.options = {
46195         'font-family'  : [ 
46196                 [ 'Helvetica,Arial,sans-serif', 'Helvetica'],
46197                 [ 'Courier New', 'Courier New'],
46198                 [ 'Tahoma', 'Tahoma'],
46199                 [ 'Times New Roman,serif', 'Times'],
46200                 [ 'Verdana','Verdana' ]
46201         ]
46202 };
46203
46204 // fixme - these need to be configurable..
46205  
46206
46207 //Roo.form.HtmlEditor.ToolbarContext.types
46208
46209
46210 Roo.apply(Roo.form.HtmlEditor.ToolbarContext.prototype,  {
46211     
46212     tb: false,
46213     
46214     rendered: false,
46215     
46216     editor : false,
46217     editorcore : false,
46218     /**
46219      * @cfg {Object} disable  List of toolbar elements to disable
46220          
46221      */
46222     disable : false,
46223     /**
46224      * @cfg {Object} styles List of styles 
46225      *    eg. { '*' : [ 'headline' ] , 'TD' : [ 'underline', 'double-underline' ] } 
46226      *
46227      * These must be defined in the page, so they get rendered correctly..
46228      * .headline { }
46229      * TD.underline { }
46230      * 
46231      */
46232     styles : false,
46233     
46234     options: false,
46235     
46236     toolbars : false,
46237     
46238     init : function(editor)
46239     {
46240         this.editor = editor;
46241         this.editorcore = editor.editorcore ? editor.editorcore : editor;
46242         var editorcore = this.editorcore;
46243         
46244         var fid = editorcore.frameId;
46245         var etb = this;
46246         function btn(id, toggle, handler){
46247             var xid = fid + '-'+ id ;
46248             return {
46249                 id : xid,
46250                 cmd : id,
46251                 cls : 'x-btn-icon x-edit-'+id,
46252                 enableToggle:toggle !== false,
46253                 scope: editorcore, // was editor...
46254                 handler:handler||editorcore.relayBtnCmd,
46255                 clickEvent:'mousedown',
46256                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
46257                 tabIndex:-1
46258             };
46259         }
46260         // create a new element.
46261         var wdiv = editor.wrap.createChild({
46262                 tag: 'div'
46263             }, editor.wrap.dom.firstChild.nextSibling, true);
46264         
46265         // can we do this more than once??
46266         
46267          // stop form submits
46268       
46269  
46270         // disable everything...
46271         var ty= Roo.form.HtmlEditor.ToolbarContext.types;
46272         this.toolbars = {};
46273            
46274         for (var i in  ty) {
46275           
46276             this.toolbars[i] = this.buildToolbar(ty[i],i);
46277         }
46278         this.tb = this.toolbars.BODY;
46279         this.tb.el.show();
46280         this.buildFooter();
46281         this.footer.show();
46282         editor.on('hide', function( ) { this.footer.hide() }, this);
46283         editor.on('show', function( ) { this.footer.show() }, this);
46284         
46285          
46286         this.rendered = true;
46287         
46288         // the all the btns;
46289         editor.on('editorevent', this.updateToolbar, this);
46290         // other toolbars need to implement this..
46291         //editor.on('editmodechange', this.updateToolbar, this);
46292     },
46293     
46294     
46295     
46296     /**
46297      * Protected method that will not generally be called directly. It triggers
46298      * a toolbar update by reading the markup state of the current selection in the editor.
46299      *
46300      * Note you can force an update by calling on('editorevent', scope, false)
46301      */
46302     updateToolbar: function(editor,ev,sel){
46303
46304         //Roo.log(ev);
46305         // capture mouse up - this is handy for selecting images..
46306         // perhaps should go somewhere else...
46307         if(!this.editorcore.activated){
46308              this.editor.onFirstFocus();
46309             return;
46310         }
46311         
46312         
46313         
46314         // http://developer.yahoo.com/yui/docs/simple-editor.js.html
46315         // selectNode - might want to handle IE?
46316         if (ev &&
46317             (ev.type == 'mouseup' || ev.type == 'click' ) &&
46318             ev.target && ev.target.tagName == 'IMG') {
46319             // they have click on an image...
46320             // let's see if we can change the selection...
46321             sel = ev.target;
46322          
46323               var nodeRange = sel.ownerDocument.createRange();
46324             try {
46325                 nodeRange.selectNode(sel);
46326             } catch (e) {
46327                 nodeRange.selectNodeContents(sel);
46328             }
46329             //nodeRange.collapse(true);
46330             var s = this.editorcore.win.getSelection();
46331             s.removeAllRanges();
46332             s.addRange(nodeRange);
46333         }  
46334         
46335       
46336         var updateFooter = sel ? false : true;
46337         
46338         
46339         var ans = this.editorcore.getAllAncestors();
46340         
46341         // pick
46342         var ty= Roo.form.HtmlEditor.ToolbarContext.types;
46343         
46344         if (!sel) { 
46345             sel = ans.length ? (ans[0] ?  ans[0]  : ans[1]) : this.editorcore.doc.body;
46346             sel = sel ? sel : this.editorcore.doc.body;
46347             sel = sel.tagName.length ? sel : this.editorcore.doc.body;
46348             
46349         }
46350         // pick a menu that exists..
46351         var tn = sel.tagName.toUpperCase();
46352         //sel = typeof(ty[tn]) != 'undefined' ? sel : this.editor.doc.body;
46353         
46354         tn = sel.tagName.toUpperCase();
46355         
46356         var lastSel = this.tb.selectedNode;
46357         
46358         this.tb.selectedNode = sel;
46359         
46360         // if current menu does not match..
46361         
46362         if ((this.tb.name != tn) || (lastSel != this.tb.selectedNode) || ev === false) {
46363                 
46364             this.tb.el.hide();
46365             ///console.log("show: " + tn);
46366             this.tb =  typeof(ty[tn]) != 'undefined' ? this.toolbars[tn] : this.toolbars['*'];
46367             this.tb.el.show();
46368             // update name
46369             this.tb.items.first().el.innerHTML = tn + ':&nbsp;';
46370             
46371             
46372             // update attributes
46373             if (this.tb.fields) {
46374                 this.tb.fields.each(function(e) {
46375                     if (e.stylename) {
46376                         e.setValue(sel.style[e.stylename]);
46377                         return;
46378                     } 
46379                    e.setValue(sel.getAttribute(e.attrname));
46380                 });
46381             }
46382             
46383             var hasStyles = false;
46384             for(var i in this.styles) {
46385                 hasStyles = true;
46386                 break;
46387             }
46388             
46389             // update styles
46390             if (hasStyles) { 
46391                 var st = this.tb.fields.item(0);
46392                 
46393                 st.store.removeAll();
46394                
46395                 
46396                 var cn = sel.className.split(/\s+/);
46397                 
46398                 var avs = [];
46399                 if (this.styles['*']) {
46400                     
46401                     Roo.each(this.styles['*'], function(v) {
46402                         avs.push( [ v , cn.indexOf(v) > -1 ? 1 : 0 ] );         
46403                     });
46404                 }
46405                 if (this.styles[tn]) { 
46406                     Roo.each(this.styles[tn], function(v) {
46407                         avs.push( [ v , cn.indexOf(v) > -1 ? 1 : 0 ] );         
46408                     });
46409                 }
46410                 
46411                 st.store.loadData(avs);
46412                 st.collapse();
46413                 st.setValue(cn);
46414             }
46415             // flag our selected Node.
46416             this.tb.selectedNode = sel;
46417            
46418            
46419             Roo.menu.MenuMgr.hideAll();
46420
46421         }
46422         
46423         if (!updateFooter) {
46424             //this.footDisp.dom.innerHTML = ''; 
46425             return;
46426         }
46427         // update the footer
46428         //
46429         var html = '';
46430         
46431         this.footerEls = ans.reverse();
46432         Roo.each(this.footerEls, function(a,i) {
46433             if (!a) { return; }
46434             html += html.length ? ' &gt; '  :  '';
46435             
46436             html += '<span class="x-ed-loc-' + i + '">' + a.tagName + '</span>';
46437             
46438         });
46439        
46440         // 
46441         var sz = this.footDisp.up('td').getSize();
46442         this.footDisp.dom.style.width = (sz.width -10) + 'px';
46443         this.footDisp.dom.style.marginLeft = '5px';
46444         
46445         this.footDisp.dom.style.overflow = 'hidden';
46446         
46447         this.footDisp.dom.innerHTML = html;
46448             
46449         //this.editorsyncValue();
46450     },
46451      
46452     
46453    
46454        
46455     // private
46456     onDestroy : function(){
46457         if(this.rendered){
46458             
46459             this.tb.items.each(function(item){
46460                 if(item.menu){
46461                     item.menu.removeAll();
46462                     if(item.menu.el){
46463                         item.menu.el.destroy();
46464                     }
46465                 }
46466                 item.destroy();
46467             });
46468              
46469         }
46470     },
46471     onFirstFocus: function() {
46472         // need to do this for all the toolbars..
46473         this.tb.items.each(function(item){
46474            item.enable();
46475         });
46476     },
46477     buildToolbar: function(tlist, nm)
46478     {
46479         var editor = this.editor;
46480         var editorcore = this.editorcore;
46481          // create a new element.
46482         var wdiv = editor.wrap.createChild({
46483                 tag: 'div'
46484             }, editor.wrap.dom.firstChild.nextSibling, true);
46485         
46486        
46487         var tb = new Roo.Toolbar(wdiv);
46488         // add the name..
46489         
46490         tb.add(nm+ ":&nbsp;");
46491         
46492         var styles = [];
46493         for(var i in this.styles) {
46494             styles.push(i);
46495         }
46496         
46497         // styles...
46498         if (styles && styles.length) {
46499             
46500             // this needs a multi-select checkbox...
46501             tb.addField( new Roo.form.ComboBox({
46502                 store: new Roo.data.SimpleStore({
46503                     id : 'val',
46504                     fields: ['val', 'selected'],
46505                     data : [] 
46506                 }),
46507                 name : '-roo-edit-className',
46508                 attrname : 'className',
46509                 displayField: 'val',
46510                 typeAhead: false,
46511                 mode: 'local',
46512                 editable : false,
46513                 triggerAction: 'all',
46514                 emptyText:'Select Style',
46515                 selectOnFocus:true,
46516                 width: 130,
46517                 listeners : {
46518                     'select': function(c, r, i) {
46519                         // initial support only for on class per el..
46520                         tb.selectedNode.className =  r ? r.get('val') : '';
46521                         editorcore.syncValue();
46522                     }
46523                 }
46524     
46525             }));
46526         }
46527         
46528         var tbc = Roo.form.HtmlEditor.ToolbarContext;
46529         var tbops = tbc.options;
46530         
46531         for (var i in tlist) {
46532             
46533             var item = tlist[i];
46534             tb.add(item.title + ":&nbsp;");
46535             
46536             
46537             //optname == used so you can configure the options available..
46538             var opts = item.opts ? item.opts : false;
46539             if (item.optname) {
46540                 opts = tbops[item.optname];
46541            
46542             }
46543             
46544             if (opts) {
46545                 // opts == pulldown..
46546                 tb.addField( new Roo.form.ComboBox({
46547                     store:   typeof(tbc.stores[i]) != 'undefined' ?  Roo.factory(tbc.stores[i],Roo.data) : new Roo.data.SimpleStore({
46548                         id : 'val',
46549                         fields: ['val', 'display'],
46550                         data : opts  
46551                     }),
46552                     name : '-roo-edit-' + i,
46553                     attrname : i,
46554                     stylename : item.style ? item.style : false,
46555                     displayField: item.displayField ? item.displayField : 'val',
46556                     valueField :  'val',
46557                     typeAhead: false,
46558                     mode: typeof(tbc.stores[i]) != 'undefined'  ? 'remote' : 'local',
46559                     editable : false,
46560                     triggerAction: 'all',
46561                     emptyText:'Select',
46562                     selectOnFocus:true,
46563                     width: item.width ? item.width  : 130,
46564                     listeners : {
46565                         'select': function(c, r, i) {
46566                             if (c.stylename) {
46567                                 tb.selectedNode.style[c.stylename] =  r.get('val');
46568                                 return;
46569                             }
46570                             tb.selectedNode.setAttribute(c.attrname, r.get('val'));
46571                         }
46572                     }
46573
46574                 }));
46575                 continue;
46576                     
46577                  
46578                 
46579                 tb.addField( new Roo.form.TextField({
46580                     name: i,
46581                     width: 100,
46582                     //allowBlank:false,
46583                     value: ''
46584                 }));
46585                 continue;
46586             }
46587             tb.addField( new Roo.form.TextField({
46588                 name: '-roo-edit-' + i,
46589                 attrname : i,
46590                 
46591                 width: item.width,
46592                 //allowBlank:true,
46593                 value: '',
46594                 listeners: {
46595                     'change' : function(f, nv, ov) {
46596                         tb.selectedNode.setAttribute(f.attrname, nv);
46597                         editorcore.syncValue();
46598                     }
46599                 }
46600             }));
46601              
46602         }
46603         
46604         var _this = this;
46605         
46606         if(nm == 'BODY'){
46607             tb.addSeparator();
46608         
46609             tb.addButton( {
46610                 text: 'Stylesheets',
46611
46612                 listeners : {
46613                     click : function ()
46614                     {
46615                         _this.editor.fireEvent('stylesheetsclick', _this.editor);
46616                     }
46617                 }
46618             });
46619         }
46620         
46621         tb.addFill();
46622         tb.addButton( {
46623             text: 'Remove Tag',
46624     
46625             listeners : {
46626                 click : function ()
46627                 {
46628                     // remove
46629                     // undo does not work.
46630                      
46631                     var sn = tb.selectedNode;
46632                     
46633                     var pn = sn.parentNode;
46634                     
46635                     var stn =  sn.childNodes[0];
46636                     var en = sn.childNodes[sn.childNodes.length - 1 ];
46637                     while (sn.childNodes.length) {
46638                         var node = sn.childNodes[0];
46639                         sn.removeChild(node);
46640                         //Roo.log(node);
46641                         pn.insertBefore(node, sn);
46642                         
46643                     }
46644                     pn.removeChild(sn);
46645                     var range = editorcore.createRange();
46646         
46647                     range.setStart(stn,0);
46648                     range.setEnd(en,0); //????
46649                     //range.selectNode(sel);
46650                     
46651                     
46652                     var selection = editorcore.getSelection();
46653                     selection.removeAllRanges();
46654                     selection.addRange(range);
46655                     
46656                     
46657                     
46658                     //_this.updateToolbar(null, null, pn);
46659                     _this.updateToolbar(null, null, null);
46660                     _this.footDisp.dom.innerHTML = ''; 
46661                 }
46662             }
46663             
46664                     
46665                 
46666             
46667         });
46668         
46669         
46670         tb.el.on('click', function(e){
46671             e.preventDefault(); // what does this do?
46672         });
46673         tb.el.setVisibilityMode( Roo.Element.DISPLAY);
46674         tb.el.hide();
46675         tb.name = nm;
46676         // dont need to disable them... as they will get hidden
46677         return tb;
46678          
46679         
46680     },
46681     buildFooter : function()
46682     {
46683         
46684         var fel = this.editor.wrap.createChild();
46685         this.footer = new Roo.Toolbar(fel);
46686         // toolbar has scrolly on left / right?
46687         var footDisp= new Roo.Toolbar.Fill();
46688         var _t = this;
46689         this.footer.add(
46690             {
46691                 text : '&lt;',
46692                 xtype: 'Button',
46693                 handler : function() {
46694                     _t.footDisp.scrollTo('left',0,true)
46695                 }
46696             }
46697         );
46698         this.footer.add( footDisp );
46699         this.footer.add( 
46700             {
46701                 text : '&gt;',
46702                 xtype: 'Button',
46703                 handler : function() {
46704                     // no animation..
46705                     _t.footDisp.select('span').last().scrollIntoView(_t.footDisp,true);
46706                 }
46707             }
46708         );
46709         var fel = Roo.get(footDisp.el);
46710         fel.addClass('x-editor-context');
46711         this.footDispWrap = fel; 
46712         this.footDispWrap.overflow  = 'hidden';
46713         
46714         this.footDisp = fel.createChild();
46715         this.footDispWrap.on('click', this.onContextClick, this)
46716         
46717         
46718     },
46719     onContextClick : function (ev,dom)
46720     {
46721         ev.preventDefault();
46722         var  cn = dom.className;
46723         //Roo.log(cn);
46724         if (!cn.match(/x-ed-loc-/)) {
46725             return;
46726         }
46727         var n = cn.split('-').pop();
46728         var ans = this.footerEls;
46729         var sel = ans[n];
46730         
46731          // pick
46732         var range = this.editorcore.createRange();
46733         
46734         range.selectNodeContents(sel);
46735         //range.selectNode(sel);
46736         
46737         
46738         var selection = this.editorcore.getSelection();
46739         selection.removeAllRanges();
46740         selection.addRange(range);
46741         
46742         
46743         
46744         this.updateToolbar(null, null, sel);
46745         
46746         
46747     }
46748     
46749     
46750     
46751     
46752     
46753 });
46754
46755
46756
46757
46758
46759 /*
46760  * Based on:
46761  * Ext JS Library 1.1.1
46762  * Copyright(c) 2006-2007, Ext JS, LLC.
46763  *
46764  * Originally Released Under LGPL - original licence link has changed is not relivant.
46765  *
46766  * Fork - LGPL
46767  * <script type="text/javascript">
46768  */
46769  
46770 /**
46771  * @class Roo.form.BasicForm
46772  * @extends Roo.util.Observable
46773  * Supplies the functionality to do "actions" on forms and initialize Roo.form.Field types on existing markup.
46774  * @constructor
46775  * @param {String/HTMLElement/Roo.Element} el The form element or its id
46776  * @param {Object} config Configuration options
46777  */
46778 Roo.form.BasicForm = function(el, config){
46779     this.allItems = [];
46780     this.childForms = [];
46781     Roo.apply(this, config);
46782     /*
46783      * The Roo.form.Field items in this form.
46784      * @type MixedCollection
46785      */
46786      
46787      
46788     this.items = new Roo.util.MixedCollection(false, function(o){
46789         return o.id || (o.id = Roo.id());
46790     });
46791     this.addEvents({
46792         /**
46793          * @event beforeaction
46794          * Fires before any action is performed. Return false to cancel the action.
46795          * @param {Form} this
46796          * @param {Action} action The action to be performed
46797          */
46798         beforeaction: true,
46799         /**
46800          * @event actionfailed
46801          * Fires when an action fails.
46802          * @param {Form} this
46803          * @param {Action} action The action that failed
46804          */
46805         actionfailed : true,
46806         /**
46807          * @event actioncomplete
46808          * Fires when an action is completed.
46809          * @param {Form} this
46810          * @param {Action} action The action that completed
46811          */
46812         actioncomplete : true
46813     });
46814     if(el){
46815         this.initEl(el);
46816     }
46817     Roo.form.BasicForm.superclass.constructor.call(this);
46818 };
46819
46820 Roo.extend(Roo.form.BasicForm, Roo.util.Observable, {
46821     /**
46822      * @cfg {String} method
46823      * The request method to use (GET or POST) for form actions if one isn't supplied in the action options.
46824      */
46825     /**
46826      * @cfg {DataReader} reader
46827      * An Roo.data.DataReader (e.g. {@link Roo.data.XmlReader}) to be used to read data when executing "load" actions.
46828      * This is optional as there is built-in support for processing JSON.
46829      */
46830     /**
46831      * @cfg {DataReader} errorReader
46832      * An Roo.data.DataReader (e.g. {@link Roo.data.XmlReader}) to be used to read data when reading validation errors on "submit" actions.
46833      * This is completely optional as there is built-in support for processing JSON.
46834      */
46835     /**
46836      * @cfg {String} url
46837      * The URL to use for form actions if one isn't supplied in the action options.
46838      */
46839     /**
46840      * @cfg {Boolean} fileUpload
46841      * Set to true if this form is a file upload.
46842      */
46843      
46844     /**
46845      * @cfg {Object} baseParams
46846      * Parameters to pass with all requests. e.g. baseParams: {id: '123', foo: 'bar'}.
46847      */
46848      /**
46849      
46850     /**
46851      * @cfg {Number} timeout Timeout for form actions in seconds (default is 30 seconds).
46852      */
46853     timeout: 30,
46854
46855     // private
46856     activeAction : null,
46857
46858     /**
46859      * @cfg {Boolean} trackResetOnLoad If set to true, form.reset() resets to the last loaded
46860      * or setValues() data instead of when the form was first created.
46861      */
46862     trackResetOnLoad : false,
46863     
46864     
46865     /**
46866      * childForms - used for multi-tab forms
46867      * @type {Array}
46868      */
46869     childForms : false,
46870     
46871     /**
46872      * allItems - full list of fields.
46873      * @type {Array}
46874      */
46875     allItems : false,
46876     
46877     /**
46878      * By default wait messages are displayed with Roo.MessageBox.wait. You can target a specific
46879      * element by passing it or its id or mask the form itself by passing in true.
46880      * @type Mixed
46881      */
46882     waitMsgTarget : false,
46883     
46884     /**
46885      * @type Boolean
46886      */
46887     disableMask : false,
46888
46889     // private
46890     initEl : function(el){
46891         this.el = Roo.get(el);
46892         this.id = this.el.id || Roo.id();
46893         this.el.on('submit', this.onSubmit, this);
46894         this.el.addClass('x-form');
46895     },
46896
46897     // private
46898     onSubmit : function(e){
46899         e.stopEvent();
46900     },
46901
46902     /**
46903      * Returns true if client-side validation on the form is successful.
46904      * @return Boolean
46905      */
46906     isValid : function(){
46907         var valid = true;
46908         this.items.each(function(f){
46909            if(!f.validate()){
46910                valid = false;
46911            }
46912         });
46913         return valid;
46914     },
46915
46916     /**
46917      * DEPRICATED Returns true if any fields in this form have changed since their original load. 
46918      * @return Boolean
46919      */
46920     isDirty : function(){
46921         var dirty = false;
46922         this.items.each(function(f){
46923            if(f.isDirty()){
46924                dirty = true;
46925                return false;
46926            }
46927         });
46928         return dirty;
46929     },
46930     
46931     /**
46932      * Returns true if any fields in this form have changed since their original load. (New version)
46933      * @return Boolean
46934      */
46935     
46936     hasChanged : function()
46937     {
46938         var dirty = false;
46939         this.items.each(function(f){
46940            if(f.hasChanged()){
46941                dirty = true;
46942                return false;
46943            }
46944         });
46945         return dirty;
46946         
46947     },
46948     /**
46949      * Resets all hasChanged to 'false' -
46950      * The old 'isDirty' used 'original value..' however this breaks reset() and a few other things.
46951      * So hasChanged storage is only to be used for this purpose
46952      * @return Boolean
46953      */
46954     resetHasChanged : function()
46955     {
46956         this.items.each(function(f){
46957            f.resetHasChanged();
46958         });
46959         
46960     },
46961     
46962     
46963     /**
46964      * Performs a predefined action (submit or load) or custom actions you define on this form.
46965      * @param {String} actionName The name of the action type
46966      * @param {Object} options (optional) The options to pass to the action.  All of the config options listed
46967      * below are supported by both the submit and load actions unless otherwise noted (custom actions could also
46968      * accept other config options):
46969      * <pre>
46970 Property          Type             Description
46971 ----------------  ---------------  ----------------------------------------------------------------------------------
46972 url               String           The url for the action (defaults to the form's url)
46973 method            String           The form method to use (defaults to the form's method, or POST if not defined)
46974 params            String/Object    The params to pass (defaults to the form's baseParams, or none if not defined)
46975 clientValidation  Boolean          Applies to submit only.  Pass true to call form.isValid() prior to posting to
46976                                    validate the form on the client (defaults to false)
46977      * </pre>
46978      * @return {BasicForm} this
46979      */
46980     doAction : function(action, options){
46981         if(typeof action == 'string'){
46982             action = new Roo.form.Action.ACTION_TYPES[action](this, options);
46983         }
46984         if(this.fireEvent('beforeaction', this, action) !== false){
46985             this.beforeAction(action);
46986             action.run.defer(100, action);
46987         }
46988         return this;
46989     },
46990
46991     /**
46992      * Shortcut to do a submit action.
46993      * @param {Object} options The options to pass to the action (see {@link #doAction} for details)
46994      * @return {BasicForm} this
46995      */
46996     submit : function(options){
46997         this.doAction('submit', options);
46998         return this;
46999     },
47000
47001     /**
47002      * Shortcut to do a load action.
47003      * @param {Object} options The options to pass to the action (see {@link #doAction} for details)
47004      * @return {BasicForm} this
47005      */
47006     load : function(options){
47007         this.doAction('load', options);
47008         return this;
47009     },
47010
47011     /**
47012      * Persists the values in this form into the passed Roo.data.Record object in a beginEdit/endEdit block.
47013      * @param {Record} record The record to edit
47014      * @return {BasicForm} this
47015      */
47016     updateRecord : function(record){
47017         record.beginEdit();
47018         var fs = record.fields;
47019         fs.each(function(f){
47020             var field = this.findField(f.name);
47021             if(field){
47022                 record.set(f.name, field.getValue());
47023             }
47024         }, this);
47025         record.endEdit();
47026         return this;
47027     },
47028
47029     /**
47030      * Loads an Roo.data.Record into this form.
47031      * @param {Record} record The record to load
47032      * @return {BasicForm} this
47033      */
47034     loadRecord : function(record){
47035         this.setValues(record.data);
47036         return this;
47037     },
47038
47039     // private
47040     beforeAction : function(action){
47041         var o = action.options;
47042         
47043         if(!this.disableMask) {
47044             if(this.waitMsgTarget === true){
47045                 this.el.mask(o.waitMsg || "Sending", 'x-mask-loading');
47046             }else if(this.waitMsgTarget){
47047                 this.waitMsgTarget = Roo.get(this.waitMsgTarget);
47048                 this.waitMsgTarget.mask(o.waitMsg || "Sending", 'x-mask-loading');
47049             }else {
47050                 Roo.MessageBox.wait(o.waitMsg || "Sending", o.waitTitle || this.waitTitle || 'Please Wait...');
47051             }
47052         }
47053         
47054          
47055     },
47056
47057     // private
47058     afterAction : function(action, success){
47059         this.activeAction = null;
47060         var o = action.options;
47061         
47062         if(!this.disableMask) {
47063             if(this.waitMsgTarget === true){
47064                 this.el.unmask();
47065             }else if(this.waitMsgTarget){
47066                 this.waitMsgTarget.unmask();
47067             }else{
47068                 Roo.MessageBox.updateProgress(1);
47069                 Roo.MessageBox.hide();
47070             }
47071         }
47072         
47073         if(success){
47074             if(o.reset){
47075                 this.reset();
47076             }
47077             Roo.callback(o.success, o.scope, [this, action]);
47078             this.fireEvent('actioncomplete', this, action);
47079             
47080         }else{
47081             
47082             // failure condition..
47083             // we have a scenario where updates need confirming.
47084             // eg. if a locking scenario exists..
47085             // we look for { errors : { needs_confirm : true }} in the response.
47086             if (
47087                 (typeof(action.result) != 'undefined')  &&
47088                 (typeof(action.result.errors) != 'undefined')  &&
47089                 (typeof(action.result.errors.needs_confirm) != 'undefined')
47090            ){
47091                 var _t = this;
47092                 Roo.MessageBox.confirm(
47093                     "Change requires confirmation",
47094                     action.result.errorMsg,
47095                     function(r) {
47096                         if (r != 'yes') {
47097                             return;
47098                         }
47099                         _t.doAction('submit', { params :  { _submit_confirmed : 1 } }  );
47100                     }
47101                     
47102                 );
47103                 
47104                 
47105                 
47106                 return;
47107             }
47108             
47109             Roo.callback(o.failure, o.scope, [this, action]);
47110             // show an error message if no failed handler is set..
47111             if (!this.hasListener('actionfailed')) {
47112                 Roo.MessageBox.alert("Error",
47113                     (typeof(action.result) != 'undefined' && typeof(action.result.errorMsg) != 'undefined') ?
47114                         action.result.errorMsg :
47115                         "Saving Failed, please check your entries or try again"
47116                 );
47117             }
47118             
47119             this.fireEvent('actionfailed', this, action);
47120         }
47121         
47122     },
47123
47124     /**
47125      * Find a Roo.form.Field in this form by id, dataIndex, name or hiddenName
47126      * @param {String} id The value to search for
47127      * @return Field
47128      */
47129     findField : function(id){
47130         var field = this.items.get(id);
47131         if(!field){
47132             this.items.each(function(f){
47133                 if(f.isFormField && (f.dataIndex == id || f.id == id || f.getName() == id)){
47134                     field = f;
47135                     return false;
47136                 }
47137             });
47138         }
47139         return field || null;
47140     },
47141
47142     /**
47143      * Add a secondary form to this one, 
47144      * Used to provide tabbed forms. One form is primary, with hidden values 
47145      * which mirror the elements from the other forms.
47146      * 
47147      * @param {Roo.form.Form} form to add.
47148      * 
47149      */
47150     addForm : function(form)
47151     {
47152        
47153         if (this.childForms.indexOf(form) > -1) {
47154             // already added..
47155             return;
47156         }
47157         this.childForms.push(form);
47158         var n = '';
47159         Roo.each(form.allItems, function (fe) {
47160             
47161             n = typeof(fe.getName) == 'undefined' ? fe.name : fe.getName();
47162             if (this.findField(n)) { // already added..
47163                 return;
47164             }
47165             var add = new Roo.form.Hidden({
47166                 name : n
47167             });
47168             add.render(this.el);
47169             
47170             this.add( add );
47171         }, this);
47172         
47173     },
47174     /**
47175      * Mark fields in this form invalid in bulk.
47176      * @param {Array/Object} errors Either an array in the form [{id:'fieldId', msg:'The message'},...] or an object hash of {id: msg, id2: msg2}
47177      * @return {BasicForm} this
47178      */
47179     markInvalid : function(errors){
47180         if(errors instanceof Array){
47181             for(var i = 0, len = errors.length; i < len; i++){
47182                 var fieldError = errors[i];
47183                 var f = this.findField(fieldError.id);
47184                 if(f){
47185                     f.markInvalid(fieldError.msg);
47186                 }
47187             }
47188         }else{
47189             var field, id;
47190             for(id in errors){
47191                 if(typeof errors[id] != 'function' && (field = this.findField(id))){
47192                     field.markInvalid(errors[id]);
47193                 }
47194             }
47195         }
47196         Roo.each(this.childForms || [], function (f) {
47197             f.markInvalid(errors);
47198         });
47199         
47200         return this;
47201     },
47202
47203     /**
47204      * Set values for fields in this form in bulk.
47205      * @param {Array/Object} values Either an array in the form [{id:'fieldId', value:'foo'},...] or an object hash of {id: value, id2: value2}
47206      * @return {BasicForm} this
47207      */
47208     setValues : function(values){
47209         if(values instanceof Array){ // array of objects
47210             for(var i = 0, len = values.length; i < len; i++){
47211                 var v = values[i];
47212                 var f = this.findField(v.id);
47213                 if(f){
47214                     f.setValue(v.value);
47215                     if(this.trackResetOnLoad){
47216                         f.originalValue = f.getValue();
47217                     }
47218                 }
47219             }
47220         }else{ // object hash
47221             var field, id;
47222             for(id in values){
47223                 if(typeof values[id] != 'function' && (field = this.findField(id))){
47224                     
47225                     if (field.setFromData && 
47226                         field.valueField && 
47227                         field.displayField &&
47228                         // combos' with local stores can 
47229                         // be queried via setValue()
47230                         // to set their value..
47231                         (field.store && !field.store.isLocal)
47232                         ) {
47233                         // it's a combo
47234                         var sd = { };
47235                         sd[field.valueField] = typeof(values[field.hiddenName]) == 'undefined' ? '' : values[field.hiddenName];
47236                         sd[field.displayField] = typeof(values[field.name]) == 'undefined' ? '' : values[field.name];
47237                         field.setFromData(sd);
47238                         
47239                     } else {
47240                         field.setValue(values[id]);
47241                     }
47242                     
47243                     
47244                     if(this.trackResetOnLoad){
47245                         field.originalValue = field.getValue();
47246                     }
47247                 }
47248             }
47249         }
47250         this.resetHasChanged();
47251         
47252         
47253         Roo.each(this.childForms || [], function (f) {
47254             f.setValues(values);
47255             f.resetHasChanged();
47256         });
47257                 
47258         return this;
47259     },
47260
47261     /**
47262      * Returns the fields in this form as an object with key/value pairs. If multiple fields exist with the same name
47263      * they are returned as an array.
47264      * @param {Boolean} asString
47265      * @return {Object}
47266      */
47267     getValues : function(asString){
47268         if (this.childForms) {
47269             // copy values from the child forms
47270             Roo.each(this.childForms, function (f) {
47271                 this.setValues(f.getValues());
47272             }, this);
47273         }
47274         
47275         
47276         
47277         var fs = Roo.lib.Ajax.serializeForm(this.el.dom);
47278         if(asString === true){
47279             return fs;
47280         }
47281         return Roo.urlDecode(fs);
47282     },
47283     
47284     /**
47285      * Returns the fields in this form as an object with key/value pairs. 
47286      * This differs from getValues as it calls getValue on each child item, rather than using dom data.
47287      * @return {Object}
47288      */
47289     getFieldValues : function(with_hidden)
47290     {
47291         if (this.childForms) {
47292             // copy values from the child forms
47293             // should this call getFieldValues - probably not as we do not currently copy
47294             // hidden fields when we generate..
47295             Roo.each(this.childForms, function (f) {
47296                 this.setValues(f.getValues());
47297             }, this);
47298         }
47299         
47300         var ret = {};
47301         this.items.each(function(f){
47302             if (!f.getName()) {
47303                 return;
47304             }
47305             var v = f.getValue();
47306             if (f.inputType =='radio') {
47307                 if (typeof(ret[f.getName()]) == 'undefined') {
47308                     ret[f.getName()] = ''; // empty..
47309                 }
47310                 
47311                 if (!f.el.dom.checked) {
47312                     return;
47313                     
47314                 }
47315                 v = f.el.dom.value;
47316                 
47317             }
47318             
47319             // not sure if this supported any more..
47320             if ((typeof(v) == 'object') && f.getRawValue) {
47321                 v = f.getRawValue() ; // dates..
47322             }
47323             // combo boxes where name != hiddenName...
47324             if (f.name != f.getName()) {
47325                 ret[f.name] = f.getRawValue();
47326             }
47327             ret[f.getName()] = v;
47328         });
47329         
47330         return ret;
47331     },
47332
47333     /**
47334      * Clears all invalid messages in this form.
47335      * @return {BasicForm} this
47336      */
47337     clearInvalid : function(){
47338         this.items.each(function(f){
47339            f.clearInvalid();
47340         });
47341         
47342         Roo.each(this.childForms || [], function (f) {
47343             f.clearInvalid();
47344         });
47345         
47346         
47347         return this;
47348     },
47349
47350     /**
47351      * Resets this form.
47352      * @return {BasicForm} this
47353      */
47354     reset : function(){
47355         this.items.each(function(f){
47356             f.reset();
47357         });
47358         
47359         Roo.each(this.childForms || [], function (f) {
47360             f.reset();
47361         });
47362         this.resetHasChanged();
47363         
47364         return this;
47365     },
47366
47367     /**
47368      * Add Roo.form components to this form.
47369      * @param {Field} field1
47370      * @param {Field} field2 (optional)
47371      * @param {Field} etc (optional)
47372      * @return {BasicForm} this
47373      */
47374     add : function(){
47375         this.items.addAll(Array.prototype.slice.call(arguments, 0));
47376         return this;
47377     },
47378
47379
47380     /**
47381      * Removes a field from the items collection (does NOT remove its markup).
47382      * @param {Field} field
47383      * @return {BasicForm} this
47384      */
47385     remove : function(field){
47386         this.items.remove(field);
47387         return this;
47388     },
47389
47390     /**
47391      * Looks at the fields in this form, checks them for an id attribute,
47392      * and calls applyTo on the existing dom element with that id.
47393      * @return {BasicForm} this
47394      */
47395     render : function(){
47396         this.items.each(function(f){
47397             if(f.isFormField && !f.rendered && document.getElementById(f.id)){ // if the element exists
47398                 f.applyTo(f.id);
47399             }
47400         });
47401         return this;
47402     },
47403
47404     /**
47405      * Calls {@link Ext#apply} for all fields in this form with the passed object.
47406      * @param {Object} values
47407      * @return {BasicForm} this
47408      */
47409     applyToFields : function(o){
47410         this.items.each(function(f){
47411            Roo.apply(f, o);
47412         });
47413         return this;
47414     },
47415
47416     /**
47417      * Calls {@link Ext#applyIf} for all field in this form with the passed object.
47418      * @param {Object} values
47419      * @return {BasicForm} this
47420      */
47421     applyIfToFields : function(o){
47422         this.items.each(function(f){
47423            Roo.applyIf(f, o);
47424         });
47425         return this;
47426     }
47427 });
47428
47429 // back compat
47430 Roo.BasicForm = Roo.form.BasicForm;/*
47431  * Based on:
47432  * Ext JS Library 1.1.1
47433  * Copyright(c) 2006-2007, Ext JS, LLC.
47434  *
47435  * Originally Released Under LGPL - original licence link has changed is not relivant.
47436  *
47437  * Fork - LGPL
47438  * <script type="text/javascript">
47439  */
47440
47441 /**
47442  * @class Roo.form.Form
47443  * @extends Roo.form.BasicForm
47444  * Adds the ability to dynamically render forms with JavaScript to {@link Roo.form.BasicForm}.
47445  * @constructor
47446  * @param {Object} config Configuration options
47447  */
47448 Roo.form.Form = function(config){
47449     var xitems =  [];
47450     if (config.items) {
47451         xitems = config.items;
47452         delete config.items;
47453     }
47454    
47455     
47456     Roo.form.Form.superclass.constructor.call(this, null, config);
47457     this.url = this.url || this.action;
47458     if(!this.root){
47459         this.root = new Roo.form.Layout(Roo.applyIf({
47460             id: Roo.id()
47461         }, config));
47462     }
47463     this.active = this.root;
47464     /**
47465      * Array of all the buttons that have been added to this form via {@link addButton}
47466      * @type Array
47467      */
47468     this.buttons = [];
47469     this.allItems = [];
47470     this.addEvents({
47471         /**
47472          * @event clientvalidation
47473          * If the monitorValid config option is true, this event fires repetitively to notify of valid state
47474          * @param {Form} this
47475          * @param {Boolean} valid true if the form has passed client-side validation
47476          */
47477         clientvalidation: true,
47478         /**
47479          * @event rendered
47480          * Fires when the form is rendered
47481          * @param {Roo.form.Form} form
47482          */
47483         rendered : true
47484     });
47485     
47486     if (this.progressUrl) {
47487             // push a hidden field onto the list of fields..
47488             this.addxtype( {
47489                     xns: Roo.form, 
47490                     xtype : 'Hidden', 
47491                     name : 'UPLOAD_IDENTIFIER' 
47492             });
47493         }
47494         
47495     
47496     Roo.each(xitems, this.addxtype, this);
47497     
47498     
47499     
47500 };
47501
47502 Roo.extend(Roo.form.Form, Roo.form.BasicForm, {
47503     /**
47504      * @cfg {Number} labelWidth The width of labels. This property cascades to child containers.
47505      */
47506     /**
47507      * @cfg {String} itemCls A css class to apply to the x-form-item of fields. This property cascades to child containers.
47508      */
47509     /**
47510      * @cfg {String} buttonAlign Valid values are "left," "center" and "right" (defaults to "center")
47511      */
47512     buttonAlign:'center',
47513
47514     /**
47515      * @cfg {Number} minButtonWidth Minimum width of all buttons in pixels (defaults to 75)
47516      */
47517     minButtonWidth:75,
47518
47519     /**
47520      * @cfg {String} labelAlign Valid values are "left," "top" and "right" (defaults to "left").
47521      * This property cascades to child containers if not set.
47522      */
47523     labelAlign:'left',
47524
47525     /**
47526      * @cfg {Boolean} monitorValid If true the form monitors its valid state <b>client-side</b> and
47527      * fires a looping event with that state. This is required to bind buttons to the valid
47528      * state using the config value formBind:true on the button.
47529      */
47530     monitorValid : false,
47531
47532     /**
47533      * @cfg {Number} monitorPoll The milliseconds to poll valid state, ignored if monitorValid is not true (defaults to 200)
47534      */
47535     monitorPoll : 200,
47536     
47537     /**
47538      * @cfg {String} progressUrl - Url to return progress data 
47539      */
47540     
47541     progressUrl : false,
47542     /**
47543      * @cfg {boolean|FormData} formData - true to use new 'FormData' post, or set to a new FormData({dom form}) Object, if
47544      * sending a formdata with extra parameters - eg uploaded elements.
47545      */
47546     
47547     formData : false,
47548     
47549     /**
47550      * Opens a new {@link Roo.form.Column} container in the layout stack. If fields are passed after the config, the
47551      * fields are added and the column is closed. If no fields are passed the column remains open
47552      * until end() is called.
47553      * @param {Object} config The config to pass to the column
47554      * @param {Field} field1 (optional)
47555      * @param {Field} field2 (optional)
47556      * @param {Field} etc (optional)
47557      * @return Column The column container object
47558      */
47559     column : function(c){
47560         var col = new Roo.form.Column(c);
47561         this.start(col);
47562         if(arguments.length > 1){ // duplicate code required because of Opera
47563             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
47564             this.end();
47565         }
47566         return col;
47567     },
47568
47569     /**
47570      * Opens a new {@link Roo.form.FieldSet} container in the layout stack. If fields are passed after the config, the
47571      * fields are added and the fieldset is closed. If no fields are passed the fieldset remains open
47572      * until end() is called.
47573      * @param {Object} config The config to pass to the fieldset
47574      * @param {Field} field1 (optional)
47575      * @param {Field} field2 (optional)
47576      * @param {Field} etc (optional)
47577      * @return FieldSet The fieldset container object
47578      */
47579     fieldset : function(c){
47580         var fs = new Roo.form.FieldSet(c);
47581         this.start(fs);
47582         if(arguments.length > 1){ // duplicate code required because of Opera
47583             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
47584             this.end();
47585         }
47586         return fs;
47587     },
47588
47589     /**
47590      * Opens a new {@link Roo.form.Layout} container in the layout stack. If fields are passed after the config, the
47591      * fields are added and the container is closed. If no fields are passed the container remains open
47592      * until end() is called.
47593      * @param {Object} config The config to pass to the Layout
47594      * @param {Field} field1 (optional)
47595      * @param {Field} field2 (optional)
47596      * @param {Field} etc (optional)
47597      * @return Layout The container object
47598      */
47599     container : function(c){
47600         var l = new Roo.form.Layout(c);
47601         this.start(l);
47602         if(arguments.length > 1){ // duplicate code required because of Opera
47603             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
47604             this.end();
47605         }
47606         return l;
47607     },
47608
47609     /**
47610      * Opens the passed container in the layout stack. The container can be any {@link Roo.form.Layout} or subclass.
47611      * @param {Object} container A Roo.form.Layout or subclass of Layout
47612      * @return {Form} this
47613      */
47614     start : function(c){
47615         // cascade label info
47616         Roo.applyIf(c, {'labelAlign': this.active.labelAlign, 'labelWidth': this.active.labelWidth, 'itemCls': this.active.itemCls});
47617         this.active.stack.push(c);
47618         c.ownerCt = this.active;
47619         this.active = c;
47620         return this;
47621     },
47622
47623     /**
47624      * Closes the current open container
47625      * @return {Form} this
47626      */
47627     end : function(){
47628         if(this.active == this.root){
47629             return this;
47630         }
47631         this.active = this.active.ownerCt;
47632         return this;
47633     },
47634
47635     /**
47636      * Add Roo.form components to the current open container (e.g. column, fieldset, etc.).  Fields added via this method
47637      * can also be passed with an additional property of fieldLabel, which if supplied, will provide the text to display
47638      * as the label of the field.
47639      * @param {Field} field1
47640      * @param {Field} field2 (optional)
47641      * @param {Field} etc. (optional)
47642      * @return {Form} this
47643      */
47644     add : function(){
47645         this.active.stack.push.apply(this.active.stack, arguments);
47646         this.allItems.push.apply(this.allItems,arguments);
47647         var r = [];
47648         for(var i = 0, a = arguments, len = a.length; i < len; i++) {
47649             if(a[i].isFormField){
47650                 r.push(a[i]);
47651             }
47652         }
47653         if(r.length > 0){
47654             Roo.form.Form.superclass.add.apply(this, r);
47655         }
47656         return this;
47657     },
47658     
47659
47660     
47661     
47662     
47663      /**
47664      * Find any element that has been added to a form, using it's ID or name
47665      * This can include framesets, columns etc. along with regular fields..
47666      * @param {String} id - id or name to find.
47667      
47668      * @return {Element} e - or false if nothing found.
47669      */
47670     findbyId : function(id)
47671     {
47672         var ret = false;
47673         if (!id) {
47674             return ret;
47675         }
47676         Roo.each(this.allItems, function(f){
47677             if (f.id == id || f.name == id ){
47678                 ret = f;
47679                 return false;
47680             }
47681         });
47682         return ret;
47683     },
47684
47685     
47686     
47687     /**
47688      * Render this form into the passed container. This should only be called once!
47689      * @param {String/HTMLElement/Element} container The element this component should be rendered into
47690      * @return {Form} this
47691      */
47692     render : function(ct)
47693     {
47694         
47695         
47696         
47697         ct = Roo.get(ct);
47698         var o = this.autoCreate || {
47699             tag: 'form',
47700             method : this.method || 'POST',
47701             id : this.id || Roo.id()
47702         };
47703         this.initEl(ct.createChild(o));
47704
47705         this.root.render(this.el);
47706         
47707        
47708              
47709         this.items.each(function(f){
47710             f.render('x-form-el-'+f.id);
47711         });
47712
47713         if(this.buttons.length > 0){
47714             // tables are required to maintain order and for correct IE layout
47715             var tb = this.el.createChild({cls:'x-form-btns-ct', cn: {
47716                 cls:"x-form-btns x-form-btns-"+this.buttonAlign,
47717                 html:'<table cellspacing="0"><tbody><tr></tr></tbody></table><div class="x-clear"></div>'
47718             }}, null, true);
47719             var tr = tb.getElementsByTagName('tr')[0];
47720             for(var i = 0, len = this.buttons.length; i < len; i++) {
47721                 var b = this.buttons[i];
47722                 var td = document.createElement('td');
47723                 td.className = 'x-form-btn-td';
47724                 b.render(tr.appendChild(td));
47725             }
47726         }
47727         if(this.monitorValid){ // initialize after render
47728             this.startMonitoring();
47729         }
47730         this.fireEvent('rendered', this);
47731         return this;
47732     },
47733
47734     /**
47735      * Adds a button to the footer of the form - this <b>must</b> be called before the form is rendered.
47736      * @param {String/Object} config A string becomes the button text, an object can either be a Button config
47737      * object or a valid Roo.DomHelper element config
47738      * @param {Function} handler The function called when the button is clicked
47739      * @param {Object} scope (optional) The scope of the handler function
47740      * @return {Roo.Button}
47741      */
47742     addButton : function(config, handler, scope){
47743         var bc = {
47744             handler: handler,
47745             scope: scope,
47746             minWidth: this.minButtonWidth,
47747             hideParent:true
47748         };
47749         if(typeof config == "string"){
47750             bc.text = config;
47751         }else{
47752             Roo.apply(bc, config);
47753         }
47754         var btn = new Roo.Button(null, bc);
47755         this.buttons.push(btn);
47756         return btn;
47757     },
47758
47759      /**
47760      * Adds a series of form elements (using the xtype property as the factory method.
47761      * Valid xtypes are:  TextField, TextArea .... Button, Layout, FieldSet, Column, (and 'end' to close a block)
47762      * @param {Object} config 
47763      */
47764     
47765     addxtype : function()
47766     {
47767         var ar = Array.prototype.slice.call(arguments, 0);
47768         var ret = false;
47769         for(var i = 0; i < ar.length; i++) {
47770             if (!ar[i]) {
47771                 continue; // skip -- if this happends something invalid got sent, we 
47772                 // should ignore it, as basically that interface element will not show up
47773                 // and that should be pretty obvious!!
47774             }
47775             
47776             if (Roo.form[ar[i].xtype]) {
47777                 ar[i].form = this;
47778                 var fe = Roo.factory(ar[i], Roo.form);
47779                 if (!ret) {
47780                     ret = fe;
47781                 }
47782                 fe.form = this;
47783                 if (fe.store) {
47784                     fe.store.form = this;
47785                 }
47786                 if (fe.isLayout) {  
47787                          
47788                     this.start(fe);
47789                     this.allItems.push(fe);
47790                     if (fe.items && fe.addxtype) {
47791                         fe.addxtype.apply(fe, fe.items);
47792                         delete fe.items;
47793                     }
47794                      this.end();
47795                     continue;
47796                 }
47797                 
47798                 
47799                  
47800                 this.add(fe);
47801               //  console.log('adding ' + ar[i].xtype);
47802             }
47803             if (ar[i].xtype == 'Button') {  
47804                 //console.log('adding button');
47805                 //console.log(ar[i]);
47806                 this.addButton(ar[i]);
47807                 this.allItems.push(fe);
47808                 continue;
47809             }
47810             
47811             if (ar[i].xtype == 'end') { // so we can add fieldsets... / layout etc.
47812                 alert('end is not supported on xtype any more, use items');
47813             //    this.end();
47814             //    //console.log('adding end');
47815             }
47816             
47817         }
47818         return ret;
47819     },
47820     
47821     /**
47822      * Starts monitoring of the valid state of this form. Usually this is done by passing the config
47823      * option "monitorValid"
47824      */
47825     startMonitoring : function(){
47826         if(!this.bound){
47827             this.bound = true;
47828             Roo.TaskMgr.start({
47829                 run : this.bindHandler,
47830                 interval : this.monitorPoll || 200,
47831                 scope: this
47832             });
47833         }
47834     },
47835
47836     /**
47837      * Stops monitoring of the valid state of this form
47838      */
47839     stopMonitoring : function(){
47840         this.bound = false;
47841     },
47842
47843     // private
47844     bindHandler : function(){
47845         if(!this.bound){
47846             return false; // stops binding
47847         }
47848         var valid = true;
47849         this.items.each(function(f){
47850             if(!f.isValid(true)){
47851                 valid = false;
47852                 return false;
47853             }
47854         });
47855         for(var i = 0, len = this.buttons.length; i < len; i++){
47856             var btn = this.buttons[i];
47857             if(btn.formBind === true && btn.disabled === valid){
47858                 btn.setDisabled(!valid);
47859             }
47860         }
47861         this.fireEvent('clientvalidation', this, valid);
47862     }
47863     
47864     
47865     
47866     
47867     
47868     
47869     
47870     
47871 });
47872
47873
47874 // back compat
47875 Roo.Form = Roo.form.Form;
47876 /*
47877  * Based on:
47878  * Ext JS Library 1.1.1
47879  * Copyright(c) 2006-2007, Ext JS, LLC.
47880  *
47881  * Originally Released Under LGPL - original licence link has changed is not relivant.
47882  *
47883  * Fork - LGPL
47884  * <script type="text/javascript">
47885  */
47886
47887 // as we use this in bootstrap.
47888 Roo.namespace('Roo.form');
47889  /**
47890  * @class Roo.form.Action
47891  * Internal Class used to handle form actions
47892  * @constructor
47893  * @param {Roo.form.BasicForm} el The form element or its id
47894  * @param {Object} config Configuration options
47895  */
47896
47897  
47898  
47899 // define the action interface
47900 Roo.form.Action = function(form, options){
47901     this.form = form;
47902     this.options = options || {};
47903 };
47904 /**
47905  * Client Validation Failed
47906  * @const 
47907  */
47908 Roo.form.Action.CLIENT_INVALID = 'client';
47909 /**
47910  * Server Validation Failed
47911  * @const 
47912  */
47913 Roo.form.Action.SERVER_INVALID = 'server';
47914  /**
47915  * Connect to Server Failed
47916  * @const 
47917  */
47918 Roo.form.Action.CONNECT_FAILURE = 'connect';
47919 /**
47920  * Reading Data from Server Failed
47921  * @const 
47922  */
47923 Roo.form.Action.LOAD_FAILURE = 'load';
47924
47925 Roo.form.Action.prototype = {
47926     type : 'default',
47927     failureType : undefined,
47928     response : undefined,
47929     result : undefined,
47930
47931     // interface method
47932     run : function(options){
47933
47934     },
47935
47936     // interface method
47937     success : function(response){
47938
47939     },
47940
47941     // interface method
47942     handleResponse : function(response){
47943
47944     },
47945
47946     // default connection failure
47947     failure : function(response){
47948         
47949         this.response = response;
47950         this.failureType = Roo.form.Action.CONNECT_FAILURE;
47951         this.form.afterAction(this, false);
47952     },
47953
47954     processResponse : function(response){
47955         this.response = response;
47956         if(!response.responseText){
47957             return true;
47958         }
47959         this.result = this.handleResponse(response);
47960         return this.result;
47961     },
47962
47963     // utility functions used internally
47964     getUrl : function(appendParams){
47965         var url = this.options.url || this.form.url || this.form.el.dom.action;
47966         if(appendParams){
47967             var p = this.getParams();
47968             if(p){
47969                 url += (url.indexOf('?') != -1 ? '&' : '?') + p;
47970             }
47971         }
47972         return url;
47973     },
47974
47975     getMethod : function(){
47976         return (this.options.method || this.form.method || this.form.el.dom.method || 'POST').toUpperCase();
47977     },
47978
47979     getParams : function(){
47980         var bp = this.form.baseParams;
47981         var p = this.options.params;
47982         if(p){
47983             if(typeof p == "object"){
47984                 p = Roo.urlEncode(Roo.applyIf(p, bp));
47985             }else if(typeof p == 'string' && bp){
47986                 p += '&' + Roo.urlEncode(bp);
47987             }
47988         }else if(bp){
47989             p = Roo.urlEncode(bp);
47990         }
47991         return p;
47992     },
47993
47994     createCallback : function(){
47995         return {
47996             success: this.success,
47997             failure: this.failure,
47998             scope: this,
47999             timeout: (this.form.timeout*1000),
48000             upload: this.form.fileUpload ? this.success : undefined
48001         };
48002     }
48003 };
48004
48005 Roo.form.Action.Submit = function(form, options){
48006     Roo.form.Action.Submit.superclass.constructor.call(this, form, options);
48007 };
48008
48009 Roo.extend(Roo.form.Action.Submit, Roo.form.Action, {
48010     type : 'submit',
48011
48012     haveProgress : false,
48013     uploadComplete : false,
48014     
48015     // uploadProgress indicator.
48016     uploadProgress : function()
48017     {
48018         if (!this.form.progressUrl) {
48019             return;
48020         }
48021         
48022         if (!this.haveProgress) {
48023             Roo.MessageBox.progress("Uploading", "Uploading");
48024         }
48025         if (this.uploadComplete) {
48026            Roo.MessageBox.hide();
48027            return;
48028         }
48029         
48030         this.haveProgress = true;
48031    
48032         var uid = this.form.findField('UPLOAD_IDENTIFIER').getValue();
48033         
48034         var c = new Roo.data.Connection();
48035         c.request({
48036             url : this.form.progressUrl,
48037             params: {
48038                 id : uid
48039             },
48040             method: 'GET',
48041             success : function(req){
48042                //console.log(data);
48043                 var rdata = false;
48044                 var edata;
48045                 try  {
48046                    rdata = Roo.decode(req.responseText)
48047                 } catch (e) {
48048                     Roo.log("Invalid data from server..");
48049                     Roo.log(edata);
48050                     return;
48051                 }
48052                 if (!rdata || !rdata.success) {
48053                     Roo.log(rdata);
48054                     Roo.MessageBox.alert(Roo.encode(rdata));
48055                     return;
48056                 }
48057                 var data = rdata.data;
48058                 
48059                 if (this.uploadComplete) {
48060                    Roo.MessageBox.hide();
48061                    return;
48062                 }
48063                    
48064                 if (data){
48065                     Roo.MessageBox.updateProgress(data.bytes_uploaded/data.bytes_total,
48066                        Math.floor((data.bytes_total - data.bytes_uploaded)/1000) + 'k remaining'
48067                     );
48068                 }
48069                 this.uploadProgress.defer(2000,this);
48070             },
48071        
48072             failure: function(data) {
48073                 Roo.log('progress url failed ');
48074                 Roo.log(data);
48075             },
48076             scope : this
48077         });
48078            
48079     },
48080     
48081     
48082     run : function()
48083     {
48084         // run get Values on the form, so it syncs any secondary forms.
48085         this.form.getValues();
48086         
48087         var o = this.options;
48088         var method = this.getMethod();
48089         var isPost = method == 'POST';
48090         if(o.clientValidation === false || this.form.isValid()){
48091             
48092             if (this.form.progressUrl) {
48093                 this.form.findField('UPLOAD_IDENTIFIER').setValue(
48094                     (new Date() * 1) + '' + Math.random());
48095                     
48096             } 
48097             
48098             
48099             Roo.Ajax.request(Roo.apply(this.createCallback(), {
48100                 form:this.form.el.dom,
48101                 url:this.getUrl(!isPost),
48102                 method: method,
48103                 params:isPost ? this.getParams() : null,
48104                 isUpload: this.form.fileUpload,
48105                 formData : this.form.formData
48106             }));
48107             
48108             this.uploadProgress();
48109
48110         }else if (o.clientValidation !== false){ // client validation failed
48111             this.failureType = Roo.form.Action.CLIENT_INVALID;
48112             this.form.afterAction(this, false);
48113         }
48114     },
48115
48116     success : function(response)
48117     {
48118         this.uploadComplete= true;
48119         if (this.haveProgress) {
48120             Roo.MessageBox.hide();
48121         }
48122         
48123         
48124         var result = this.processResponse(response);
48125         if(result === true || result.success){
48126             this.form.afterAction(this, true);
48127             return;
48128         }
48129         if(result.errors){
48130             this.form.markInvalid(result.errors);
48131             this.failureType = Roo.form.Action.SERVER_INVALID;
48132         }
48133         this.form.afterAction(this, false);
48134     },
48135     failure : function(response)
48136     {
48137         this.uploadComplete= true;
48138         if (this.haveProgress) {
48139             Roo.MessageBox.hide();
48140         }
48141         
48142         this.response = response;
48143         this.failureType = Roo.form.Action.CONNECT_FAILURE;
48144         this.form.afterAction(this, false);
48145     },
48146     
48147     handleResponse : function(response){
48148         if(this.form.errorReader){
48149             var rs = this.form.errorReader.read(response);
48150             var errors = [];
48151             if(rs.records){
48152                 for(var i = 0, len = rs.records.length; i < len; i++) {
48153                     var r = rs.records[i];
48154                     errors[i] = r.data;
48155                 }
48156             }
48157             if(errors.length < 1){
48158                 errors = null;
48159             }
48160             return {
48161                 success : rs.success,
48162                 errors : errors
48163             };
48164         }
48165         var ret = false;
48166         try {
48167             ret = Roo.decode(response.responseText);
48168         } catch (e) {
48169             ret = {
48170                 success: false,
48171                 errorMsg: "Failed to read server message: " + (response ? response.responseText : ' - no message'),
48172                 errors : []
48173             };
48174         }
48175         return ret;
48176         
48177     }
48178 });
48179
48180
48181 Roo.form.Action.Load = function(form, options){
48182     Roo.form.Action.Load.superclass.constructor.call(this, form, options);
48183     this.reader = this.form.reader;
48184 };
48185
48186 Roo.extend(Roo.form.Action.Load, Roo.form.Action, {
48187     type : 'load',
48188
48189     run : function(){
48190         
48191         Roo.Ajax.request(Roo.apply(
48192                 this.createCallback(), {
48193                     method:this.getMethod(),
48194                     url:this.getUrl(false),
48195                     params:this.getParams()
48196         }));
48197     },
48198
48199     success : function(response){
48200         
48201         var result = this.processResponse(response);
48202         if(result === true || !result.success || !result.data){
48203             this.failureType = Roo.form.Action.LOAD_FAILURE;
48204             this.form.afterAction(this, false);
48205             return;
48206         }
48207         this.form.clearInvalid();
48208         this.form.setValues(result.data);
48209         this.form.afterAction(this, true);
48210     },
48211
48212     handleResponse : function(response){
48213         if(this.form.reader){
48214             var rs = this.form.reader.read(response);
48215             var data = rs.records && rs.records[0] ? rs.records[0].data : null;
48216             return {
48217                 success : rs.success,
48218                 data : data
48219             };
48220         }
48221         return Roo.decode(response.responseText);
48222     }
48223 });
48224
48225 Roo.form.Action.ACTION_TYPES = {
48226     'load' : Roo.form.Action.Load,
48227     'submit' : Roo.form.Action.Submit
48228 };/*
48229  * Based on:
48230  * Ext JS Library 1.1.1
48231  * Copyright(c) 2006-2007, Ext JS, LLC.
48232  *
48233  * Originally Released Under LGPL - original licence link has changed is not relivant.
48234  *
48235  * Fork - LGPL
48236  * <script type="text/javascript">
48237  */
48238  
48239 /**
48240  * @class Roo.form.Layout
48241  * @extends Roo.Component
48242  * Creates a container for layout and rendering of fields in an {@link Roo.form.Form}.
48243  * @constructor
48244  * @param {Object} config Configuration options
48245  */
48246 Roo.form.Layout = function(config){
48247     var xitems = [];
48248     if (config.items) {
48249         xitems = config.items;
48250         delete config.items;
48251     }
48252     Roo.form.Layout.superclass.constructor.call(this, config);
48253     this.stack = [];
48254     Roo.each(xitems, this.addxtype, this);
48255      
48256 };
48257
48258 Roo.extend(Roo.form.Layout, Roo.Component, {
48259     /**
48260      * @cfg {String/Object} autoCreate
48261      * A DomHelper element spec used to autocreate the layout (defaults to {tag: 'div', cls: 'x-form-ct'})
48262      */
48263     /**
48264      * @cfg {String/Object/Function} style
48265      * A style specification string, e.g. "width:100px", or object in the form {width:"100px"}, or
48266      * a function which returns such a specification.
48267      */
48268     /**
48269      * @cfg {String} labelAlign
48270      * Valid values are "left," "top" and "right" (defaults to "left")
48271      */
48272     /**
48273      * @cfg {Number} labelWidth
48274      * Fixed width in pixels of all field labels (defaults to undefined)
48275      */
48276     /**
48277      * @cfg {Boolean} clear
48278      * True to add a clearing element at the end of this layout, equivalent to CSS clear: both (defaults to true)
48279      */
48280     clear : true,
48281     /**
48282      * @cfg {String} labelSeparator
48283      * The separator to use after field labels (defaults to ':')
48284      */
48285     labelSeparator : ':',
48286     /**
48287      * @cfg {Boolean} hideLabels
48288      * True to suppress the display of field labels in this layout (defaults to false)
48289      */
48290     hideLabels : false,
48291
48292     // private
48293     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct'},
48294     
48295     isLayout : true,
48296     
48297     // private
48298     onRender : function(ct, position){
48299         if(this.el){ // from markup
48300             this.el = Roo.get(this.el);
48301         }else {  // generate
48302             var cfg = this.getAutoCreate();
48303             this.el = ct.createChild(cfg, position);
48304         }
48305         if(this.style){
48306             this.el.applyStyles(this.style);
48307         }
48308         if(this.labelAlign){
48309             this.el.addClass('x-form-label-'+this.labelAlign);
48310         }
48311         if(this.hideLabels){
48312             this.labelStyle = "display:none";
48313             this.elementStyle = "padding-left:0;";
48314         }else{
48315             if(typeof this.labelWidth == 'number'){
48316                 this.labelStyle = "width:"+this.labelWidth+"px;";
48317                 this.elementStyle = "padding-left:"+((this.labelWidth+(typeof this.labelPad == 'number' ? this.labelPad : 5))+'px')+";";
48318             }
48319             if(this.labelAlign == 'top'){
48320                 this.labelStyle = "width:auto;";
48321                 this.elementStyle = "padding-left:0;";
48322             }
48323         }
48324         var stack = this.stack;
48325         var slen = stack.length;
48326         if(slen > 0){
48327             if(!this.fieldTpl){
48328                 var t = new Roo.Template(
48329                     '<div class="x-form-item {5}">',
48330                         '<label for="{0}" style="{2}">{1}{4}</label>',
48331                         '<div class="x-form-element" id="x-form-el-{0}" style="{3}">',
48332                         '</div>',
48333                     '</div><div class="x-form-clear-left"></div>'
48334                 );
48335                 t.disableFormats = true;
48336                 t.compile();
48337                 Roo.form.Layout.prototype.fieldTpl = t;
48338             }
48339             for(var i = 0; i < slen; i++) {
48340                 if(stack[i].isFormField){
48341                     this.renderField(stack[i]);
48342                 }else{
48343                     this.renderComponent(stack[i]);
48344                 }
48345             }
48346         }
48347         if(this.clear){
48348             this.el.createChild({cls:'x-form-clear'});
48349         }
48350     },
48351
48352     // private
48353     renderField : function(f){
48354         f.fieldEl = Roo.get(this.fieldTpl.append(this.el, [
48355                f.id, //0
48356                f.fieldLabel, //1
48357                f.labelStyle||this.labelStyle||'', //2
48358                this.elementStyle||'', //3
48359                typeof f.labelSeparator == 'undefined' ? this.labelSeparator : f.labelSeparator, //4
48360                f.itemCls||this.itemCls||''  //5
48361        ], true).getPrevSibling());
48362     },
48363
48364     // private
48365     renderComponent : function(c){
48366         c.render(c.isLayout ? this.el : this.el.createChild());    
48367     },
48368     /**
48369      * Adds a object form elements (using the xtype property as the factory method.)
48370      * Valid xtypes are:  TextField, TextArea .... Button, Layout, FieldSet, Column
48371      * @param {Object} config 
48372      */
48373     addxtype : function(o)
48374     {
48375         // create the lement.
48376         o.form = this.form;
48377         var fe = Roo.factory(o, Roo.form);
48378         this.form.allItems.push(fe);
48379         this.stack.push(fe);
48380         
48381         if (fe.isFormField) {
48382             this.form.items.add(fe);
48383         }
48384          
48385         return fe;
48386     }
48387 });
48388
48389 /**
48390  * @class Roo.form.Column
48391  * @extends Roo.form.Layout
48392  * Creates a column container for layout and rendering of fields in an {@link Roo.form.Form}.
48393  * @constructor
48394  * @param {Object} config Configuration options
48395  */
48396 Roo.form.Column = function(config){
48397     Roo.form.Column.superclass.constructor.call(this, config);
48398 };
48399
48400 Roo.extend(Roo.form.Column, Roo.form.Layout, {
48401     /**
48402      * @cfg {Number/String} width
48403      * The fixed width of the column in pixels or CSS value (defaults to "auto")
48404      */
48405     /**
48406      * @cfg {String/Object} autoCreate
48407      * A DomHelper element spec used to autocreate the column (defaults to {tag: 'div', cls: 'x-form-ct x-form-column'})
48408      */
48409
48410     // private
48411     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct x-form-column'},
48412
48413     // private
48414     onRender : function(ct, position){
48415         Roo.form.Column.superclass.onRender.call(this, ct, position);
48416         if(this.width){
48417             this.el.setWidth(this.width);
48418         }
48419     }
48420 });
48421
48422
48423 /**
48424  * @class Roo.form.Row
48425  * @extends Roo.form.Layout
48426  * Creates a row container for layout and rendering of fields in an {@link Roo.form.Form}.
48427  * @constructor
48428  * @param {Object} config Configuration options
48429  */
48430
48431  
48432 Roo.form.Row = function(config){
48433     Roo.form.Row.superclass.constructor.call(this, config);
48434 };
48435  
48436 Roo.extend(Roo.form.Row, Roo.form.Layout, {
48437       /**
48438      * @cfg {Number/String} width
48439      * The fixed width of the column in pixels or CSS value (defaults to "auto")
48440      */
48441     /**
48442      * @cfg {Number/String} height
48443      * The fixed height of the column in pixels or CSS value (defaults to "auto")
48444      */
48445     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct x-form-row'},
48446     
48447     padWidth : 20,
48448     // private
48449     onRender : function(ct, position){
48450         //console.log('row render');
48451         if(!this.rowTpl){
48452             var t = new Roo.Template(
48453                 '<div class="x-form-item {5}" style="float:left;width:{6}px">',
48454                     '<label for="{0}" style="{2}">{1}{4}</label>',
48455                     '<div class="x-form-element" id="x-form-el-{0}" style="{3}">',
48456                     '</div>',
48457                 '</div>'
48458             );
48459             t.disableFormats = true;
48460             t.compile();
48461             Roo.form.Layout.prototype.rowTpl = t;
48462         }
48463         this.fieldTpl = this.rowTpl;
48464         
48465         //console.log('lw' + this.labelWidth +', la:' + this.labelAlign);
48466         var labelWidth = 100;
48467         
48468         if ((this.labelAlign != 'top')) {
48469             if (typeof this.labelWidth == 'number') {
48470                 labelWidth = this.labelWidth
48471             }
48472             this.padWidth =  20 + labelWidth;
48473             
48474         }
48475         
48476         Roo.form.Column.superclass.onRender.call(this, ct, position);
48477         if(this.width){
48478             this.el.setWidth(this.width);
48479         }
48480         if(this.height){
48481             this.el.setHeight(this.height);
48482         }
48483     },
48484     
48485     // private
48486     renderField : function(f){
48487         f.fieldEl = this.fieldTpl.append(this.el, [
48488                f.id, f.fieldLabel,
48489                f.labelStyle||this.labelStyle||'',
48490                this.elementStyle||'',
48491                typeof f.labelSeparator == 'undefined' ? this.labelSeparator : f.labelSeparator,
48492                f.itemCls||this.itemCls||'',
48493                f.width ? f.width + this.padWidth : 160 + this.padWidth
48494        ],true);
48495     }
48496 });
48497  
48498
48499 /**
48500  * @class Roo.form.FieldSet
48501  * @extends Roo.form.Layout
48502  * Creates a fieldset container for layout and rendering of fields in an {@link Roo.form.Form}.
48503  * @constructor
48504  * @param {Object} config Configuration options
48505  */
48506 Roo.form.FieldSet = function(config){
48507     Roo.form.FieldSet.superclass.constructor.call(this, config);
48508 };
48509
48510 Roo.extend(Roo.form.FieldSet, Roo.form.Layout, {
48511     /**
48512      * @cfg {String} legend
48513      * The text to display as the legend for the FieldSet (defaults to '')
48514      */
48515     /**
48516      * @cfg {String/Object} autoCreate
48517      * A DomHelper element spec used to autocreate the fieldset (defaults to {tag: 'fieldset', cn: {tag:'legend'}})
48518      */
48519
48520     // private
48521     defaultAutoCreate : {tag: 'fieldset', cn: {tag:'legend'}},
48522
48523     // private
48524     onRender : function(ct, position){
48525         Roo.form.FieldSet.superclass.onRender.call(this, ct, position);
48526         if(this.legend){
48527             this.setLegend(this.legend);
48528         }
48529     },
48530
48531     // private
48532     setLegend : function(text){
48533         if(this.rendered){
48534             this.el.child('legend').update(text);
48535         }
48536     }
48537 });/*
48538  * Based on:
48539  * Ext JS Library 1.1.1
48540  * Copyright(c) 2006-2007, Ext JS, LLC.
48541  *
48542  * Originally Released Under LGPL - original licence link has changed is not relivant.
48543  *
48544  * Fork - LGPL
48545  * <script type="text/javascript">
48546  */
48547 /**
48548  * @class Roo.form.VTypes
48549  * Overridable validation definitions. The validations provided are basic and intended to be easily customizable and extended.
48550  * @singleton
48551  */
48552 Roo.form.VTypes = function(){
48553     // closure these in so they are only created once.
48554     var alpha = /^[a-zA-Z_]+$/;
48555     var alphanum = /^[a-zA-Z0-9_]+$/;
48556     var email = /^([\w]+)(.[\w]+)*@([\w-]+\.){1,5}([A-Za-z]){2,24}$/;
48557     var url = /(((https?)|(ftp)):\/\/([\-\w]+\.)+\w{2,3}(\/[%\-\w]+(\.\w{2,})?)*(([\w\-\.\?\\\/+@&#;`~=%!]*)(\.\w{2,})?)*\/?)/i;
48558
48559     // All these messages and functions are configurable
48560     return {
48561         /**
48562          * The function used to validate email addresses
48563          * @param {String} value The email address
48564          */
48565         'email' : function(v){
48566             return email.test(v);
48567         },
48568         /**
48569          * The error text to display when the email validation function returns false
48570          * @type String
48571          */
48572         'emailText' : 'This field should be an e-mail address in the format "user@domain.com"',
48573         /**
48574          * The keystroke filter mask to be applied on email input
48575          * @type RegExp
48576          */
48577         'emailMask' : /[a-z0-9_\.\-@]/i,
48578
48579         /**
48580          * The function used to validate URLs
48581          * @param {String} value The URL
48582          */
48583         'url' : function(v){
48584             return url.test(v);
48585         },
48586         /**
48587          * The error text to display when the url validation function returns false
48588          * @type String
48589          */
48590         'urlText' : 'This field should be a URL in the format "http:/'+'/www.domain.com"',
48591         
48592         /**
48593          * The function used to validate alpha values
48594          * @param {String} value The value
48595          */
48596         'alpha' : function(v){
48597             return alpha.test(v);
48598         },
48599         /**
48600          * The error text to display when the alpha validation function returns false
48601          * @type String
48602          */
48603         'alphaText' : 'This field should only contain letters and _',
48604         /**
48605          * The keystroke filter mask to be applied on alpha input
48606          * @type RegExp
48607          */
48608         'alphaMask' : /[a-z_]/i,
48609
48610         /**
48611          * The function used to validate alphanumeric values
48612          * @param {String} value The value
48613          */
48614         'alphanum' : function(v){
48615             return alphanum.test(v);
48616         },
48617         /**
48618          * The error text to display when the alphanumeric validation function returns false
48619          * @type String
48620          */
48621         'alphanumText' : 'This field should only contain letters, numbers and _',
48622         /**
48623          * The keystroke filter mask to be applied on alphanumeric input
48624          * @type RegExp
48625          */
48626         'alphanumMask' : /[a-z0-9_]/i
48627     };
48628 }();//<script type="text/javascript">
48629
48630 /**
48631  * @class Roo.form.FCKeditor
48632  * @extends Roo.form.TextArea
48633  * Wrapper around the FCKEditor http://www.fckeditor.net
48634  * @constructor
48635  * Creates a new FCKeditor
48636  * @param {Object} config Configuration options
48637  */
48638 Roo.form.FCKeditor = function(config){
48639     Roo.form.FCKeditor.superclass.constructor.call(this, config);
48640     this.addEvents({
48641          /**
48642          * @event editorinit
48643          * Fired when the editor is initialized - you can add extra handlers here..
48644          * @param {FCKeditor} this
48645          * @param {Object} the FCK object.
48646          */
48647         editorinit : true
48648     });
48649     
48650     
48651 };
48652 Roo.form.FCKeditor.editors = { };
48653 Roo.extend(Roo.form.FCKeditor, Roo.form.TextArea,
48654 {
48655     //defaultAutoCreate : {
48656     //    tag : "textarea",style   : "width:100px;height:60px;" ,autocomplete    : "off"
48657     //},
48658     // private
48659     /**
48660      * @cfg {Object} fck options - see fck manual for details.
48661      */
48662     fckconfig : false,
48663     
48664     /**
48665      * @cfg {Object} fck toolbar set (Basic or Default)
48666      */
48667     toolbarSet : 'Basic',
48668     /**
48669      * @cfg {Object} fck BasePath
48670      */ 
48671     basePath : '/fckeditor/',
48672     
48673     
48674     frame : false,
48675     
48676     value : '',
48677     
48678    
48679     onRender : function(ct, position)
48680     {
48681         if(!this.el){
48682             this.defaultAutoCreate = {
48683                 tag: "textarea",
48684                 style:"width:300px;height:60px;",
48685                 autocomplete: "new-password"
48686             };
48687         }
48688         Roo.form.FCKeditor.superclass.onRender.call(this, ct, position);
48689         /*
48690         if(this.grow){
48691             this.textSizeEl = Roo.DomHelper.append(document.body, {tag: "pre", cls: "x-form-grow-sizer"});
48692             if(this.preventScrollbars){
48693                 this.el.setStyle("overflow", "hidden");
48694             }
48695             this.el.setHeight(this.growMin);
48696         }
48697         */
48698         //console.log('onrender' + this.getId() );
48699         Roo.form.FCKeditor.editors[this.getId()] = this;
48700          
48701
48702         this.replaceTextarea() ;
48703         
48704     },
48705     
48706     getEditor : function() {
48707         return this.fckEditor;
48708     },
48709     /**
48710      * Sets a data value into the field and validates it.  To set the value directly without validation see {@link #setRawValue}.
48711      * @param {Mixed} value The value to set
48712      */
48713     
48714     
48715     setValue : function(value)
48716     {
48717         //console.log('setValue: ' + value);
48718         
48719         if(typeof(value) == 'undefined') { // not sure why this is happending...
48720             return;
48721         }
48722         Roo.form.FCKeditor.superclass.setValue.apply(this,[value]);
48723         
48724         //if(!this.el || !this.getEditor()) {
48725         //    this.value = value;
48726             //this.setValue.defer(100,this,[value]);    
48727         //    return;
48728         //} 
48729         
48730         if(!this.getEditor()) {
48731             return;
48732         }
48733         
48734         this.getEditor().SetData(value);
48735         
48736         //
48737
48738     },
48739
48740     /**
48741      * Returns the normalized data value (undefined or emptyText will be returned as '').  To return the raw value see {@link #getRawValue}.
48742      * @return {Mixed} value The field value
48743      */
48744     getValue : function()
48745     {
48746         
48747         if (this.frame && this.frame.dom.style.display == 'none') {
48748             return Roo.form.FCKeditor.superclass.getValue.call(this);
48749         }
48750         
48751         if(!this.el || !this.getEditor()) {
48752            
48753            // this.getValue.defer(100,this); 
48754             return this.value;
48755         }
48756        
48757         
48758         var value=this.getEditor().GetData();
48759         Roo.form.FCKeditor.superclass.setValue.apply(this,[value]);
48760         return Roo.form.FCKeditor.superclass.getValue.call(this);
48761         
48762
48763     },
48764
48765     /**
48766      * Returns the raw data value which may or may not be a valid, defined value.  To return a normalized value see {@link #getValue}.
48767      * @return {Mixed} value The field value
48768      */
48769     getRawValue : function()
48770     {
48771         if (this.frame && this.frame.dom.style.display == 'none') {
48772             return Roo.form.FCKeditor.superclass.getRawValue.call(this);
48773         }
48774         
48775         if(!this.el || !this.getEditor()) {
48776             //this.getRawValue.defer(100,this); 
48777             return this.value;
48778             return;
48779         }
48780         
48781         
48782         
48783         var value=this.getEditor().GetData();
48784         Roo.form.FCKeditor.superclass.setRawValue.apply(this,[value]);
48785         return Roo.form.FCKeditor.superclass.getRawValue.call(this);
48786          
48787     },
48788     
48789     setSize : function(w,h) {
48790         
48791         
48792         
48793         //if (this.frame && this.frame.dom.style.display == 'none') {
48794         //    Roo.form.FCKeditor.superclass.setSize.apply(this, [w, h]);
48795         //    return;
48796         //}
48797         //if(!this.el || !this.getEditor()) {
48798         //    this.setSize.defer(100,this, [w,h]); 
48799         //    return;
48800         //}
48801         
48802         
48803         
48804         Roo.form.FCKeditor.superclass.setSize.apply(this, [w, h]);
48805         
48806         this.frame.dom.setAttribute('width', w);
48807         this.frame.dom.setAttribute('height', h);
48808         this.frame.setSize(w,h);
48809         
48810     },
48811     
48812     toggleSourceEdit : function(value) {
48813         
48814       
48815          
48816         this.el.dom.style.display = value ? '' : 'none';
48817         this.frame.dom.style.display = value ?  'none' : '';
48818         
48819     },
48820     
48821     
48822     focus: function(tag)
48823     {
48824         if (this.frame.dom.style.display == 'none') {
48825             return Roo.form.FCKeditor.superclass.focus.call(this);
48826         }
48827         if(!this.el || !this.getEditor()) {
48828             this.focus.defer(100,this, [tag]); 
48829             return;
48830         }
48831         
48832         
48833         
48834         
48835         var tgs = this.getEditor().EditorDocument.getElementsByTagName(tag);
48836         this.getEditor().Focus();
48837         if (tgs.length) {
48838             if (!this.getEditor().Selection.GetSelection()) {
48839                 this.focus.defer(100,this, [tag]); 
48840                 return;
48841             }
48842             
48843             
48844             var r = this.getEditor().EditorDocument.createRange();
48845             r.setStart(tgs[0],0);
48846             r.setEnd(tgs[0],0);
48847             this.getEditor().Selection.GetSelection().removeAllRanges();
48848             this.getEditor().Selection.GetSelection().addRange(r);
48849             this.getEditor().Focus();
48850         }
48851         
48852     },
48853     
48854     
48855     
48856     replaceTextarea : function()
48857     {
48858         if ( document.getElementById( this.getId() + '___Frame' ) ) {
48859             return ;
48860         }
48861         //if ( !this.checkBrowser || this._isCompatibleBrowser() )
48862         //{
48863             // We must check the elements firstly using the Id and then the name.
48864         var oTextarea = document.getElementById( this.getId() );
48865         
48866         var colElementsByName = document.getElementsByName( this.getId() ) ;
48867          
48868         oTextarea.style.display = 'none' ;
48869
48870         if ( oTextarea.tabIndex ) {            
48871             this.TabIndex = oTextarea.tabIndex ;
48872         }
48873         
48874         this._insertHtmlBefore( this._getConfigHtml(), oTextarea ) ;
48875         this._insertHtmlBefore( this._getIFrameHtml(), oTextarea ) ;
48876         this.frame = Roo.get(this.getId() + '___Frame')
48877     },
48878     
48879     _getConfigHtml : function()
48880     {
48881         var sConfig = '' ;
48882
48883         for ( var o in this.fckconfig ) {
48884             sConfig += sConfig.length > 0  ? '&amp;' : '';
48885             sConfig += encodeURIComponent( o ) + '=' + encodeURIComponent( this.fckconfig[o] ) ;
48886         }
48887
48888         return '<input type="hidden" id="' + this.getId() + '___Config" value="' + sConfig + '" style="display:none" />' ;
48889     },
48890     
48891     
48892     _getIFrameHtml : function()
48893     {
48894         var sFile = 'fckeditor.html' ;
48895         /* no idea what this is about..
48896         try
48897         {
48898             if ( (/fcksource=true/i).test( window.top.location.search ) )
48899                 sFile = 'fckeditor.original.html' ;
48900         }
48901         catch (e) { 
48902         */
48903
48904         var sLink = this.basePath + 'editor/' + sFile + '?InstanceName=' + encodeURIComponent( this.getId() ) ;
48905         sLink += this.toolbarSet ? ( '&amp;Toolbar=' + this.toolbarSet)  : '';
48906         
48907         
48908         var html = '<iframe id="' + this.getId() +
48909             '___Frame" src="' + sLink +
48910             '" width="' + this.width +
48911             '" height="' + this.height + '"' +
48912             (this.tabIndex ?  ' tabindex="' + this.tabIndex + '"' :'' ) +
48913             ' frameborder="0" scrolling="no"></iframe>' ;
48914
48915         return html ;
48916     },
48917     
48918     _insertHtmlBefore : function( html, element )
48919     {
48920         if ( element.insertAdjacentHTML )       {
48921             // IE
48922             element.insertAdjacentHTML( 'beforeBegin', html ) ;
48923         } else { // Gecko
48924             var oRange = document.createRange() ;
48925             oRange.setStartBefore( element ) ;
48926             var oFragment = oRange.createContextualFragment( html );
48927             element.parentNode.insertBefore( oFragment, element ) ;
48928         }
48929     }
48930     
48931     
48932   
48933     
48934     
48935     
48936     
48937
48938 });
48939
48940 //Roo.reg('fckeditor', Roo.form.FCKeditor);
48941
48942 function FCKeditor_OnComplete(editorInstance){
48943     var f = Roo.form.FCKeditor.editors[editorInstance.Name];
48944     f.fckEditor = editorInstance;
48945     //console.log("loaded");
48946     f.fireEvent('editorinit', f, editorInstance);
48947
48948   
48949
48950  
48951
48952
48953
48954
48955
48956
48957
48958
48959
48960
48961
48962
48963
48964
48965
48966 //<script type="text/javascript">
48967 /**
48968  * @class Roo.form.GridField
48969  * @extends Roo.form.Field
48970  * Embed a grid (or editable grid into a form)
48971  * STATUS ALPHA
48972  * 
48973  * This embeds a grid in a form, the value of the field should be the json encoded array of rows
48974  * it needs 
48975  * xgrid.store = Roo.data.Store
48976  * xgrid.store.proxy = Roo.data.MemoryProxy (data = [] )
48977  * xgrid.store.reader = Roo.data.JsonReader 
48978  * 
48979  * 
48980  * @constructor
48981  * Creates a new GridField
48982  * @param {Object} config Configuration options
48983  */
48984 Roo.form.GridField = function(config){
48985     Roo.form.GridField.superclass.constructor.call(this, config);
48986      
48987 };
48988
48989 Roo.extend(Roo.form.GridField, Roo.form.Field,  {
48990     /**
48991      * @cfg {Number} width  - used to restrict width of grid..
48992      */
48993     width : 100,
48994     /**
48995      * @cfg {Number} height - used to restrict height of grid..
48996      */
48997     height : 50,
48998      /**
48999      * @cfg {Object} xgrid (xtype'd description of grid) { xtype : 'Grid', dataSource: .... }
49000          * 
49001          *}
49002      */
49003     xgrid : false, 
49004     /**
49005      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49006      * {tag: "input", type: "checkbox", autocomplete: "off"})
49007      */
49008    // defaultAutoCreate : { tag: 'div' },
49009     defaultAutoCreate : { tag: 'input', type: 'hidden', autocomplete: 'new-password'},
49010     /**
49011      * @cfg {String} addTitle Text to include for adding a title.
49012      */
49013     addTitle : false,
49014     //
49015     onResize : function(){
49016         Roo.form.Field.superclass.onResize.apply(this, arguments);
49017     },
49018
49019     initEvents : function(){
49020         // Roo.form.Checkbox.superclass.initEvents.call(this);
49021         // has no events...
49022        
49023     },
49024
49025
49026     getResizeEl : function(){
49027         return this.wrap;
49028     },
49029
49030     getPositionEl : function(){
49031         return this.wrap;
49032     },
49033
49034     // private
49035     onRender : function(ct, position){
49036         
49037         this.style = this.style || 'overflow: hidden; border:1px solid #c3daf9;';
49038         var style = this.style;
49039         delete this.style;
49040         
49041         Roo.form.GridField.superclass.onRender.call(this, ct, position);
49042         this.wrap = this.el.wrap({cls: ''}); // not sure why ive done thsi...
49043         this.viewEl = this.wrap.createChild({ tag: 'div' });
49044         if (style) {
49045             this.viewEl.applyStyles(style);
49046         }
49047         if (this.width) {
49048             this.viewEl.setWidth(this.width);
49049         }
49050         if (this.height) {
49051             this.viewEl.setHeight(this.height);
49052         }
49053         //if(this.inputValue !== undefined){
49054         //this.setValue(this.value);
49055         
49056         
49057         this.grid = new Roo.grid[this.xgrid.xtype](this.viewEl, this.xgrid);
49058         
49059         
49060         this.grid.render();
49061         this.grid.getDataSource().on('remove', this.refreshValue, this);
49062         this.grid.getDataSource().on('update', this.refreshValue, this);
49063         this.grid.on('afteredit', this.refreshValue, this);
49064  
49065     },
49066      
49067     
49068     /**
49069      * Sets the value of the item. 
49070      * @param {String} either an object  or a string..
49071      */
49072     setValue : function(v){
49073         //this.value = v;
49074         v = v || []; // empty set..
49075         // this does not seem smart - it really only affects memoryproxy grids..
49076         if (this.grid && this.grid.getDataSource() && typeof(v) != 'undefined') {
49077             var ds = this.grid.getDataSource();
49078             // assumes a json reader..
49079             var data = {}
49080             data[ds.reader.meta.root ] =  typeof(v) == 'string' ? Roo.decode(v) : v;
49081             ds.loadData( data);
49082         }
49083         // clear selection so it does not get stale.
49084         if (this.grid.sm) { 
49085             this.grid.sm.clearSelections();
49086         }
49087         
49088         Roo.form.GridField.superclass.setValue.call(this, v);
49089         this.refreshValue();
49090         // should load data in the grid really....
49091     },
49092     
49093     // private
49094     refreshValue: function() {
49095          var val = [];
49096         this.grid.getDataSource().each(function(r) {
49097             val.push(r.data);
49098         });
49099         this.el.dom.value = Roo.encode(val);
49100     }
49101     
49102      
49103     
49104     
49105 });/*
49106  * Based on:
49107  * Ext JS Library 1.1.1
49108  * Copyright(c) 2006-2007, Ext JS, LLC.
49109  *
49110  * Originally Released Under LGPL - original licence link has changed is not relivant.
49111  *
49112  * Fork - LGPL
49113  * <script type="text/javascript">
49114  */
49115 /**
49116  * @class Roo.form.DisplayField
49117  * @extends Roo.form.Field
49118  * A generic Field to display non-editable data.
49119  * @cfg {Boolean} closable (true|false) default false
49120  * @constructor
49121  * Creates a new Display Field item.
49122  * @param {Object} config Configuration options
49123  */
49124 Roo.form.DisplayField = function(config){
49125     Roo.form.DisplayField.superclass.constructor.call(this, config);
49126     
49127     this.addEvents({
49128         /**
49129          * @event close
49130          * Fires after the click the close btn
49131              * @param {Roo.form.DisplayField} this
49132              */
49133         close : true
49134     });
49135 };
49136
49137 Roo.extend(Roo.form.DisplayField, Roo.form.TextField,  {
49138     inputType:      'hidden',
49139     allowBlank:     true,
49140     readOnly:         true,
49141     
49142  
49143     /**
49144      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
49145      */
49146     focusClass : undefined,
49147     /**
49148      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
49149      */
49150     fieldClass: 'x-form-field',
49151     
49152      /**
49153      * @cfg {Function} valueRenderer The renderer for the field (so you can reformat output). should return raw HTML
49154      */
49155     valueRenderer: undefined,
49156     
49157     width: 100,
49158     /**
49159      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49160      * {tag: "input", type: "checkbox", autocomplete: "off"})
49161      */
49162      
49163  //   defaultAutoCreate : { tag: 'input', type: 'hidden', autocomplete: 'off'},
49164  
49165     closable : false,
49166     
49167     onResize : function(){
49168         Roo.form.DisplayField.superclass.onResize.apply(this, arguments);
49169         
49170     },
49171
49172     initEvents : function(){
49173         // Roo.form.Checkbox.superclass.initEvents.call(this);
49174         // has no events...
49175         
49176         if(this.closable){
49177             this.closeEl.on('click', this.onClose, this);
49178         }
49179        
49180     },
49181
49182
49183     getResizeEl : function(){
49184         return this.wrap;
49185     },
49186
49187     getPositionEl : function(){
49188         return this.wrap;
49189     },
49190
49191     // private
49192     onRender : function(ct, position){
49193         
49194         Roo.form.DisplayField.superclass.onRender.call(this, ct, position);
49195         //if(this.inputValue !== undefined){
49196         this.wrap = this.el.wrap();
49197         
49198         this.viewEl = this.wrap.createChild({ tag: 'div', cls: 'x-form-displayfield'});
49199         
49200         if(this.closable){
49201             this.closeEl = this.wrap.createChild({ tag: 'div', cls: 'x-dlg-close'});
49202         }
49203         
49204         if (this.bodyStyle) {
49205             this.viewEl.applyStyles(this.bodyStyle);
49206         }
49207         //this.viewEl.setStyle('padding', '2px');
49208         
49209         this.setValue(this.value);
49210         
49211     },
49212 /*
49213     // private
49214     initValue : Roo.emptyFn,
49215
49216   */
49217
49218         // private
49219     onClick : function(){
49220         
49221     },
49222
49223     /**
49224      * Sets the checked state of the checkbox.
49225      * @param {Boolean/String} checked True, 'true', '1', or 'on' to check the checkbox, any other value will uncheck it.
49226      */
49227     setValue : function(v){
49228         this.value = v;
49229         var html = this.valueRenderer ?  this.valueRenderer(v) : String.format('{0}', v);
49230         // this might be called before we have a dom element..
49231         if (!this.viewEl) {
49232             return;
49233         }
49234         this.viewEl.dom.innerHTML = html;
49235         Roo.form.DisplayField.superclass.setValue.call(this, v);
49236
49237     },
49238     
49239     onClose : function(e)
49240     {
49241         e.preventDefault();
49242         
49243         this.fireEvent('close', this);
49244     }
49245 });/*
49246  * 
49247  * Licence- LGPL
49248  * 
49249  */
49250
49251 /**
49252  * @class Roo.form.DayPicker
49253  * @extends Roo.form.Field
49254  * A Day picker show [M] [T] [W] ....
49255  * @constructor
49256  * Creates a new Day Picker
49257  * @param {Object} config Configuration options
49258  */
49259 Roo.form.DayPicker= function(config){
49260     Roo.form.DayPicker.superclass.constructor.call(this, config);
49261      
49262 };
49263
49264 Roo.extend(Roo.form.DayPicker, Roo.form.Field,  {
49265     /**
49266      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
49267      */
49268     focusClass : undefined,
49269     /**
49270      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
49271      */
49272     fieldClass: "x-form-field",
49273    
49274     /**
49275      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49276      * {tag: "input", type: "checkbox", autocomplete: "off"})
49277      */
49278     defaultAutoCreate : { tag: "input", type: 'hidden', autocomplete: "new-password"},
49279     
49280    
49281     actionMode : 'viewEl', 
49282     //
49283     // private
49284  
49285     inputType : 'hidden',
49286     
49287      
49288     inputElement: false, // real input element?
49289     basedOn: false, // ????
49290     
49291     isFormField: true, // not sure where this is needed!!!!
49292
49293     onResize : function(){
49294         Roo.form.Checkbox.superclass.onResize.apply(this, arguments);
49295         if(!this.boxLabel){
49296             this.el.alignTo(this.wrap, 'c-c');
49297         }
49298     },
49299
49300     initEvents : function(){
49301         Roo.form.Checkbox.superclass.initEvents.call(this);
49302         this.el.on("click", this.onClick,  this);
49303         this.el.on("change", this.onClick,  this);
49304     },
49305
49306
49307     getResizeEl : function(){
49308         return this.wrap;
49309     },
49310
49311     getPositionEl : function(){
49312         return this.wrap;
49313     },
49314
49315     
49316     // private
49317     onRender : function(ct, position){
49318         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
49319        
49320         this.wrap = this.el.wrap({cls: 'x-form-daypick-item '});
49321         
49322         var r1 = '<table><tr>';
49323         var r2 = '<tr class="x-form-daypick-icons">';
49324         for (var i=0; i < 7; i++) {
49325             r1+= '<td><div>' + Date.dayNames[i].substring(0,3) + '</div></td>';
49326             r2+= '<td><img class="x-menu-item-icon" src="' + Roo.BLANK_IMAGE_URL  +'"></td>';
49327         }
49328         
49329         var viewEl = this.wrap.createChild( r1 + '</tr>' + r2 + '</tr></table>');
49330         viewEl.select('img').on('click', this.onClick, this);
49331         this.viewEl = viewEl;   
49332         
49333         
49334         // this will not work on Chrome!!!
49335         this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
49336         this.el.on('propertychange', this.setFromHidden,  this);  //ie
49337         
49338         
49339           
49340
49341     },
49342
49343     // private
49344     initValue : Roo.emptyFn,
49345
49346     /**
49347      * Returns the checked state of the checkbox.
49348      * @return {Boolean} True if checked, else false
49349      */
49350     getValue : function(){
49351         return this.el.dom.value;
49352         
49353     },
49354
49355         // private
49356     onClick : function(e){ 
49357         //this.setChecked(!this.checked);
49358         Roo.get(e.target).toggleClass('x-menu-item-checked');
49359         this.refreshValue();
49360         //if(this.el.dom.checked != this.checked){
49361         //    this.setValue(this.el.dom.checked);
49362        // }
49363     },
49364     
49365     // private
49366     refreshValue : function()
49367     {
49368         var val = '';
49369         this.viewEl.select('img',true).each(function(e,i,n)  {
49370             val += e.is(".x-menu-item-checked") ? String(n) : '';
49371         });
49372         this.setValue(val, true);
49373     },
49374
49375     /**
49376      * Sets the checked state of the checkbox.
49377      * On is always based on a string comparison between inputValue and the param.
49378      * @param {Boolean/String} value - the value to set 
49379      * @param {Boolean/String} suppressEvent - whether to suppress the checkchange event.
49380      */
49381     setValue : function(v,suppressEvent){
49382         if (!this.el.dom) {
49383             return;
49384         }
49385         var old = this.el.dom.value ;
49386         this.el.dom.value = v;
49387         if (suppressEvent) {
49388             return ;
49389         }
49390          
49391         // update display..
49392         this.viewEl.select('img',true).each(function(e,i,n)  {
49393             
49394             var on = e.is(".x-menu-item-checked");
49395             var newv = v.indexOf(String(n)) > -1;
49396             if (on != newv) {
49397                 e.toggleClass('x-menu-item-checked');
49398             }
49399             
49400         });
49401         
49402         
49403         this.fireEvent('change', this, v, old);
49404         
49405         
49406     },
49407    
49408     // handle setting of hidden value by some other method!!?!?
49409     setFromHidden: function()
49410     {
49411         if(!this.el){
49412             return;
49413         }
49414         //console.log("SET FROM HIDDEN");
49415         //alert('setFrom hidden');
49416         this.setValue(this.el.dom.value);
49417     },
49418     
49419     onDestroy : function()
49420     {
49421         if(this.viewEl){
49422             Roo.get(this.viewEl).remove();
49423         }
49424          
49425         Roo.form.DayPicker.superclass.onDestroy.call(this);
49426     }
49427
49428 });/*
49429  * RooJS Library 1.1.1
49430  * Copyright(c) 2008-2011  Alan Knowles
49431  *
49432  * License - LGPL
49433  */
49434  
49435
49436 /**
49437  * @class Roo.form.ComboCheck
49438  * @extends Roo.form.ComboBox
49439  * A combobox for multiple select items.
49440  *
49441  * FIXME - could do with a reset button..
49442  * 
49443  * @constructor
49444  * Create a new ComboCheck
49445  * @param {Object} config Configuration options
49446  */
49447 Roo.form.ComboCheck = function(config){
49448     Roo.form.ComboCheck.superclass.constructor.call(this, config);
49449     // should verify some data...
49450     // like
49451     // hiddenName = required..
49452     // displayField = required
49453     // valudField == required
49454     var req= [ 'hiddenName', 'displayField', 'valueField' ];
49455     var _t = this;
49456     Roo.each(req, function(e) {
49457         if ((typeof(_t[e]) == 'undefined' ) || !_t[e].length) {
49458             throw "Roo.form.ComboCheck : missing value for: " + e;
49459         }
49460     });
49461     
49462     
49463 };
49464
49465 Roo.extend(Roo.form.ComboCheck, Roo.form.ComboBox, {
49466      
49467      
49468     editable : false,
49469      
49470     selectedClass: 'x-menu-item-checked', 
49471     
49472     // private
49473     onRender : function(ct, position){
49474         var _t = this;
49475         
49476         
49477         
49478         if(!this.tpl){
49479             var cls = 'x-combo-list';
49480
49481             
49482             this.tpl =  new Roo.Template({
49483                 html :  '<div class="'+cls+'-item x-menu-check-item">' +
49484                    '<img class="x-menu-item-icon" style="margin: 0px;" src="' + Roo.BLANK_IMAGE_URL + '">' + 
49485                    '<span>{' + this.displayField + '}</span>' +
49486                     '</div>' 
49487                 
49488             });
49489         }
49490  
49491         
49492         Roo.form.ComboCheck.superclass.onRender.call(this, ct, position);
49493         this.view.singleSelect = false;
49494         this.view.multiSelect = true;
49495         this.view.toggleSelect = true;
49496         this.pageTb.add(new Roo.Toolbar.Fill(), {
49497             
49498             text: 'Done',
49499             handler: function()
49500             {
49501                 _t.collapse();
49502             }
49503         });
49504     },
49505     
49506     onViewOver : function(e, t){
49507         // do nothing...
49508         return;
49509         
49510     },
49511     
49512     onViewClick : function(doFocus,index){
49513         return;
49514         
49515     },
49516     select: function () {
49517         //Roo.log("SELECT CALLED");
49518     },
49519      
49520     selectByValue : function(xv, scrollIntoView){
49521         var ar = this.getValueArray();
49522         var sels = [];
49523         
49524         Roo.each(ar, function(v) {
49525             if(v === undefined || v === null){
49526                 return;
49527             }
49528             var r = this.findRecord(this.valueField, v);
49529             if(r){
49530                 sels.push(this.store.indexOf(r))
49531                 
49532             }
49533         },this);
49534         this.view.select(sels);
49535         return false;
49536     },
49537     
49538     
49539     
49540     onSelect : function(record, index){
49541        // Roo.log("onselect Called");
49542        // this is only called by the clear button now..
49543         this.view.clearSelections();
49544         this.setValue('[]');
49545         if (this.value != this.valueBefore) {
49546             this.fireEvent('change', this, this.value, this.valueBefore);
49547             this.valueBefore = this.value;
49548         }
49549     },
49550     getValueArray : function()
49551     {
49552         var ar = [] ;
49553         
49554         try {
49555             //Roo.log(this.value);
49556             if (typeof(this.value) == 'undefined') {
49557                 return [];
49558             }
49559             var ar = Roo.decode(this.value);
49560             return  ar instanceof Array ? ar : []; //?? valid?
49561             
49562         } catch(e) {
49563             Roo.log(e + "\nRoo.form.ComboCheck:getValueArray  invalid data:" + this.getValue());
49564             return [];
49565         }
49566          
49567     },
49568     expand : function ()
49569     {
49570         
49571         Roo.form.ComboCheck.superclass.expand.call(this);
49572         this.valueBefore = typeof(this.value) == 'undefined' ? '' : this.value;
49573         //this.valueBefore = typeof(this.valueBefore) == 'undefined' ? '' : this.valueBefore;
49574         
49575
49576     },
49577     
49578     collapse : function(){
49579         Roo.form.ComboCheck.superclass.collapse.call(this);
49580         var sl = this.view.getSelectedIndexes();
49581         var st = this.store;
49582         var nv = [];
49583         var tv = [];
49584         var r;
49585         Roo.each(sl, function(i) {
49586             r = st.getAt(i);
49587             nv.push(r.get(this.valueField));
49588         },this);
49589         this.setValue(Roo.encode(nv));
49590         if (this.value != this.valueBefore) {
49591
49592             this.fireEvent('change', this, this.value, this.valueBefore);
49593             this.valueBefore = this.value;
49594         }
49595         
49596     },
49597     
49598     setValue : function(v){
49599         // Roo.log(v);
49600         this.value = v;
49601         
49602         var vals = this.getValueArray();
49603         var tv = [];
49604         Roo.each(vals, function(k) {
49605             var r = this.findRecord(this.valueField, k);
49606             if(r){
49607                 tv.push(r.data[this.displayField]);
49608             }else if(this.valueNotFoundText !== undefined){
49609                 tv.push( this.valueNotFoundText );
49610             }
49611         },this);
49612        // Roo.log(tv);
49613         
49614         Roo.form.ComboBox.superclass.setValue.call(this, tv.join(', '));
49615         this.hiddenField.value = v;
49616         this.value = v;
49617     }
49618     
49619 });/*
49620  * Based on:
49621  * Ext JS Library 1.1.1
49622  * Copyright(c) 2006-2007, Ext JS, LLC.
49623  *
49624  * Originally Released Under LGPL - original licence link has changed is not relivant.
49625  *
49626  * Fork - LGPL
49627  * <script type="text/javascript">
49628  */
49629  
49630 /**
49631  * @class Roo.form.Signature
49632  * @extends Roo.form.Field
49633  * Signature field.  
49634  * @constructor
49635  * 
49636  * @param {Object} config Configuration options
49637  */
49638
49639 Roo.form.Signature = function(config){
49640     Roo.form.Signature.superclass.constructor.call(this, config);
49641     
49642     this.addEvents({// not in used??
49643          /**
49644          * @event confirm
49645          * Fires when the 'confirm' icon is pressed (add a listener to enable add button)
49646              * @param {Roo.form.Signature} combo This combo box
49647              */
49648         'confirm' : true,
49649         /**
49650          * @event reset
49651          * Fires when the 'edit' icon is pressed (add a listener to enable add button)
49652              * @param {Roo.form.ComboBox} combo This combo box
49653              * @param {Roo.data.Record|false} record The data record returned from the underlying store (or false on nothing selected)
49654              */
49655         'reset' : true
49656     });
49657 };
49658
49659 Roo.extend(Roo.form.Signature, Roo.form.Field,  {
49660     /**
49661      * @cfg {Object} labels Label to use when rendering a form.
49662      * defaults to 
49663      * labels : { 
49664      *      clear : "Clear",
49665      *      confirm : "Confirm"
49666      *  }
49667      */
49668     labels : { 
49669         clear : "Clear",
49670         confirm : "Confirm"
49671     },
49672     /**
49673      * @cfg {Number} width The signature panel width (defaults to 300)
49674      */
49675     width: 300,
49676     /**
49677      * @cfg {Number} height The signature panel height (defaults to 100)
49678      */
49679     height : 100,
49680     /**
49681      * @cfg {Boolean} allowBlank False to validate that the value length > 0 (defaults to false)
49682      */
49683     allowBlank : false,
49684     
49685     //private
49686     // {Object} signPanel The signature SVG panel element (defaults to {})
49687     signPanel : {},
49688     // {Boolean} isMouseDown False to validate that the mouse down event (defaults to false)
49689     isMouseDown : false,
49690     // {Boolean} isConfirmed validate the signature is confirmed or not for submitting form (defaults to false)
49691     isConfirmed : false,
49692     // {String} signatureTmp SVG mapping string (defaults to empty string)
49693     signatureTmp : '',
49694     
49695     
49696     defaultAutoCreate : { // modified by initCompnoent..
49697         tag: "input",
49698         type:"hidden"
49699     },
49700
49701     // private
49702     onRender : function(ct, position){
49703         
49704         Roo.form.Signature.superclass.onRender.call(this, ct, position);
49705         
49706         this.wrap = this.el.wrap({
49707             cls:'x-form-signature-wrap', style : 'width: ' + this.width + 'px', cn:{cls:'x-form-signature'}
49708         });
49709         
49710         this.createToolbar(this);
49711         this.signPanel = this.wrap.createChild({
49712                 tag: 'div',
49713                 style: 'width: ' + this.width + 'px; height: ' + this.height + 'px; border: 0;'
49714             }, this.el
49715         );
49716             
49717         this.svgID = Roo.id();
49718         this.svgEl = this.signPanel.createChild({
49719               xmlns : 'http://www.w3.org/2000/svg',
49720               tag : 'svg',
49721               id : this.svgID + "-svg",
49722               width: this.width,
49723               height: this.height,
49724               viewBox: '0 0 '+this.width+' '+this.height,
49725               cn : [
49726                 {
49727                     tag: "rect",
49728                     id: this.svgID + "-svg-r",
49729                     width: this.width,
49730                     height: this.height,
49731                     fill: "#ffa"
49732                 },
49733                 {
49734                     tag: "line",
49735                     id: this.svgID + "-svg-l",
49736                     x1: "0", // start
49737                     y1: (this.height*0.8), // start set the line in 80% of height
49738                     x2: this.width, // end
49739                     y2: (this.height*0.8), // end set the line in 80% of height
49740                     'stroke': "#666",
49741                     'stroke-width': "1",
49742                     'stroke-dasharray': "3",
49743                     'shape-rendering': "crispEdges",
49744                     'pointer-events': "none"
49745                 },
49746                 {
49747                     tag: "path",
49748                     id: this.svgID + "-svg-p",
49749                     'stroke': "navy",
49750                     'stroke-width': "3",
49751                     'fill': "none",
49752                     'pointer-events': 'none'
49753                 }
49754               ]
49755         });
49756         this.createSVG();
49757         this.svgBox = this.svgEl.dom.getScreenCTM();
49758     },
49759     createSVG : function(){ 
49760         var svg = this.signPanel;
49761         var r = svg.select('#'+ this.svgID + '-svg-r', true).first().dom;
49762         var t = this;
49763
49764         r.addEventListener('mousedown', function(e) { return t.down(e); }, false);
49765         r.addEventListener('mousemove', function(e) { return t.move(e); }, false);
49766         r.addEventListener('mouseup', function(e) { return t.up(e); }, false);
49767         r.addEventListener('mouseout', function(e) { return t.up(e); }, false);
49768         r.addEventListener('touchstart', function(e) { return t.down(e); }, false);
49769         r.addEventListener('touchmove', function(e) { return t.move(e); }, false);
49770         r.addEventListener('touchend', function(e) { return t.up(e); }, false);
49771         
49772     },
49773     isTouchEvent : function(e){
49774         return e.type.match(/^touch/);
49775     },
49776     getCoords : function (e) {
49777         var pt    = this.svgEl.dom.createSVGPoint();
49778         pt.x = e.clientX; 
49779         pt.y = e.clientY;
49780         if (this.isTouchEvent(e)) {
49781             pt.x =  e.targetTouches[0].clientX;
49782             pt.y = e.targetTouches[0].clientY;
49783         }
49784         var a = this.svgEl.dom.getScreenCTM();
49785         var b = a.inverse();
49786         var mx = pt.matrixTransform(b);
49787         return mx.x + ',' + mx.y;
49788     },
49789     //mouse event headler 
49790     down : function (e) {
49791         this.signatureTmp += 'M' + this.getCoords(e) + ' ';
49792         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr('d', this.signatureTmp);
49793         
49794         this.isMouseDown = true;
49795         
49796         e.preventDefault();
49797     },
49798     move : function (e) {
49799         if (this.isMouseDown) {
49800             this.signatureTmp += 'L' + this.getCoords(e) + ' ';
49801             this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', this.signatureTmp);
49802         }
49803         
49804         e.preventDefault();
49805     },
49806     up : function (e) {
49807         this.isMouseDown = false;
49808         var sp = this.signatureTmp.split(' ');
49809         
49810         if(sp.length > 1){
49811             if(!sp[sp.length-2].match(/^L/)){
49812                 sp.pop();
49813                 sp.pop();
49814                 sp.push("");
49815                 this.signatureTmp = sp.join(" ");
49816             }
49817         }
49818         if(this.getValue() != this.signatureTmp){
49819             this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
49820             this.isConfirmed = false;
49821         }
49822         e.preventDefault();
49823     },
49824     
49825     /**
49826      * Protected method that will not generally be called directly. It
49827      * is called when the editor creates its toolbar. Override this method if you need to
49828      * add custom toolbar buttons.
49829      * @param {HtmlEditor} editor
49830      */
49831     createToolbar : function(editor){
49832          function btn(id, toggle, handler){
49833             var xid = fid + '-'+ id ;
49834             return {
49835                 id : xid,
49836                 cmd : id,
49837                 cls : 'x-btn-icon x-edit-'+id,
49838                 enableToggle:toggle !== false,
49839                 scope: editor, // was editor...
49840                 handler:handler||editor.relayBtnCmd,
49841                 clickEvent:'mousedown',
49842                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
49843                 tabIndex:-1
49844             };
49845         }
49846         
49847         
49848         var tb = new Roo.Toolbar(editor.wrap.dom.firstChild);
49849         this.tb = tb;
49850         this.tb.add(
49851            {
49852                 cls : ' x-signature-btn x-signature-'+id,
49853                 scope: editor, // was editor...
49854                 handler: this.reset,
49855                 clickEvent:'mousedown',
49856                 text: this.labels.clear
49857             },
49858             {
49859                  xtype : 'Fill',
49860                  xns: Roo.Toolbar
49861             }, 
49862             {
49863                 cls : '  x-signature-btn x-signature-'+id,
49864                 scope: editor, // was editor...
49865                 handler: this.confirmHandler,
49866                 clickEvent:'mousedown',
49867                 text: this.labels.confirm
49868             }
49869         );
49870     
49871     },
49872     //public
49873     /**
49874      * when user is clicked confirm then show this image.....
49875      * 
49876      * @return {String} Image Data URI
49877      */
49878     getImageDataURI : function(){
49879         var svg = this.svgEl.dom.parentNode.innerHTML;
49880         var src = 'data:image/svg+xml;base64,'+window.btoa(svg);
49881         return src; 
49882     },
49883     /**
49884      * 
49885      * @return {Boolean} this.isConfirmed
49886      */
49887     getConfirmed : function(){
49888         return this.isConfirmed;
49889     },
49890     /**
49891      * 
49892      * @return {Number} this.width
49893      */
49894     getWidth : function(){
49895         return this.width;
49896     },
49897     /**
49898      * 
49899      * @return {Number} this.height
49900      */
49901     getHeight : function(){
49902         return this.height;
49903     },
49904     // private
49905     getSignature : function(){
49906         return this.signatureTmp;
49907     },
49908     // private
49909     reset : function(){
49910         this.signatureTmp = '';
49911         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
49912         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', '');
49913         this.isConfirmed = false;
49914         Roo.form.Signature.superclass.reset.call(this);
49915     },
49916     setSignature : function(s){
49917         this.signatureTmp = s;
49918         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
49919         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', s);
49920         this.setValue(s);
49921         this.isConfirmed = false;
49922         Roo.form.Signature.superclass.reset.call(this);
49923     }, 
49924     test : function(){
49925 //        Roo.log(this.signPanel.dom.contentWindow.up())
49926     },
49927     //private
49928     setConfirmed : function(){
49929         
49930         
49931         
49932 //        Roo.log(Roo.get(this.signPanel.dom.contentWindow.r).attr('fill', '#cfc'));
49933     },
49934     // private
49935     confirmHandler : function(){
49936         if(!this.getSignature()){
49937             return;
49938         }
49939         
49940         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#cfc');
49941         this.setValue(this.getSignature());
49942         this.isConfirmed = true;
49943         
49944         this.fireEvent('confirm', this);
49945     },
49946     // private
49947     // Subclasses should provide the validation implementation by overriding this
49948     validateValue : function(value){
49949         if(this.allowBlank){
49950             return true;
49951         }
49952         
49953         if(this.isConfirmed){
49954             return true;
49955         }
49956         return false;
49957     }
49958 });/*
49959  * Based on:
49960  * Ext JS Library 1.1.1
49961  * Copyright(c) 2006-2007, Ext JS, LLC.
49962  *
49963  * Originally Released Under LGPL - original licence link has changed is not relivant.
49964  *
49965  * Fork - LGPL
49966  * <script type="text/javascript">
49967  */
49968  
49969
49970 /**
49971  * @class Roo.form.ComboBox
49972  * @extends Roo.form.TriggerField
49973  * A combobox control with support for autocomplete, remote-loading, paging and many other features.
49974  * @constructor
49975  * Create a new ComboBox.
49976  * @param {Object} config Configuration options
49977  */
49978 Roo.form.Select = function(config){
49979     Roo.form.Select.superclass.constructor.call(this, config);
49980      
49981 };
49982
49983 Roo.extend(Roo.form.Select , Roo.form.ComboBox, {
49984     /**
49985      * @cfg {String/HTMLElement/Element} transform The id, DOM node or element of an existing select to convert to a ComboBox
49986      */
49987     /**
49988      * @cfg {Boolean} lazyRender True to prevent the ComboBox from rendering until requested (should always be used when
49989      * rendering into an Roo.Editor, defaults to false)
49990      */
49991     /**
49992      * @cfg {Boolean/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to:
49993      * {tag: "input", type: "text", size: "24", autocomplete: "off"})
49994      */
49995     /**
49996      * @cfg {Roo.data.Store} store The data store to which this combo is bound (defaults to undefined)
49997      */
49998     /**
49999      * @cfg {String} title If supplied, a header element is created containing this text and added into the top of
50000      * the dropdown list (defaults to undefined, with no header element)
50001      */
50002
50003      /**
50004      * @cfg {String/Roo.Template} tpl The template to use to render the output
50005      */
50006      
50007     // private
50008     defaultAutoCreate : {tag: "select"  },
50009     /**
50010      * @cfg {Number} listWidth The width in pixels of the dropdown list (defaults to the width of the ComboBox field)
50011      */
50012     listWidth: undefined,
50013     /**
50014      * @cfg {String} displayField The underlying data field name to bind to this CombBox (defaults to undefined if
50015      * mode = 'remote' or 'text' if mode = 'local')
50016      */
50017     displayField: undefined,
50018     /**
50019      * @cfg {String} valueField The underlying data value name to bind to this CombBox (defaults to undefined if
50020      * mode = 'remote' or 'value' if mode = 'local'). 
50021      * Note: use of a valueField requires the user make a selection
50022      * in order for a value to be mapped.
50023      */
50024     valueField: undefined,
50025     
50026     
50027     /**
50028      * @cfg {String} hiddenName If specified, a hidden form field with this name is dynamically generated to store the
50029      * field's data value (defaults to the underlying DOM element's name)
50030      */
50031     hiddenName: undefined,
50032     /**
50033      * @cfg {String} listClass CSS class to apply to the dropdown list element (defaults to '')
50034      */
50035     listClass: '',
50036     /**
50037      * @cfg {String} selectedClass CSS class to apply to the selected item in the dropdown list (defaults to 'x-combo-selected')
50038      */
50039     selectedClass: 'x-combo-selected',
50040     /**
50041      * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
50042      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-arrow-trigger'
50043      * which displays a downward arrow icon).
50044      */
50045     triggerClass : 'x-form-arrow-trigger',
50046     /**
50047      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
50048      */
50049     shadow:'sides',
50050     /**
50051      * @cfg {String} listAlign A valid anchor position value. See {@link Roo.Element#alignTo} for details on supported
50052      * anchor positions (defaults to 'tl-bl')
50053      */
50054     listAlign: 'tl-bl?',
50055     /**
50056      * @cfg {Number} maxHeight The maximum height in pixels of the dropdown list before scrollbars are shown (defaults to 300)
50057      */
50058     maxHeight: 300,
50059     /**
50060      * @cfg {String} triggerAction The action to execute when the trigger field is activated.  Use 'all' to run the
50061      * query specified by the allQuery config option (defaults to 'query')
50062      */
50063     triggerAction: 'query',
50064     /**
50065      * @cfg {Number} minChars The minimum number of characters the user must type before autocomplete and typeahead activate
50066      * (defaults to 4, does not apply if editable = false)
50067      */
50068     minChars : 4,
50069     /**
50070      * @cfg {Boolean} typeAhead True to populate and autoselect the remainder of the text being typed after a configurable
50071      * delay (typeAheadDelay) if it matches a known value (defaults to false)
50072      */
50073     typeAhead: false,
50074     /**
50075      * @cfg {Number} queryDelay The length of time in milliseconds to delay between the start of typing and sending the
50076      * query to filter the dropdown list (defaults to 500 if mode = 'remote' or 10 if mode = 'local')
50077      */
50078     queryDelay: 500,
50079     /**
50080      * @cfg {Number} pageSize If greater than 0, a paging toolbar is displayed in the footer of the dropdown list and the
50081      * filter queries will execute with page start and limit parameters.  Only applies when mode = 'remote' (defaults to 0)
50082      */
50083     pageSize: 0,
50084     /**
50085      * @cfg {Boolean} selectOnFocus True to select any existing text in the field immediately on focus.  Only applies
50086      * when editable = true (defaults to false)
50087      */
50088     selectOnFocus:false,
50089     /**
50090      * @cfg {String} queryParam Name of the query as it will be passed on the querystring (defaults to 'query')
50091      */
50092     queryParam: 'query',
50093     /**
50094      * @cfg {String} loadingText The text to display in the dropdown list while data is loading.  Only applies
50095      * when mode = 'remote' (defaults to 'Loading...')
50096      */
50097     loadingText: 'Loading...',
50098     /**
50099      * @cfg {Boolean} resizable True to add a resize handle to the bottom of the dropdown list (defaults to false)
50100      */
50101     resizable: false,
50102     /**
50103      * @cfg {Number} handleHeight The height in pixels of the dropdown list resize handle if resizable = true (defaults to 8)
50104      */
50105     handleHeight : 8,
50106     /**
50107      * @cfg {Boolean} editable False to prevent the user from typing text directly into the field, just like a
50108      * traditional select (defaults to true)
50109      */
50110     editable: true,
50111     /**
50112      * @cfg {String} allQuery The text query to send to the server to return all records for the list with no filtering (defaults to '')
50113      */
50114     allQuery: '',
50115     /**
50116      * @cfg {String} mode Set to 'local' if the ComboBox loads local data (defaults to 'remote' which loads from the server)
50117      */
50118     mode: 'remote',
50119     /**
50120      * @cfg {Number} minListWidth The minimum width of the dropdown list in pixels (defaults to 70, will be ignored if
50121      * listWidth has a higher value)
50122      */
50123     minListWidth : 70,
50124     /**
50125      * @cfg {Boolean} forceSelection True to restrict the selected value to one of the values in the list, false to
50126      * allow the user to set arbitrary text into the field (defaults to false)
50127      */
50128     forceSelection:false,
50129     /**
50130      * @cfg {Number} typeAheadDelay The length of time in milliseconds to wait until the typeahead text is displayed
50131      * if typeAhead = true (defaults to 250)
50132      */
50133     typeAheadDelay : 250,
50134     /**
50135      * @cfg {String} valueNotFoundText When using a name/value combo, if the value passed to setValue is not found in
50136      * the store, valueNotFoundText will be displayed as the field text if defined (defaults to undefined)
50137      */
50138     valueNotFoundText : undefined,
50139     
50140     /**
50141      * @cfg {String} defaultValue The value displayed after loading the store.
50142      */
50143     defaultValue: '',
50144     
50145     /**
50146      * @cfg {Boolean} blockFocus Prevents all focus calls, so it can work with things like HTML edtor bar
50147      */
50148     blockFocus : false,
50149     
50150     /**
50151      * @cfg {Boolean} disableClear Disable showing of clear button.
50152      */
50153     disableClear : false,
50154     /**
50155      * @cfg {Boolean} alwaysQuery  Disable caching of results, and always send query
50156      */
50157     alwaysQuery : false,
50158     
50159     //private
50160     addicon : false,
50161     editicon: false,
50162     
50163     // element that contains real text value.. (when hidden is used..)
50164      
50165     // private
50166     onRender : function(ct, position){
50167         Roo.form.Field.prototype.onRender.call(this, ct, position);
50168         
50169         if(this.store){
50170             this.store.on('beforeload', this.onBeforeLoad, this);
50171             this.store.on('load', this.onLoad, this);
50172             this.store.on('loadexception', this.onLoadException, this);
50173             this.store.load({});
50174         }
50175         
50176         
50177         
50178     },
50179
50180     // private
50181     initEvents : function(){
50182         //Roo.form.ComboBox.superclass.initEvents.call(this);
50183  
50184     },
50185
50186     onDestroy : function(){
50187        
50188         if(this.store){
50189             this.store.un('beforeload', this.onBeforeLoad, this);
50190             this.store.un('load', this.onLoad, this);
50191             this.store.un('loadexception', this.onLoadException, this);
50192         }
50193         //Roo.form.ComboBox.superclass.onDestroy.call(this);
50194     },
50195
50196     // private
50197     fireKey : function(e){
50198         if(e.isNavKeyPress() && !this.list.isVisible()){
50199             this.fireEvent("specialkey", this, e);
50200         }
50201     },
50202
50203     // private
50204     onResize: function(w, h){
50205         
50206         return; 
50207     
50208         
50209     },
50210
50211     /**
50212      * Allow or prevent the user from directly editing the field text.  If false is passed,
50213      * the user will only be able to select from the items defined in the dropdown list.  This method
50214      * is the runtime equivalent of setting the 'editable' config option at config time.
50215      * @param {Boolean} value True to allow the user to directly edit the field text
50216      */
50217     setEditable : function(value){
50218          
50219     },
50220
50221     // private
50222     onBeforeLoad : function(){
50223         
50224         Roo.log("Select before load");
50225         return;
50226     
50227         this.innerList.update(this.loadingText ?
50228                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
50229         //this.restrictHeight();
50230         this.selectedIndex = -1;
50231     },
50232
50233     // private
50234     onLoad : function(){
50235
50236     
50237         var dom = this.el.dom;
50238         dom.innerHTML = '';
50239          var od = dom.ownerDocument;
50240          
50241         if (this.emptyText) {
50242             var op = od.createElement('option');
50243             op.setAttribute('value', '');
50244             op.innerHTML = String.format('{0}', this.emptyText);
50245             dom.appendChild(op);
50246         }
50247         if(this.store.getCount() > 0){
50248            
50249             var vf = this.valueField;
50250             var df = this.displayField;
50251             this.store.data.each(function(r) {
50252                 // which colmsn to use... testing - cdoe / title..
50253                 var op = od.createElement('option');
50254                 op.setAttribute('value', r.data[vf]);
50255                 op.innerHTML = String.format('{0}', r.data[df]);
50256                 dom.appendChild(op);
50257             });
50258             if (typeof(this.defaultValue != 'undefined')) {
50259                 this.setValue(this.defaultValue);
50260             }
50261             
50262              
50263         }else{
50264             //this.onEmptyResults();
50265         }
50266         //this.el.focus();
50267     },
50268     // private
50269     onLoadException : function()
50270     {
50271         dom.innerHTML = '';
50272             
50273         Roo.log("Select on load exception");
50274         return;
50275     
50276         this.collapse();
50277         Roo.log(this.store.reader.jsonData);
50278         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
50279             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
50280         }
50281         
50282         
50283     },
50284     // private
50285     onTypeAhead : function(){
50286          
50287     },
50288
50289     // private
50290     onSelect : function(record, index){
50291         Roo.log('on select?');
50292         return;
50293         if(this.fireEvent('beforeselect', this, record, index) !== false){
50294             this.setFromData(index > -1 ? record.data : false);
50295             this.collapse();
50296             this.fireEvent('select', this, record, index);
50297         }
50298     },
50299
50300     /**
50301      * Returns the currently selected field value or empty string if no value is set.
50302      * @return {String} value The selected value
50303      */
50304     getValue : function(){
50305         var dom = this.el.dom;
50306         this.value = dom.options[dom.selectedIndex].value;
50307         return this.value;
50308         
50309     },
50310
50311     /**
50312      * Clears any text/value currently set in the field
50313      */
50314     clearValue : function(){
50315         this.value = '';
50316         this.el.dom.selectedIndex = this.emptyText ? 0 : -1;
50317         
50318     },
50319
50320     /**
50321      * Sets the specified value into the field.  If the value finds a match, the corresponding record text
50322      * will be displayed in the field.  If the value does not match the data value of an existing item,
50323      * and the valueNotFoundText config option is defined, it will be displayed as the default field text.
50324      * Otherwise the field will be blank (although the value will still be set).
50325      * @param {String} value The value to match
50326      */
50327     setValue : function(v){
50328         var d = this.el.dom;
50329         for (var i =0; i < d.options.length;i++) {
50330             if (v == d.options[i].value) {
50331                 d.selectedIndex = i;
50332                 this.value = v;
50333                 return;
50334             }
50335         }
50336         this.clearValue();
50337     },
50338     /**
50339      * @property {Object} the last set data for the element
50340      */
50341     
50342     lastData : false,
50343     /**
50344      * Sets the value of the field based on a object which is related to the record format for the store.
50345      * @param {Object} value the value to set as. or false on reset?
50346      */
50347     setFromData : function(o){
50348         Roo.log('setfrom data?');
50349          
50350         
50351         
50352     },
50353     // private
50354     reset : function(){
50355         this.clearValue();
50356     },
50357     // private
50358     findRecord : function(prop, value){
50359         
50360         return false;
50361     
50362         var record;
50363         if(this.store.getCount() > 0){
50364             this.store.each(function(r){
50365                 if(r.data[prop] == value){
50366                     record = r;
50367                     return false;
50368                 }
50369                 return true;
50370             });
50371         }
50372         return record;
50373     },
50374     
50375     getName: function()
50376     {
50377         // returns hidden if it's set..
50378         if (!this.rendered) {return ''};
50379         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
50380         
50381     },
50382      
50383
50384     
50385
50386     // private
50387     onEmptyResults : function(){
50388         Roo.log('empty results');
50389         //this.collapse();
50390     },
50391
50392     /**
50393      * Returns true if the dropdown list is expanded, else false.
50394      */
50395     isExpanded : function(){
50396         return false;
50397     },
50398
50399     /**
50400      * Select an item in the dropdown list by its data value. This function does NOT cause the select event to fire.
50401      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
50402      * @param {String} value The data value of the item to select
50403      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
50404      * selected item if it is not currently in view (defaults to true)
50405      * @return {Boolean} True if the value matched an item in the list, else false
50406      */
50407     selectByValue : function(v, scrollIntoView){
50408         Roo.log('select By Value');
50409         return false;
50410     
50411         if(v !== undefined && v !== null){
50412             var r = this.findRecord(this.valueField || this.displayField, v);
50413             if(r){
50414                 this.select(this.store.indexOf(r), scrollIntoView);
50415                 return true;
50416             }
50417         }
50418         return false;
50419     },
50420
50421     /**
50422      * Select an item in the dropdown list by its numeric index in the list. This function does NOT cause the select event to fire.
50423      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
50424      * @param {Number} index The zero-based index of the list item to select
50425      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
50426      * selected item if it is not currently in view (defaults to true)
50427      */
50428     select : function(index, scrollIntoView){
50429         Roo.log('select ');
50430         return  ;
50431         
50432         this.selectedIndex = index;
50433         this.view.select(index);
50434         if(scrollIntoView !== false){
50435             var el = this.view.getNode(index);
50436             if(el){
50437                 this.innerList.scrollChildIntoView(el, false);
50438             }
50439         }
50440     },
50441
50442       
50443
50444     // private
50445     validateBlur : function(){
50446         
50447         return;
50448         
50449     },
50450
50451     // private
50452     initQuery : function(){
50453         this.doQuery(this.getRawValue());
50454     },
50455
50456     // private
50457     doForce : function(){
50458         if(this.el.dom.value.length > 0){
50459             this.el.dom.value =
50460                 this.lastSelectionText === undefined ? '' : this.lastSelectionText;
50461              
50462         }
50463     },
50464
50465     /**
50466      * Execute a query to filter the dropdown list.  Fires the beforequery event prior to performing the
50467      * query allowing the query action to be canceled if needed.
50468      * @param {String} query The SQL query to execute
50469      * @param {Boolean} forceAll True to force the query to execute even if there are currently fewer characters
50470      * in the field than the minimum specified by the minChars config option.  It also clears any filter previously
50471      * saved in the current store (defaults to false)
50472      */
50473     doQuery : function(q, forceAll){
50474         
50475         Roo.log('doQuery?');
50476         if(q === undefined || q === null){
50477             q = '';
50478         }
50479         var qe = {
50480             query: q,
50481             forceAll: forceAll,
50482             combo: this,
50483             cancel:false
50484         };
50485         if(this.fireEvent('beforequery', qe)===false || qe.cancel){
50486             return false;
50487         }
50488         q = qe.query;
50489         forceAll = qe.forceAll;
50490         if(forceAll === true || (q.length >= this.minChars)){
50491             if(this.lastQuery != q || this.alwaysQuery){
50492                 this.lastQuery = q;
50493                 if(this.mode == 'local'){
50494                     this.selectedIndex = -1;
50495                     if(forceAll){
50496                         this.store.clearFilter();
50497                     }else{
50498                         this.store.filter(this.displayField, q);
50499                     }
50500                     this.onLoad();
50501                 }else{
50502                     this.store.baseParams[this.queryParam] = q;
50503                     this.store.load({
50504                         params: this.getParams(q)
50505                     });
50506                     this.expand();
50507                 }
50508             }else{
50509                 this.selectedIndex = -1;
50510                 this.onLoad();   
50511             }
50512         }
50513     },
50514
50515     // private
50516     getParams : function(q){
50517         var p = {};
50518         //p[this.queryParam] = q;
50519         if(this.pageSize){
50520             p.start = 0;
50521             p.limit = this.pageSize;
50522         }
50523         return p;
50524     },
50525
50526     /**
50527      * Hides the dropdown list if it is currently expanded. Fires the 'collapse' event on completion.
50528      */
50529     collapse : function(){
50530         
50531     },
50532
50533     // private
50534     collapseIf : function(e){
50535         
50536     },
50537
50538     /**
50539      * Expands the dropdown list if it is currently hidden. Fires the 'expand' event on completion.
50540      */
50541     expand : function(){
50542         
50543     } ,
50544
50545     // private
50546      
50547
50548     /** 
50549     * @cfg {Boolean} grow 
50550     * @hide 
50551     */
50552     /** 
50553     * @cfg {Number} growMin 
50554     * @hide 
50555     */
50556     /** 
50557     * @cfg {Number} growMax 
50558     * @hide 
50559     */
50560     /**
50561      * @hide
50562      * @method autoSize
50563      */
50564     
50565     setWidth : function()
50566     {
50567         
50568     },
50569     getResizeEl : function(){
50570         return this.el;
50571     }
50572 });//<script type="text/javasscript">
50573  
50574
50575 /**
50576  * @class Roo.DDView
50577  * A DnD enabled version of Roo.View.
50578  * @param {Element/String} container The Element in which to create the View.
50579  * @param {String} tpl The template string used to create the markup for each element of the View
50580  * @param {Object} config The configuration properties. These include all the config options of
50581  * {@link Roo.View} plus some specific to this class.<br>
50582  * <p>
50583  * Drag/drop is implemented by adding {@link Roo.data.Record}s to the target DDView. If copying is
50584  * not being performed, the original {@link Roo.data.Record} is removed from the source DDView.<br>
50585  * <p>
50586  * The following extra CSS rules are needed to provide insertion point highlighting:<pre><code>
50587 .x-view-drag-insert-above {
50588         border-top:1px dotted #3366cc;
50589 }
50590 .x-view-drag-insert-below {
50591         border-bottom:1px dotted #3366cc;
50592 }
50593 </code></pre>
50594  * 
50595  */
50596  
50597 Roo.DDView = function(container, tpl, config) {
50598     Roo.DDView.superclass.constructor.apply(this, arguments);
50599     this.getEl().setStyle("outline", "0px none");
50600     this.getEl().unselectable();
50601     if (this.dragGroup) {
50602                 this.setDraggable(this.dragGroup.split(","));
50603     }
50604     if (this.dropGroup) {
50605                 this.setDroppable(this.dropGroup.split(","));
50606     }
50607     if (this.deletable) {
50608         this.setDeletable();
50609     }
50610     this.isDirtyFlag = false;
50611         this.addEvents({
50612                 "drop" : true
50613         });
50614 };
50615
50616 Roo.extend(Roo.DDView, Roo.View, {
50617 /**     @cfg {String/Array} dragGroup The ddgroup name(s) for the View's DragZone. */
50618 /**     @cfg {String/Array} dropGroup The ddgroup name(s) for the View's DropZone. */
50619 /**     @cfg {Boolean} copy Causes drag operations to copy nodes rather than move. */
50620 /**     @cfg {Boolean} allowCopy Causes ctrl/drag operations to copy nodes rather than move. */
50621
50622         isFormField: true,
50623
50624         reset: Roo.emptyFn,
50625         
50626         clearInvalid: Roo.form.Field.prototype.clearInvalid,
50627
50628         validate: function() {
50629                 return true;
50630         },
50631         
50632         destroy: function() {
50633                 this.purgeListeners();
50634                 this.getEl.removeAllListeners();
50635                 this.getEl().remove();
50636                 if (this.dragZone) {
50637                         if (this.dragZone.destroy) {
50638                                 this.dragZone.destroy();
50639                         }
50640                 }
50641                 if (this.dropZone) {
50642                         if (this.dropZone.destroy) {
50643                                 this.dropZone.destroy();
50644                         }
50645                 }
50646         },
50647
50648 /**     Allows this class to be an Roo.form.Field so it can be found using {@link Roo.form.BasicForm#findField}. */
50649         getName: function() {
50650                 return this.name;
50651         },
50652
50653 /**     Loads the View from a JSON string representing the Records to put into the Store. */
50654         setValue: function(v) {
50655                 if (!this.store) {
50656                         throw "DDView.setValue(). DDView must be constructed with a valid Store";
50657                 }
50658                 var data = {};
50659                 data[this.store.reader.meta.root] = v ? [].concat(v) : [];
50660                 this.store.proxy = new Roo.data.MemoryProxy(data);
50661                 this.store.load();
50662         },
50663
50664 /**     @return {String} a parenthesised list of the ids of the Records in the View. */
50665         getValue: function() {
50666                 var result = '(';
50667                 this.store.each(function(rec) {
50668                         result += rec.id + ',';
50669                 });
50670                 return result.substr(0, result.length - 1) + ')';
50671         },
50672         
50673         getIds: function() {
50674                 var i = 0, result = new Array(this.store.getCount());
50675                 this.store.each(function(rec) {
50676                         result[i++] = rec.id;
50677                 });
50678                 return result;
50679         },
50680         
50681         isDirty: function() {
50682                 return this.isDirtyFlag;
50683         },
50684
50685 /**
50686  *      Part of the Roo.dd.DropZone interface. If no target node is found, the
50687  *      whole Element becomes the target, and this causes the drop gesture to append.
50688  */
50689     getTargetFromEvent : function(e) {
50690                 var target = e.getTarget();
50691                 while ((target !== null) && (target.parentNode != this.el.dom)) {
50692                 target = target.parentNode;
50693                 }
50694                 if (!target) {
50695                         target = this.el.dom.lastChild || this.el.dom;
50696                 }
50697                 return target;
50698     },
50699
50700 /**
50701  *      Create the drag data which consists of an object which has the property "ddel" as
50702  *      the drag proxy element. 
50703  */
50704     getDragData : function(e) {
50705         var target = this.findItemFromChild(e.getTarget());
50706                 if(target) {
50707                         this.handleSelection(e);
50708                         var selNodes = this.getSelectedNodes();
50709             var dragData = {
50710                 source: this,
50711                 copy: this.copy || (this.allowCopy && e.ctrlKey),
50712                 nodes: selNodes,
50713                 records: []
50714                         };
50715                         var selectedIndices = this.getSelectedIndexes();
50716                         for (var i = 0; i < selectedIndices.length; i++) {
50717                                 dragData.records.push(this.store.getAt(selectedIndices[i]));
50718                         }
50719                         if (selNodes.length == 1) {
50720                                 dragData.ddel = target.cloneNode(true); // the div element
50721                         } else {
50722                                 var div = document.createElement('div'); // create the multi element drag "ghost"
50723                                 div.className = 'multi-proxy';
50724                                 for (var i = 0, len = selNodes.length; i < len; i++) {
50725                                         div.appendChild(selNodes[i].cloneNode(true));
50726                                 }
50727                                 dragData.ddel = div;
50728                         }
50729             //console.log(dragData)
50730             //console.log(dragData.ddel.innerHTML)
50731                         return dragData;
50732                 }
50733         //console.log('nodragData')
50734                 return false;
50735     },
50736     
50737 /**     Specify to which ddGroup items in this DDView may be dragged. */
50738     setDraggable: function(ddGroup) {
50739         if (ddGroup instanceof Array) {
50740                 Roo.each(ddGroup, this.setDraggable, this);
50741                 return;
50742         }
50743         if (this.dragZone) {
50744                 this.dragZone.addToGroup(ddGroup);
50745         } else {
50746                         this.dragZone = new Roo.dd.DragZone(this.getEl(), {
50747                                 containerScroll: true,
50748                                 ddGroup: ddGroup 
50749
50750                         });
50751 //                      Draggability implies selection. DragZone's mousedown selects the element.
50752                         if (!this.multiSelect) { this.singleSelect = true; }
50753
50754 //                      Wire the DragZone's handlers up to methods in *this*
50755                         this.dragZone.getDragData = this.getDragData.createDelegate(this);
50756                 }
50757     },
50758
50759 /**     Specify from which ddGroup this DDView accepts drops. */
50760     setDroppable: function(ddGroup) {
50761         if (ddGroup instanceof Array) {
50762                 Roo.each(ddGroup, this.setDroppable, this);
50763                 return;
50764         }
50765         if (this.dropZone) {
50766                 this.dropZone.addToGroup(ddGroup);
50767         } else {
50768                         this.dropZone = new Roo.dd.DropZone(this.getEl(), {
50769                                 containerScroll: true,
50770                                 ddGroup: ddGroup
50771                         });
50772
50773 //                      Wire the DropZone's handlers up to methods in *this*
50774                         this.dropZone.getTargetFromEvent = this.getTargetFromEvent.createDelegate(this);
50775                         this.dropZone.onNodeEnter = this.onNodeEnter.createDelegate(this);
50776                         this.dropZone.onNodeOver = this.onNodeOver.createDelegate(this);
50777                         this.dropZone.onNodeOut = this.onNodeOut.createDelegate(this);
50778                         this.dropZone.onNodeDrop = this.onNodeDrop.createDelegate(this);
50779                 }
50780     },
50781
50782 /**     Decide whether to drop above or below a View node. */
50783     getDropPoint : function(e, n, dd){
50784         if (n == this.el.dom) { return "above"; }
50785                 var t = Roo.lib.Dom.getY(n), b = t + n.offsetHeight;
50786                 var c = t + (b - t) / 2;
50787                 var y = Roo.lib.Event.getPageY(e);
50788                 if(y <= c) {
50789                         return "above";
50790                 }else{
50791                         return "below";
50792                 }
50793     },
50794
50795     onNodeEnter : function(n, dd, e, data){
50796                 return false;
50797     },
50798     
50799     onNodeOver : function(n, dd, e, data){
50800                 var pt = this.getDropPoint(e, n, dd);
50801                 // set the insert point style on the target node
50802                 var dragElClass = this.dropNotAllowed;
50803                 if (pt) {
50804                         var targetElClass;
50805                         if (pt == "above"){
50806                                 dragElClass = n.previousSibling ? "x-tree-drop-ok-between" : "x-tree-drop-ok-above";
50807                                 targetElClass = "x-view-drag-insert-above";
50808                         } else {
50809                                 dragElClass = n.nextSibling ? "x-tree-drop-ok-between" : "x-tree-drop-ok-below";
50810                                 targetElClass = "x-view-drag-insert-below";
50811                         }
50812                         if (this.lastInsertClass != targetElClass){
50813                                 Roo.fly(n).replaceClass(this.lastInsertClass, targetElClass);
50814                                 this.lastInsertClass = targetElClass;
50815                         }
50816                 }
50817                 return dragElClass;
50818         },
50819
50820     onNodeOut : function(n, dd, e, data){
50821                 this.removeDropIndicators(n);
50822     },
50823
50824     onNodeDrop : function(n, dd, e, data){
50825         if (this.fireEvent("drop", this, n, dd, e, data) === false) {
50826                 return false;
50827         }
50828         var pt = this.getDropPoint(e, n, dd);
50829                 var insertAt = (n == this.el.dom) ? this.nodes.length : n.nodeIndex;
50830                 if (pt == "below") { insertAt++; }
50831                 for (var i = 0; i < data.records.length; i++) {
50832                         var r = data.records[i];
50833                         var dup = this.store.getById(r.id);
50834                         if (dup && (dd != this.dragZone)) {
50835                                 Roo.fly(this.getNode(this.store.indexOf(dup))).frame("red", 1);
50836                         } else {
50837                                 if (data.copy) {
50838                                         this.store.insert(insertAt++, r.copy());
50839                                 } else {
50840                                         data.source.isDirtyFlag = true;
50841                                         r.store.remove(r);
50842                                         this.store.insert(insertAt++, r);
50843                                 }
50844                                 this.isDirtyFlag = true;
50845                         }
50846                 }
50847                 this.dragZone.cachedTarget = null;
50848                 return true;
50849     },
50850
50851     removeDropIndicators : function(n){
50852                 if(n){
50853                         Roo.fly(n).removeClass([
50854                                 "x-view-drag-insert-above",
50855                                 "x-view-drag-insert-below"]);
50856                         this.lastInsertClass = "_noclass";
50857                 }
50858     },
50859
50860 /**
50861  *      Utility method. Add a delete option to the DDView's context menu.
50862  *      @param {String} imageUrl The URL of the "delete" icon image.
50863  */
50864         setDeletable: function(imageUrl) {
50865                 if (!this.singleSelect && !this.multiSelect) {
50866                         this.singleSelect = true;
50867                 }
50868                 var c = this.getContextMenu();
50869                 this.contextMenu.on("itemclick", function(item) {
50870                         switch (item.id) {
50871                                 case "delete":
50872                                         this.remove(this.getSelectedIndexes());
50873                                         break;
50874                         }
50875                 }, this);
50876                 this.contextMenu.add({
50877                         icon: imageUrl,
50878                         id: "delete",
50879                         text: 'Delete'
50880                 });
50881         },
50882         
50883 /**     Return the context menu for this DDView. */
50884         getContextMenu: function() {
50885                 if (!this.contextMenu) {
50886 //                      Create the View's context menu
50887                         this.contextMenu = new Roo.menu.Menu({
50888                                 id: this.id + "-contextmenu"
50889                         });
50890                         this.el.on("contextmenu", this.showContextMenu, this);
50891                 }
50892                 return this.contextMenu;
50893         },
50894         
50895         disableContextMenu: function() {
50896                 if (this.contextMenu) {
50897                         this.el.un("contextmenu", this.showContextMenu, this);
50898                 }
50899         },
50900
50901         showContextMenu: function(e, item) {
50902         item = this.findItemFromChild(e.getTarget());
50903                 if (item) {
50904                         e.stopEvent();
50905                         this.select(this.getNode(item), this.multiSelect && e.ctrlKey, true);
50906                         this.contextMenu.showAt(e.getXY());
50907             }
50908     },
50909
50910 /**
50911  *      Remove {@link Roo.data.Record}s at the specified indices.
50912  *      @param {Array/Number} selectedIndices The index (or Array of indices) of Records to remove.
50913  */
50914     remove: function(selectedIndices) {
50915                 selectedIndices = [].concat(selectedIndices);
50916                 for (var i = 0; i < selectedIndices.length; i++) {
50917                         var rec = this.store.getAt(selectedIndices[i]);
50918                         this.store.remove(rec);
50919                 }
50920     },
50921
50922 /**
50923  *      Double click fires the event, but also, if this is draggable, and there is only one other
50924  *      related DropZone, it transfers the selected node.
50925  */
50926     onDblClick : function(e){
50927         var item = this.findItemFromChild(e.getTarget());
50928         if(item){
50929             if (this.fireEvent("dblclick", this, this.indexOf(item), item, e) === false) {
50930                 return false;
50931             }
50932             if (this.dragGroup) {
50933                     var targets = Roo.dd.DragDropMgr.getRelated(this.dragZone, true);
50934                     while (targets.indexOf(this.dropZone) > -1) {
50935                             targets.remove(this.dropZone);
50936                                 }
50937                     if (targets.length == 1) {
50938                                         this.dragZone.cachedTarget = null;
50939                         var el = Roo.get(targets[0].getEl());
50940                         var box = el.getBox(true);
50941                         targets[0].onNodeDrop(el.dom, {
50942                                 target: el.dom,
50943                                 xy: [box.x, box.y + box.height - 1]
50944                         }, null, this.getDragData(e));
50945                     }
50946                 }
50947         }
50948     },
50949     
50950     handleSelection: function(e) {
50951                 this.dragZone.cachedTarget = null;
50952         var item = this.findItemFromChild(e.getTarget());
50953         if (!item) {
50954                 this.clearSelections(true);
50955                 return;
50956         }
50957                 if (item && (this.multiSelect || this.singleSelect)){
50958                         if(this.multiSelect && e.shiftKey && (!e.ctrlKey) && this.lastSelection){
50959                                 this.select(this.getNodes(this.indexOf(this.lastSelection), item.nodeIndex), false);
50960                         }else if (this.isSelected(this.getNode(item)) && e.ctrlKey){
50961                                 this.unselect(item);
50962                         } else {
50963                                 this.select(item, this.multiSelect && e.ctrlKey);
50964                                 this.lastSelection = item;
50965                         }
50966                 }
50967     },
50968
50969     onItemClick : function(item, index, e){
50970                 if(this.fireEvent("beforeclick", this, index, item, e) === false){
50971                         return false;
50972                 }
50973                 return true;
50974     },
50975
50976     unselect : function(nodeInfo, suppressEvent){
50977                 var node = this.getNode(nodeInfo);
50978                 if(node && this.isSelected(node)){
50979                         if(this.fireEvent("beforeselect", this, node, this.selections) !== false){
50980                                 Roo.fly(node).removeClass(this.selectedClass);
50981                                 this.selections.remove(node);
50982                                 if(!suppressEvent){
50983                                         this.fireEvent("selectionchange", this, this.selections);
50984                                 }
50985                         }
50986                 }
50987     }
50988 });
50989 /*
50990  * Based on:
50991  * Ext JS Library 1.1.1
50992  * Copyright(c) 2006-2007, Ext JS, LLC.
50993  *
50994  * Originally Released Under LGPL - original licence link has changed is not relivant.
50995  *
50996  * Fork - LGPL
50997  * <script type="text/javascript">
50998  */
50999  
51000 /**
51001  * @class Roo.LayoutManager
51002  * @extends Roo.util.Observable
51003  * Base class for layout managers.
51004  */
51005 Roo.LayoutManager = function(container, config){
51006     Roo.LayoutManager.superclass.constructor.call(this);
51007     this.el = Roo.get(container);
51008     // ie scrollbar fix
51009     if(this.el.dom == document.body && Roo.isIE && !config.allowScroll){
51010         document.body.scroll = "no";
51011     }else if(this.el.dom != document.body && this.el.getStyle('position') == 'static'){
51012         this.el.position('relative');
51013     }
51014     this.id = this.el.id;
51015     this.el.addClass("x-layout-container");
51016     /** false to disable window resize monitoring @type Boolean */
51017     this.monitorWindowResize = true;
51018     this.regions = {};
51019     this.addEvents({
51020         /**
51021          * @event layout
51022          * Fires when a layout is performed. 
51023          * @param {Roo.LayoutManager} this
51024          */
51025         "layout" : true,
51026         /**
51027          * @event regionresized
51028          * Fires when the user resizes a region. 
51029          * @param {Roo.LayoutRegion} region The resized region
51030          * @param {Number} newSize The new size (width for east/west, height for north/south)
51031          */
51032         "regionresized" : true,
51033         /**
51034          * @event regioncollapsed
51035          * Fires when a region is collapsed. 
51036          * @param {Roo.LayoutRegion} region The collapsed region
51037          */
51038         "regioncollapsed" : true,
51039         /**
51040          * @event regionexpanded
51041          * Fires when a region is expanded.  
51042          * @param {Roo.LayoutRegion} region The expanded region
51043          */
51044         "regionexpanded" : true
51045     });
51046     this.updating = false;
51047     Roo.EventManager.onWindowResize(this.onWindowResize, this, true);
51048 };
51049
51050 Roo.extend(Roo.LayoutManager, Roo.util.Observable, {
51051     /**
51052      * Returns true if this layout is currently being updated
51053      * @return {Boolean}
51054      */
51055     isUpdating : function(){
51056         return this.updating; 
51057     },
51058     
51059     /**
51060      * Suspend the LayoutManager from doing auto-layouts while
51061      * making multiple add or remove calls
51062      */
51063     beginUpdate : function(){
51064         this.updating = true;    
51065     },
51066     
51067     /**
51068      * Restore auto-layouts and optionally disable the manager from performing a layout
51069      * @param {Boolean} noLayout true to disable a layout update 
51070      */
51071     endUpdate : function(noLayout){
51072         this.updating = false;
51073         if(!noLayout){
51074             this.layout();
51075         }    
51076     },
51077     
51078     layout: function(){
51079         
51080     },
51081     
51082     onRegionResized : function(region, newSize){
51083         this.fireEvent("regionresized", region, newSize);
51084         this.layout();
51085     },
51086     
51087     onRegionCollapsed : function(region){
51088         this.fireEvent("regioncollapsed", region);
51089     },
51090     
51091     onRegionExpanded : function(region){
51092         this.fireEvent("regionexpanded", region);
51093     },
51094         
51095     /**
51096      * Returns the size of the current view. This method normalizes document.body and element embedded layouts and
51097      * performs box-model adjustments.
51098      * @return {Object} The size as an object {width: (the width), height: (the height)}
51099      */
51100     getViewSize : function(){
51101         var size;
51102         if(this.el.dom != document.body){
51103             size = this.el.getSize();
51104         }else{
51105             size = {width: Roo.lib.Dom.getViewWidth(), height: Roo.lib.Dom.getViewHeight()};
51106         }
51107         size.width -= this.el.getBorderWidth("lr")-this.el.getPadding("lr");
51108         size.height -= this.el.getBorderWidth("tb")-this.el.getPadding("tb");
51109         return size;
51110     },
51111     
51112     /**
51113      * Returns the Element this layout is bound to.
51114      * @return {Roo.Element}
51115      */
51116     getEl : function(){
51117         return this.el;
51118     },
51119     
51120     /**
51121      * Returns the specified region.
51122      * @param {String} target The region key ('center', 'north', 'south', 'east' or 'west')
51123      * @return {Roo.LayoutRegion}
51124      */
51125     getRegion : function(target){
51126         return this.regions[target.toLowerCase()];
51127     },
51128     
51129     onWindowResize : function(){
51130         if(this.monitorWindowResize){
51131             this.layout();
51132         }
51133     }
51134 });/*
51135  * Based on:
51136  * Ext JS Library 1.1.1
51137  * Copyright(c) 2006-2007, Ext JS, LLC.
51138  *
51139  * Originally Released Under LGPL - original licence link has changed is not relivant.
51140  *
51141  * Fork - LGPL
51142  * <script type="text/javascript">
51143  */
51144 /**
51145  * @class Roo.BorderLayout
51146  * @extends Roo.LayoutManager
51147  * This class represents a common layout manager used in desktop applications. For screenshots and more details,
51148  * please see: <br><br>
51149  * <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>
51150  * <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>
51151  * Example:
51152  <pre><code>
51153  var layout = new Roo.BorderLayout(document.body, {
51154     north: {
51155         initialSize: 25,
51156         titlebar: false
51157     },
51158     west: {
51159         split:true,
51160         initialSize: 200,
51161         minSize: 175,
51162         maxSize: 400,
51163         titlebar: true,
51164         collapsible: true
51165     },
51166     east: {
51167         split:true,
51168         initialSize: 202,
51169         minSize: 175,
51170         maxSize: 400,
51171         titlebar: true,
51172         collapsible: true
51173     },
51174     south: {
51175         split:true,
51176         initialSize: 100,
51177         minSize: 100,
51178         maxSize: 200,
51179         titlebar: true,
51180         collapsible: true
51181     },
51182     center: {
51183         titlebar: true,
51184         autoScroll:true,
51185         resizeTabs: true,
51186         minTabWidth: 50,
51187         preferredTabWidth: 150
51188     }
51189 });
51190
51191 // shorthand
51192 var CP = Roo.ContentPanel;
51193
51194 layout.beginUpdate();
51195 layout.add("north", new CP("north", "North"));
51196 layout.add("south", new CP("south", {title: "South", closable: true}));
51197 layout.add("west", new CP("west", {title: "West"}));
51198 layout.add("east", new CP("autoTabs", {title: "Auto Tabs", closable: true}));
51199 layout.add("center", new CP("center1", {title: "Close Me", closable: true}));
51200 layout.add("center", new CP("center2", {title: "Center Panel", closable: false}));
51201 layout.getRegion("center").showPanel("center1");
51202 layout.endUpdate();
51203 </code></pre>
51204
51205 <b>The container the layout is rendered into can be either the body element or any other element.
51206 If it is not the body element, the container needs to either be an absolute positioned element,
51207 or you will need to add "position:relative" to the css of the container.  You will also need to specify
51208 the container size if it is not the body element.</b>
51209
51210 * @constructor
51211 * Create a new BorderLayout
51212 * @param {String/HTMLElement/Element} container The container this layout is bound to
51213 * @param {Object} config Configuration options
51214  */
51215 Roo.BorderLayout = function(container, config){
51216     config = config || {};
51217     Roo.BorderLayout.superclass.constructor.call(this, container, config);
51218     this.factory = config.factory || Roo.BorderLayout.RegionFactory;
51219     for(var i = 0, len = this.factory.validRegions.length; i < len; i++) {
51220         var target = this.factory.validRegions[i];
51221         if(config[target]){
51222             this.addRegion(target, config[target]);
51223         }
51224     }
51225 };
51226
51227 Roo.extend(Roo.BorderLayout, Roo.LayoutManager, {
51228     /**
51229      * Creates and adds a new region if it doesn't already exist.
51230      * @param {String} target The target region key (north, south, east, west or center).
51231      * @param {Object} config The regions config object
51232      * @return {BorderLayoutRegion} The new region
51233      */
51234     addRegion : function(target, config){
51235         if(!this.regions[target]){
51236             var r = this.factory.create(target, this, config);
51237             this.bindRegion(target, r);
51238         }
51239         return this.regions[target];
51240     },
51241
51242     // private (kinda)
51243     bindRegion : function(name, r){
51244         this.regions[name] = r;
51245         r.on("visibilitychange", this.layout, this);
51246         r.on("paneladded", this.layout, this);
51247         r.on("panelremoved", this.layout, this);
51248         r.on("invalidated", this.layout, this);
51249         r.on("resized", this.onRegionResized, this);
51250         r.on("collapsed", this.onRegionCollapsed, this);
51251         r.on("expanded", this.onRegionExpanded, this);
51252     },
51253
51254     /**
51255      * Performs a layout update.
51256      */
51257     layout : function(){
51258         if(this.updating) {
51259             return;
51260         }
51261         var size = this.getViewSize();
51262         var w = size.width;
51263         var h = size.height;
51264         var centerW = w;
51265         var centerH = h;
51266         var centerY = 0;
51267         var centerX = 0;
51268         //var x = 0, y = 0;
51269
51270         var rs = this.regions;
51271         var north = rs["north"];
51272         var south = rs["south"]; 
51273         var west = rs["west"];
51274         var east = rs["east"];
51275         var center = rs["center"];
51276         //if(this.hideOnLayout){ // not supported anymore
51277             //c.el.setStyle("display", "none");
51278         //}
51279         if(north && north.isVisible()){
51280             var b = north.getBox();
51281             var m = north.getMargins();
51282             b.width = w - (m.left+m.right);
51283             b.x = m.left;
51284             b.y = m.top;
51285             centerY = b.height + b.y + m.bottom;
51286             centerH -= centerY;
51287             north.updateBox(this.safeBox(b));
51288         }
51289         if(south && south.isVisible()){
51290             var b = south.getBox();
51291             var m = south.getMargins();
51292             b.width = w - (m.left+m.right);
51293             b.x = m.left;
51294             var totalHeight = (b.height + m.top + m.bottom);
51295             b.y = h - totalHeight + m.top;
51296             centerH -= totalHeight;
51297             south.updateBox(this.safeBox(b));
51298         }
51299         if(west && west.isVisible()){
51300             var b = west.getBox();
51301             var m = west.getMargins();
51302             b.height = centerH - (m.top+m.bottom);
51303             b.x = m.left;
51304             b.y = centerY + m.top;
51305             var totalWidth = (b.width + m.left + m.right);
51306             centerX += totalWidth;
51307             centerW -= totalWidth;
51308             west.updateBox(this.safeBox(b));
51309         }
51310         if(east && east.isVisible()){
51311             var b = east.getBox();
51312             var m = east.getMargins();
51313             b.height = centerH - (m.top+m.bottom);
51314             var totalWidth = (b.width + m.left + m.right);
51315             b.x = w - totalWidth + m.left;
51316             b.y = centerY + m.top;
51317             centerW -= totalWidth;
51318             east.updateBox(this.safeBox(b));
51319         }
51320         if(center){
51321             var m = center.getMargins();
51322             var centerBox = {
51323                 x: centerX + m.left,
51324                 y: centerY + m.top,
51325                 width: centerW - (m.left+m.right),
51326                 height: centerH - (m.top+m.bottom)
51327             };
51328             //if(this.hideOnLayout){
51329                 //center.el.setStyle("display", "block");
51330             //}
51331             center.updateBox(this.safeBox(centerBox));
51332         }
51333         this.el.repaint();
51334         this.fireEvent("layout", this);
51335     },
51336
51337     // private
51338     safeBox : function(box){
51339         box.width = Math.max(0, box.width);
51340         box.height = Math.max(0, box.height);
51341         return box;
51342     },
51343
51344     /**
51345      * Adds a ContentPanel (or subclass) to this layout.
51346      * @param {String} target The target region key (north, south, east, west or center).
51347      * @param {Roo.ContentPanel} panel The panel to add
51348      * @return {Roo.ContentPanel} The added panel
51349      */
51350     add : function(target, panel){
51351          
51352         target = target.toLowerCase();
51353         return this.regions[target].add(panel);
51354     },
51355
51356     /**
51357      * Remove a ContentPanel (or subclass) to this layout.
51358      * @param {String} target The target region key (north, south, east, west or center).
51359      * @param {Number/String/Roo.ContentPanel} panel The index, id or panel to remove
51360      * @return {Roo.ContentPanel} The removed panel
51361      */
51362     remove : function(target, panel){
51363         target = target.toLowerCase();
51364         return this.regions[target].remove(panel);
51365     },
51366
51367     /**
51368      * Searches all regions for a panel with the specified id
51369      * @param {String} panelId
51370      * @return {Roo.ContentPanel} The panel or null if it wasn't found
51371      */
51372     findPanel : function(panelId){
51373         var rs = this.regions;
51374         for(var target in rs){
51375             if(typeof rs[target] != "function"){
51376                 var p = rs[target].getPanel(panelId);
51377                 if(p){
51378                     return p;
51379                 }
51380             }
51381         }
51382         return null;
51383     },
51384
51385     /**
51386      * Searches all regions for a panel with the specified id and activates (shows) it.
51387      * @param {String/ContentPanel} panelId The panels id or the panel itself
51388      * @return {Roo.ContentPanel} The shown panel or null
51389      */
51390     showPanel : function(panelId) {
51391       var rs = this.regions;
51392       for(var target in rs){
51393          var r = rs[target];
51394          if(typeof r != "function"){
51395             if(r.hasPanel(panelId)){
51396                return r.showPanel(panelId);
51397             }
51398          }
51399       }
51400       return null;
51401    },
51402
51403    /**
51404      * Restores this layout's state using Roo.state.Manager or the state provided by the passed provider.
51405      * @param {Roo.state.Provider} provider (optional) An alternate state provider
51406      */
51407     restoreState : function(provider){
51408         if(!provider){
51409             provider = Roo.state.Manager;
51410         }
51411         var sm = new Roo.LayoutStateManager();
51412         sm.init(this, provider);
51413     },
51414
51415     /**
51416      * Adds a batch of multiple ContentPanels dynamically by passing a special regions config object.  This config
51417      * object should contain properties for each region to add ContentPanels to, and each property's value should be
51418      * a valid ContentPanel config object.  Example:
51419      * <pre><code>
51420 // Create the main layout
51421 var layout = new Roo.BorderLayout('main-ct', {
51422     west: {
51423         split:true,
51424         minSize: 175,
51425         titlebar: true
51426     },
51427     center: {
51428         title:'Components'
51429     }
51430 }, 'main-ct');
51431
51432 // Create and add multiple ContentPanels at once via configs
51433 layout.batchAdd({
51434    west: {
51435        id: 'source-files',
51436        autoCreate:true,
51437        title:'Ext Source Files',
51438        autoScroll:true,
51439        fitToFrame:true
51440    },
51441    center : {
51442        el: cview,
51443        autoScroll:true,
51444        fitToFrame:true,
51445        toolbar: tb,
51446        resizeEl:'cbody'
51447    }
51448 });
51449 </code></pre>
51450      * @param {Object} regions An object containing ContentPanel configs by region name
51451      */
51452     batchAdd : function(regions){
51453         this.beginUpdate();
51454         for(var rname in regions){
51455             var lr = this.regions[rname];
51456             if(lr){
51457                 this.addTypedPanels(lr, regions[rname]);
51458             }
51459         }
51460         this.endUpdate();
51461     },
51462
51463     // private
51464     addTypedPanels : function(lr, ps){
51465         if(typeof ps == 'string'){
51466             lr.add(new Roo.ContentPanel(ps));
51467         }
51468         else if(ps instanceof Array){
51469             for(var i =0, len = ps.length; i < len; i++){
51470                 this.addTypedPanels(lr, ps[i]);
51471             }
51472         }
51473         else if(!ps.events){ // raw config?
51474             var el = ps.el;
51475             delete ps.el; // prevent conflict
51476             lr.add(new Roo.ContentPanel(el || Roo.id(), ps));
51477         }
51478         else {  // panel object assumed!
51479             lr.add(ps);
51480         }
51481     },
51482     /**
51483      * Adds a xtype elements to the layout.
51484      * <pre><code>
51485
51486 layout.addxtype({
51487        xtype : 'ContentPanel',
51488        region: 'west',
51489        items: [ .... ]
51490    }
51491 );
51492
51493 layout.addxtype({
51494         xtype : 'NestedLayoutPanel',
51495         region: 'west',
51496         layout: {
51497            center: { },
51498            west: { }   
51499         },
51500         items : [ ... list of content panels or nested layout panels.. ]
51501    }
51502 );
51503 </code></pre>
51504      * @param {Object} cfg Xtype definition of item to add.
51505      */
51506     addxtype : function(cfg)
51507     {
51508         // basically accepts a pannel...
51509         // can accept a layout region..!?!?
51510         //Roo.log('Roo.BorderLayout add ' + cfg.xtype)
51511         
51512         if (!cfg.xtype.match(/Panel$/)) {
51513             return false;
51514         }
51515         var ret = false;
51516         
51517         if (typeof(cfg.region) == 'undefined') {
51518             Roo.log("Failed to add Panel, region was not set");
51519             Roo.log(cfg);
51520             return false;
51521         }
51522         var region = cfg.region;
51523         delete cfg.region;
51524         
51525           
51526         var xitems = [];
51527         if (cfg.items) {
51528             xitems = cfg.items;
51529             delete cfg.items;
51530         }
51531         var nb = false;
51532         
51533         switch(cfg.xtype) 
51534         {
51535             case 'ContentPanel':  // ContentPanel (el, cfg)
51536             case 'ScrollPanel':  // ContentPanel (el, cfg)
51537             case 'ViewPanel': 
51538                 if(cfg.autoCreate) {
51539                     ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
51540                 } else {
51541                     var el = this.el.createChild();
51542                     ret = new Roo[cfg.xtype](el, cfg); // new panel!!!!!
51543                 }
51544                 
51545                 this.add(region, ret);
51546                 break;
51547             
51548             
51549             case 'TreePanel': // our new panel!
51550                 cfg.el = this.el.createChild();
51551                 ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
51552                 this.add(region, ret);
51553                 break;
51554             
51555             case 'NestedLayoutPanel': 
51556                 // create a new Layout (which is  a Border Layout...
51557                 var el = this.el.createChild();
51558                 var clayout = cfg.layout;
51559                 delete cfg.layout;
51560                 clayout.items   = clayout.items  || [];
51561                 // replace this exitems with the clayout ones..
51562                 xitems = clayout.items;
51563                  
51564                 
51565                 if (region == 'center' && this.active && this.getRegion('center').panels.length < 1) {
51566                     cfg.background = false;
51567                 }
51568                 var layout = new Roo.BorderLayout(el, clayout);
51569                 
51570                 ret = new Roo[cfg.xtype](layout, cfg); // new panel!!!!!
51571                 //console.log('adding nested layout panel '  + cfg.toSource());
51572                 this.add(region, ret);
51573                 nb = {}; /// find first...
51574                 break;
51575                 
51576             case 'GridPanel': 
51577             
51578                 // needs grid and region
51579                 
51580                 //var el = this.getRegion(region).el.createChild();
51581                 var el = this.el.createChild();
51582                 // create the grid first...
51583                 
51584                 var grid = new Roo.grid[cfg.grid.xtype](el, cfg.grid);
51585                 delete cfg.grid;
51586                 if (region == 'center' && this.active ) {
51587                     cfg.background = false;
51588                 }
51589                 ret = new Roo[cfg.xtype](grid, cfg); // new panel!!!!!
51590                 
51591                 this.add(region, ret);
51592                 if (cfg.background) {
51593                     ret.on('activate', function(gp) {
51594                         if (!gp.grid.rendered) {
51595                             gp.grid.render();
51596                         }
51597                     });
51598                 } else {
51599                     grid.render();
51600                 }
51601                 break;
51602            
51603            
51604            
51605                 
51606                 
51607                 
51608             default:
51609                 if (typeof(Roo[cfg.xtype]) != 'undefined') {
51610                     
51611                     ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
51612                     this.add(region, ret);
51613                 } else {
51614                 
51615                     alert("Can not add '" + cfg.xtype + "' to BorderLayout");
51616                     return null;
51617                 }
51618                 
51619              // GridPanel (grid, cfg)
51620             
51621         }
51622         this.beginUpdate();
51623         // add children..
51624         var region = '';
51625         var abn = {};
51626         Roo.each(xitems, function(i)  {
51627             region = nb && i.region ? i.region : false;
51628             
51629             var add = ret.addxtype(i);
51630            
51631             if (region) {
51632                 nb[region] = nb[region] == undefined ? 0 : nb[region]+1;
51633                 if (!i.background) {
51634                     abn[region] = nb[region] ;
51635                 }
51636             }
51637             
51638         });
51639         this.endUpdate();
51640
51641         // make the last non-background panel active..
51642         //if (nb) { Roo.log(abn); }
51643         if (nb) {
51644             
51645             for(var r in abn) {
51646                 region = this.getRegion(r);
51647                 if (region) {
51648                     // tried using nb[r], but it does not work..
51649                      
51650                     region.showPanel(abn[r]);
51651                    
51652                 }
51653             }
51654         }
51655         return ret;
51656         
51657     }
51658 });
51659
51660 /**
51661  * Shortcut for creating a new BorderLayout object and adding one or more ContentPanels to it in a single step, handling
51662  * the beginUpdate and endUpdate calls internally.  The key to this method is the <b>panels</b> property that can be
51663  * provided with each region config, which allows you to add ContentPanel configs in addition to the region configs
51664  * during creation.  The following code is equivalent to the constructor-based example at the beginning of this class:
51665  * <pre><code>
51666 // shorthand
51667 var CP = Roo.ContentPanel;
51668
51669 var layout = Roo.BorderLayout.create({
51670     north: {
51671         initialSize: 25,
51672         titlebar: false,
51673         panels: [new CP("north", "North")]
51674     },
51675     west: {
51676         split:true,
51677         initialSize: 200,
51678         minSize: 175,
51679         maxSize: 400,
51680         titlebar: true,
51681         collapsible: true,
51682         panels: [new CP("west", {title: "West"})]
51683     },
51684     east: {
51685         split:true,
51686         initialSize: 202,
51687         minSize: 175,
51688         maxSize: 400,
51689         titlebar: true,
51690         collapsible: true,
51691         panels: [new CP("autoTabs", {title: "Auto Tabs", closable: true})]
51692     },
51693     south: {
51694         split:true,
51695         initialSize: 100,
51696         minSize: 100,
51697         maxSize: 200,
51698         titlebar: true,
51699         collapsible: true,
51700         panels: [new CP("south", {title: "South", closable: true})]
51701     },
51702     center: {
51703         titlebar: true,
51704         autoScroll:true,
51705         resizeTabs: true,
51706         minTabWidth: 50,
51707         preferredTabWidth: 150,
51708         panels: [
51709             new CP("center1", {title: "Close Me", closable: true}),
51710             new CP("center2", {title: "Center Panel", closable: false})
51711         ]
51712     }
51713 }, document.body);
51714
51715 layout.getRegion("center").showPanel("center1");
51716 </code></pre>
51717  * @param config
51718  * @param targetEl
51719  */
51720 Roo.BorderLayout.create = function(config, targetEl){
51721     var layout = new Roo.BorderLayout(targetEl || document.body, config);
51722     layout.beginUpdate();
51723     var regions = Roo.BorderLayout.RegionFactory.validRegions;
51724     for(var j = 0, jlen = regions.length; j < jlen; j++){
51725         var lr = regions[j];
51726         if(layout.regions[lr] && config[lr].panels){
51727             var r = layout.regions[lr];
51728             var ps = config[lr].panels;
51729             layout.addTypedPanels(r, ps);
51730         }
51731     }
51732     layout.endUpdate();
51733     return layout;
51734 };
51735
51736 // private
51737 Roo.BorderLayout.RegionFactory = {
51738     // private
51739     validRegions : ["north","south","east","west","center"],
51740
51741     // private
51742     create : function(target, mgr, config){
51743         target = target.toLowerCase();
51744         if(config.lightweight || config.basic){
51745             return new Roo.BasicLayoutRegion(mgr, config, target);
51746         }
51747         switch(target){
51748             case "north":
51749                 return new Roo.NorthLayoutRegion(mgr, config);
51750             case "south":
51751                 return new Roo.SouthLayoutRegion(mgr, config);
51752             case "east":
51753                 return new Roo.EastLayoutRegion(mgr, config);
51754             case "west":
51755                 return new Roo.WestLayoutRegion(mgr, config);
51756             case "center":
51757                 return new Roo.CenterLayoutRegion(mgr, config);
51758         }
51759         throw 'Layout region "'+target+'" not supported.';
51760     }
51761 };/*
51762  * Based on:
51763  * Ext JS Library 1.1.1
51764  * Copyright(c) 2006-2007, Ext JS, LLC.
51765  *
51766  * Originally Released Under LGPL - original licence link has changed is not relivant.
51767  *
51768  * Fork - LGPL
51769  * <script type="text/javascript">
51770  */
51771  
51772 /**
51773  * @class Roo.BasicLayoutRegion
51774  * @extends Roo.util.Observable
51775  * This class represents a lightweight region in a layout manager. This region does not move dom nodes
51776  * and does not have a titlebar, tabs or any other features. All it does is size and position 
51777  * panels. To create a BasicLayoutRegion, add lightweight:true or basic:true to your regions config.
51778  */
51779 Roo.BasicLayoutRegion = function(mgr, config, pos, skipConfig){
51780     this.mgr = mgr;
51781     this.position  = pos;
51782     this.events = {
51783         /**
51784          * @scope Roo.BasicLayoutRegion
51785          */
51786         
51787         /**
51788          * @event beforeremove
51789          * Fires before a panel is removed (or closed). To cancel the removal set "e.cancel = true" on the event argument.
51790          * @param {Roo.LayoutRegion} this
51791          * @param {Roo.ContentPanel} panel The panel
51792          * @param {Object} e The cancel event object
51793          */
51794         "beforeremove" : true,
51795         /**
51796          * @event invalidated
51797          * Fires when the layout for this region is changed.
51798          * @param {Roo.LayoutRegion} this
51799          */
51800         "invalidated" : true,
51801         /**
51802          * @event visibilitychange
51803          * Fires when this region is shown or hidden 
51804          * @param {Roo.LayoutRegion} this
51805          * @param {Boolean} visibility true or false
51806          */
51807         "visibilitychange" : true,
51808         /**
51809          * @event paneladded
51810          * Fires when a panel is added. 
51811          * @param {Roo.LayoutRegion} this
51812          * @param {Roo.ContentPanel} panel The panel
51813          */
51814         "paneladded" : true,
51815         /**
51816          * @event panelremoved
51817          * Fires when a panel is removed. 
51818          * @param {Roo.LayoutRegion} this
51819          * @param {Roo.ContentPanel} panel The panel
51820          */
51821         "panelremoved" : true,
51822         /**
51823          * @event beforecollapse
51824          * Fires when this region before collapse.
51825          * @param {Roo.LayoutRegion} this
51826          */
51827         "beforecollapse" : true,
51828         /**
51829          * @event collapsed
51830          * Fires when this region is collapsed.
51831          * @param {Roo.LayoutRegion} this
51832          */
51833         "collapsed" : true,
51834         /**
51835          * @event expanded
51836          * Fires when this region is expanded.
51837          * @param {Roo.LayoutRegion} this
51838          */
51839         "expanded" : true,
51840         /**
51841          * @event slideshow
51842          * Fires when this region is slid into view.
51843          * @param {Roo.LayoutRegion} this
51844          */
51845         "slideshow" : true,
51846         /**
51847          * @event slidehide
51848          * Fires when this region slides out of view. 
51849          * @param {Roo.LayoutRegion} this
51850          */
51851         "slidehide" : true,
51852         /**
51853          * @event panelactivated
51854          * Fires when a panel is activated. 
51855          * @param {Roo.LayoutRegion} this
51856          * @param {Roo.ContentPanel} panel The activated panel
51857          */
51858         "panelactivated" : true,
51859         /**
51860          * @event resized
51861          * Fires when the user resizes this region. 
51862          * @param {Roo.LayoutRegion} this
51863          * @param {Number} newSize The new size (width for east/west, height for north/south)
51864          */
51865         "resized" : true
51866     };
51867     /** A collection of panels in this region. @type Roo.util.MixedCollection */
51868     this.panels = new Roo.util.MixedCollection();
51869     this.panels.getKey = this.getPanelId.createDelegate(this);
51870     this.box = null;
51871     this.activePanel = null;
51872     // ensure listeners are added...
51873     
51874     if (config.listeners || config.events) {
51875         Roo.BasicLayoutRegion.superclass.constructor.call(this, {
51876             listeners : config.listeners || {},
51877             events : config.events || {}
51878         });
51879     }
51880     
51881     if(skipConfig !== true){
51882         this.applyConfig(config);
51883     }
51884 };
51885
51886 Roo.extend(Roo.BasicLayoutRegion, Roo.util.Observable, {
51887     getPanelId : function(p){
51888         return p.getId();
51889     },
51890     
51891     applyConfig : function(config){
51892         this.margins = config.margins || this.margins || {top: 0, left: 0, right:0, bottom: 0};
51893         this.config = config;
51894         
51895     },
51896     
51897     /**
51898      * Resizes the region to the specified size. For vertical regions (west, east) this adjusts 
51899      * the width, for horizontal (north, south) the height.
51900      * @param {Number} newSize The new width or height
51901      */
51902     resizeTo : function(newSize){
51903         var el = this.el ? this.el :
51904                  (this.activePanel ? this.activePanel.getEl() : null);
51905         if(el){
51906             switch(this.position){
51907                 case "east":
51908                 case "west":
51909                     el.setWidth(newSize);
51910                     this.fireEvent("resized", this, newSize);
51911                 break;
51912                 case "north":
51913                 case "south":
51914                     el.setHeight(newSize);
51915                     this.fireEvent("resized", this, newSize);
51916                 break;                
51917             }
51918         }
51919     },
51920     
51921     getBox : function(){
51922         return this.activePanel ? this.activePanel.getEl().getBox(false, true) : null;
51923     },
51924     
51925     getMargins : function(){
51926         return this.margins;
51927     },
51928     
51929     updateBox : function(box){
51930         this.box = box;
51931         var el = this.activePanel.getEl();
51932         el.dom.style.left = box.x + "px";
51933         el.dom.style.top = box.y + "px";
51934         this.activePanel.setSize(box.width, box.height);
51935     },
51936     
51937     /**
51938      * Returns the container element for this region.
51939      * @return {Roo.Element}
51940      */
51941     getEl : function(){
51942         return this.activePanel;
51943     },
51944     
51945     /**
51946      * Returns true if this region is currently visible.
51947      * @return {Boolean}
51948      */
51949     isVisible : function(){
51950         return this.activePanel ? true : false;
51951     },
51952     
51953     setActivePanel : function(panel){
51954         panel = this.getPanel(panel);
51955         if(this.activePanel && this.activePanel != panel){
51956             this.activePanel.setActiveState(false);
51957             this.activePanel.getEl().setLeftTop(-10000,-10000);
51958         }
51959         this.activePanel = panel;
51960         panel.setActiveState(true);
51961         if(this.box){
51962             panel.setSize(this.box.width, this.box.height);
51963         }
51964         this.fireEvent("panelactivated", this, panel);
51965         this.fireEvent("invalidated");
51966     },
51967     
51968     /**
51969      * Show the specified panel.
51970      * @param {Number/String/ContentPanel} panelId The panels index, id or the panel itself
51971      * @return {Roo.ContentPanel} The shown panel or null
51972      */
51973     showPanel : function(panel){
51974         if(panel = this.getPanel(panel)){
51975             this.setActivePanel(panel);
51976         }
51977         return panel;
51978     },
51979     
51980     /**
51981      * Get the active panel for this region.
51982      * @return {Roo.ContentPanel} The active panel or null
51983      */
51984     getActivePanel : function(){
51985         return this.activePanel;
51986     },
51987     
51988     /**
51989      * Add the passed ContentPanel(s)
51990      * @param {ContentPanel...} panel The ContentPanel(s) to add (you can pass more than one)
51991      * @return {Roo.ContentPanel} The panel added (if only one was added)
51992      */
51993     add : function(panel){
51994         if(arguments.length > 1){
51995             for(var i = 0, len = arguments.length; i < len; i++) {
51996                 this.add(arguments[i]);
51997             }
51998             return null;
51999         }
52000         if(this.hasPanel(panel)){
52001             this.showPanel(panel);
52002             return panel;
52003         }
52004         var el = panel.getEl();
52005         if(el.dom.parentNode != this.mgr.el.dom){
52006             this.mgr.el.dom.appendChild(el.dom);
52007         }
52008         if(panel.setRegion){
52009             panel.setRegion(this);
52010         }
52011         this.panels.add(panel);
52012         el.setStyle("position", "absolute");
52013         if(!panel.background){
52014             this.setActivePanel(panel);
52015             if(this.config.initialSize && this.panels.getCount()==1){
52016                 this.resizeTo(this.config.initialSize);
52017             }
52018         }
52019         this.fireEvent("paneladded", this, panel);
52020         return panel;
52021     },
52022     
52023     /**
52024      * Returns true if the panel is in this region.
52025      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52026      * @return {Boolean}
52027      */
52028     hasPanel : function(panel){
52029         if(typeof panel == "object"){ // must be panel obj
52030             panel = panel.getId();
52031         }
52032         return this.getPanel(panel) ? true : false;
52033     },
52034     
52035     /**
52036      * Removes the specified panel. If preservePanel is not true (either here or in the config), the panel is destroyed.
52037      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52038      * @param {Boolean} preservePanel Overrides the config preservePanel option
52039      * @return {Roo.ContentPanel} The panel that was removed
52040      */
52041     remove : function(panel, preservePanel){
52042         panel = this.getPanel(panel);
52043         if(!panel){
52044             return null;
52045         }
52046         var e = {};
52047         this.fireEvent("beforeremove", this, panel, e);
52048         if(e.cancel === true){
52049             return null;
52050         }
52051         var panelId = panel.getId();
52052         this.panels.removeKey(panelId);
52053         return panel;
52054     },
52055     
52056     /**
52057      * Returns the panel specified or null if it's not in this region.
52058      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52059      * @return {Roo.ContentPanel}
52060      */
52061     getPanel : function(id){
52062         if(typeof id == "object"){ // must be panel obj
52063             return id;
52064         }
52065         return this.panels.get(id);
52066     },
52067     
52068     /**
52069      * Returns this regions position (north/south/east/west/center).
52070      * @return {String} 
52071      */
52072     getPosition: function(){
52073         return this.position;    
52074     }
52075 });/*
52076  * Based on:
52077  * Ext JS Library 1.1.1
52078  * Copyright(c) 2006-2007, Ext JS, LLC.
52079  *
52080  * Originally Released Under LGPL - original licence link has changed is not relivant.
52081  *
52082  * Fork - LGPL
52083  * <script type="text/javascript">
52084  */
52085  
52086 /**
52087  * @class Roo.LayoutRegion
52088  * @extends Roo.BasicLayoutRegion
52089  * This class represents a region in a layout manager.
52090  * @cfg {Boolean}   collapsible     False to disable collapsing (defaults to true)
52091  * @cfg {Boolean}   collapsed       True to set the initial display to collapsed (defaults to false)
52092  * @cfg {Boolean}   floatable       False to disable floating (defaults to true)
52093  * @cfg {Object}    margins         Margins for the element (defaults to {top: 0, left: 0, right:0, bottom: 0})
52094  * @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})
52095  * @cfg {String}    tabPosition     (top|bottom) "top" or "bottom" (defaults to "bottom")
52096  * @cfg {String}    collapsedTitle  Optional string message to display in the collapsed block of a north or south region
52097  * @cfg {Boolean}   alwaysShowTabs  True to always display tabs even when there is only 1 panel (defaults to false)
52098  * @cfg {Boolean}   autoScroll      True to enable overflow scrolling (defaults to false)
52099  * @cfg {Boolean}   titlebar        True to display a title bar (defaults to true)
52100  * @cfg {String}    title           The title for the region (overrides panel titles)
52101  * @cfg {Boolean}   animate         True to animate expand/collapse (defaults to false)
52102  * @cfg {Boolean}   autoHide        False to disable auto hiding when the mouse leaves the "floated" region (defaults to true)
52103  * @cfg {Boolean}   preservePanels  True to preserve removed panels so they can be readded later (defaults to false)
52104  * @cfg {Boolean}   closeOnTab      True to place the close icon on the tabs instead of the region titlebar (defaults to false)
52105  * @cfg {Boolean}   hideTabs        True to hide the tab strip (defaults to false)
52106  * @cfg {Boolean}   resizeTabs      True to enable automatic tab resizing. This will resize the tabs so they are all the same size and fit within
52107  *                      the space available, similar to FireFox 1.5 tabs (defaults to false)
52108  * @cfg {Number}    minTabWidth     The minimum tab width (defaults to 40)
52109  * @cfg {Number}    preferredTabWidth The preferred tab width (defaults to 150)
52110  * @cfg {Boolean}   showPin         True to show a pin button
52111  * @cfg {Boolean}   hidden          True to start the region hidden (defaults to false)
52112  * @cfg {Boolean}   hideWhenEmpty   True to hide the region when it has no panels
52113  * @cfg {Boolean}   disableTabTips  True to disable tab tooltips
52114  * @cfg {Number}    width           For East/West panels
52115  * @cfg {Number}    height          For North/South panels
52116  * @cfg {Boolean}   split           To show the splitter
52117  * @cfg {Boolean}   toolbar         xtype configuration for a toolbar - shows on right of tabbar
52118  */
52119 Roo.LayoutRegion = function(mgr, config, pos){
52120     Roo.LayoutRegion.superclass.constructor.call(this, mgr, config, pos, true);
52121     var dh = Roo.DomHelper;
52122     /** This region's container element 
52123     * @type Roo.Element */
52124     this.el = dh.append(mgr.el.dom, {tag: "div", cls: "x-layout-panel x-layout-panel-" + this.position}, true);
52125     /** This region's title element 
52126     * @type Roo.Element */
52127
52128     this.titleEl = dh.append(this.el.dom, {tag: "div", unselectable: "on", cls: "x-unselectable x-layout-panel-hd x-layout-title-"+this.position, children:[
52129         {tag: "span", cls: "x-unselectable x-layout-panel-hd-text", unselectable: "on", html: "&#160;"},
52130         {tag: "div", cls: "x-unselectable x-layout-panel-hd-tools", unselectable: "on"}
52131     ]}, true);
52132     this.titleEl.enableDisplayMode();
52133     /** This region's title text element 
52134     * @type HTMLElement */
52135     this.titleTextEl = this.titleEl.dom.firstChild;
52136     this.tools = Roo.get(this.titleEl.dom.childNodes[1], true);
52137     this.closeBtn = this.createTool(this.tools.dom, "x-layout-close");
52138     this.closeBtn.enableDisplayMode();
52139     this.closeBtn.on("click", this.closeClicked, this);
52140     this.closeBtn.hide();
52141
52142     this.createBody(config);
52143     this.visible = true;
52144     this.collapsed = false;
52145
52146     if(config.hideWhenEmpty){
52147         this.hide();
52148         this.on("paneladded", this.validateVisibility, this);
52149         this.on("panelremoved", this.validateVisibility, this);
52150     }
52151     this.applyConfig(config);
52152 };
52153
52154 Roo.extend(Roo.LayoutRegion, Roo.BasicLayoutRegion, {
52155
52156     createBody : function(){
52157         /** This region's body element 
52158         * @type Roo.Element */
52159         this.bodyEl = this.el.createChild({tag: "div", cls: "x-layout-panel-body"});
52160     },
52161
52162     applyConfig : function(c){
52163         if(c.collapsible && this.position != "center" && !this.collapsedEl){
52164             var dh = Roo.DomHelper;
52165             if(c.titlebar !== false){
52166                 this.collapseBtn = this.createTool(this.tools.dom, "x-layout-collapse-"+this.position);
52167                 this.collapseBtn.on("click", this.collapse, this);
52168                 this.collapseBtn.enableDisplayMode();
52169
52170                 if(c.showPin === true || this.showPin){
52171                     this.stickBtn = this.createTool(this.tools.dom, "x-layout-stick");
52172                     this.stickBtn.enableDisplayMode();
52173                     this.stickBtn.on("click", this.expand, this);
52174                     this.stickBtn.hide();
52175                 }
52176             }
52177             /** This region's collapsed element
52178             * @type Roo.Element */
52179             this.collapsedEl = dh.append(this.mgr.el.dom, {cls: "x-layout-collapsed x-layout-collapsed-"+this.position, children:[
52180                 {cls: "x-layout-collapsed-tools", children:[{cls: "x-layout-ctools-inner"}]}
52181             ]}, true);
52182             if(c.floatable !== false){
52183                this.collapsedEl.addClassOnOver("x-layout-collapsed-over");
52184                this.collapsedEl.on("click", this.collapseClick, this);
52185             }
52186
52187             if(c.collapsedTitle && (this.position == "north" || this.position== "south")) {
52188                 this.collapsedTitleTextEl = dh.append(this.collapsedEl.dom, {tag: "div", cls: "x-unselectable x-layout-panel-hd-text",
52189                    id: "message", unselectable: "on", style:{"float":"left"}});
52190                this.collapsedTitleTextEl.innerHTML = c.collapsedTitle;
52191              }
52192             this.expandBtn = this.createTool(this.collapsedEl.dom.firstChild.firstChild, "x-layout-expand-"+this.position);
52193             this.expandBtn.on("click", this.expand, this);
52194         }
52195         if(this.collapseBtn){
52196             this.collapseBtn.setVisible(c.collapsible == true);
52197         }
52198         this.cmargins = c.cmargins || this.cmargins ||
52199                          (this.position == "west" || this.position == "east" ?
52200                              {top: 0, left: 2, right:2, bottom: 0} :
52201                              {top: 2, left: 0, right:0, bottom: 2});
52202         this.margins = c.margins || this.margins || {top: 0, left: 0, right:0, bottom: 0};
52203         this.bottomTabs = c.tabPosition != "top";
52204         this.autoScroll = c.autoScroll || false;
52205         if(this.autoScroll){
52206             this.bodyEl.setStyle("overflow", "auto");
52207         }else{
52208             this.bodyEl.setStyle("overflow", "hidden");
52209         }
52210         //if(c.titlebar !== false){
52211             if((!c.titlebar && !c.title) || c.titlebar === false){
52212                 this.titleEl.hide();
52213             }else{
52214                 this.titleEl.show();
52215                 if(c.title){
52216                     this.titleTextEl.innerHTML = c.title;
52217                 }
52218             }
52219         //}
52220         this.duration = c.duration || .30;
52221         this.slideDuration = c.slideDuration || .45;
52222         this.config = c;
52223         if(c.collapsed){
52224             this.collapse(true);
52225         }
52226         if(c.hidden){
52227             this.hide();
52228         }
52229     },
52230     /**
52231      * Returns true if this region is currently visible.
52232      * @return {Boolean}
52233      */
52234     isVisible : function(){
52235         return this.visible;
52236     },
52237
52238     /**
52239      * Updates the title for collapsed north/south regions (used with {@link #collapsedTitle} config option)
52240      * @param {String} title (optional) The title text (accepts HTML markup, defaults to the numeric character reference for a non-breaking space, "&amp;#160;")
52241      */
52242     setCollapsedTitle : function(title){
52243         title = title || "&#160;";
52244         if(this.collapsedTitleTextEl){
52245             this.collapsedTitleTextEl.innerHTML = title;
52246         }
52247     },
52248
52249     getBox : function(){
52250         var b;
52251         if(!this.collapsed){
52252             b = this.el.getBox(false, true);
52253         }else{
52254             b = this.collapsedEl.getBox(false, true);
52255         }
52256         return b;
52257     },
52258
52259     getMargins : function(){
52260         return this.collapsed ? this.cmargins : this.margins;
52261     },
52262
52263     highlight : function(){
52264         this.el.addClass("x-layout-panel-dragover");
52265     },
52266
52267     unhighlight : function(){
52268         this.el.removeClass("x-layout-panel-dragover");
52269     },
52270
52271     updateBox : function(box){
52272         this.box = box;
52273         if(!this.collapsed){
52274             this.el.dom.style.left = box.x + "px";
52275             this.el.dom.style.top = box.y + "px";
52276             this.updateBody(box.width, box.height);
52277         }else{
52278             this.collapsedEl.dom.style.left = box.x + "px";
52279             this.collapsedEl.dom.style.top = box.y + "px";
52280             this.collapsedEl.setSize(box.width, box.height);
52281         }
52282         if(this.tabs){
52283             this.tabs.autoSizeTabs();
52284         }
52285     },
52286
52287     updateBody : function(w, h){
52288         if(w !== null){
52289             this.el.setWidth(w);
52290             w -= this.el.getBorderWidth("rl");
52291             if(this.config.adjustments){
52292                 w += this.config.adjustments[0];
52293             }
52294         }
52295         if(h !== null){
52296             this.el.setHeight(h);
52297             h = this.titleEl && this.titleEl.isDisplayed() ? h - (this.titleEl.getHeight()||0) : h;
52298             h -= this.el.getBorderWidth("tb");
52299             if(this.config.adjustments){
52300                 h += this.config.adjustments[1];
52301             }
52302             this.bodyEl.setHeight(h);
52303             if(this.tabs){
52304                 h = this.tabs.syncHeight(h);
52305             }
52306         }
52307         if(this.panelSize){
52308             w = w !== null ? w : this.panelSize.width;
52309             h = h !== null ? h : this.panelSize.height;
52310         }
52311         if(this.activePanel){
52312             var el = this.activePanel.getEl();
52313             w = w !== null ? w : el.getWidth();
52314             h = h !== null ? h : el.getHeight();
52315             this.panelSize = {width: w, height: h};
52316             this.activePanel.setSize(w, h);
52317         }
52318         if(Roo.isIE && this.tabs){
52319             this.tabs.el.repaint();
52320         }
52321     },
52322
52323     /**
52324      * Returns the container element for this region.
52325      * @return {Roo.Element}
52326      */
52327     getEl : function(){
52328         return this.el;
52329     },
52330
52331     /**
52332      * Hides this region.
52333      */
52334     hide : function(){
52335         if(!this.collapsed){
52336             this.el.dom.style.left = "-2000px";
52337             this.el.hide();
52338         }else{
52339             this.collapsedEl.dom.style.left = "-2000px";
52340             this.collapsedEl.hide();
52341         }
52342         this.visible = false;
52343         this.fireEvent("visibilitychange", this, false);
52344     },
52345
52346     /**
52347      * Shows this region if it was previously hidden.
52348      */
52349     show : function(){
52350         if(!this.collapsed){
52351             this.el.show();
52352         }else{
52353             this.collapsedEl.show();
52354         }
52355         this.visible = true;
52356         this.fireEvent("visibilitychange", this, true);
52357     },
52358
52359     closeClicked : function(){
52360         if(this.activePanel){
52361             this.remove(this.activePanel);
52362         }
52363     },
52364
52365     collapseClick : function(e){
52366         if(this.isSlid){
52367            e.stopPropagation();
52368            this.slideIn();
52369         }else{
52370            e.stopPropagation();
52371            this.slideOut();
52372         }
52373     },
52374
52375     /**
52376      * Collapses this region.
52377      * @param {Boolean} skipAnim (optional) true to collapse the element without animation (if animate is true)
52378      */
52379     collapse : function(skipAnim, skipCheck = false){
52380         if(this.collapsed) {
52381             return;
52382         }
52383         
52384         if(skipCheck || this.fireEvent("beforecollapse", this) != false){
52385             
52386             this.collapsed = true;
52387             if(this.split){
52388                 this.split.el.hide();
52389             }
52390             if(this.config.animate && skipAnim !== true){
52391                 this.fireEvent("invalidated", this);
52392                 this.animateCollapse();
52393             }else{
52394                 this.el.setLocation(-20000,-20000);
52395                 this.el.hide();
52396                 this.collapsedEl.show();
52397                 this.fireEvent("collapsed", this);
52398                 this.fireEvent("invalidated", this);
52399             }
52400         }
52401         
52402     },
52403
52404     animateCollapse : function(){
52405         // overridden
52406     },
52407
52408     /**
52409      * Expands this region if it was previously collapsed.
52410      * @param {Roo.EventObject} e The event that triggered the expand (or null if calling manually)
52411      * @param {Boolean} skipAnim (optional) true to expand the element without animation (if animate is true)
52412      */
52413     expand : function(e, skipAnim){
52414         if(e) {
52415             e.stopPropagation();
52416         }
52417         if(!this.collapsed || this.el.hasActiveFx()) {
52418             return;
52419         }
52420         if(this.isSlid){
52421             this.afterSlideIn();
52422             skipAnim = true;
52423         }
52424         this.collapsed = false;
52425         if(this.config.animate && skipAnim !== true){
52426             this.animateExpand();
52427         }else{
52428             this.el.show();
52429             if(this.split){
52430                 this.split.el.show();
52431             }
52432             this.collapsedEl.setLocation(-2000,-2000);
52433             this.collapsedEl.hide();
52434             this.fireEvent("invalidated", this);
52435             this.fireEvent("expanded", this);
52436         }
52437     },
52438
52439     animateExpand : function(){
52440         // overridden
52441     },
52442
52443     initTabs : function()
52444     {
52445         this.bodyEl.setStyle("overflow", "hidden");
52446         var ts = new Roo.TabPanel(
52447                 this.bodyEl.dom,
52448                 {
52449                     tabPosition: this.bottomTabs ? 'bottom' : 'top',
52450                     disableTooltips: this.config.disableTabTips,
52451                     toolbar : this.config.toolbar
52452                 }
52453         );
52454         if(this.config.hideTabs){
52455             ts.stripWrap.setDisplayed(false);
52456         }
52457         this.tabs = ts;
52458         ts.resizeTabs = this.config.resizeTabs === true;
52459         ts.minTabWidth = this.config.minTabWidth || 40;
52460         ts.maxTabWidth = this.config.maxTabWidth || 250;
52461         ts.preferredTabWidth = this.config.preferredTabWidth || 150;
52462         ts.monitorResize = false;
52463         ts.bodyEl.setStyle("overflow", this.config.autoScroll ? "auto" : "hidden");
52464         ts.bodyEl.addClass('x-layout-tabs-body');
52465         this.panels.each(this.initPanelAsTab, this);
52466     },
52467
52468     initPanelAsTab : function(panel){
52469         var ti = this.tabs.addTab(panel.getEl().id, panel.getTitle(), null,
52470                     this.config.closeOnTab && panel.isClosable());
52471         if(panel.tabTip !== undefined){
52472             ti.setTooltip(panel.tabTip);
52473         }
52474         ti.on("activate", function(){
52475               this.setActivePanel(panel);
52476         }, this);
52477         if(this.config.closeOnTab){
52478             ti.on("beforeclose", function(t, e){
52479                 e.cancel = true;
52480                 this.remove(panel);
52481             }, this);
52482         }
52483         return ti;
52484     },
52485
52486     updatePanelTitle : function(panel, title){
52487         if(this.activePanel == panel){
52488             this.updateTitle(title);
52489         }
52490         if(this.tabs){
52491             var ti = this.tabs.getTab(panel.getEl().id);
52492             ti.setText(title);
52493             if(panel.tabTip !== undefined){
52494                 ti.setTooltip(panel.tabTip);
52495             }
52496         }
52497     },
52498
52499     updateTitle : function(title){
52500         if(this.titleTextEl && !this.config.title){
52501             this.titleTextEl.innerHTML = (typeof title != "undefined" && title.length > 0 ? title : "&#160;");
52502         }
52503     },
52504
52505     setActivePanel : function(panel){
52506         panel = this.getPanel(panel);
52507         if(this.activePanel && this.activePanel != panel){
52508             this.activePanel.setActiveState(false);
52509         }
52510         this.activePanel = panel;
52511         panel.setActiveState(true);
52512         if(this.panelSize){
52513             panel.setSize(this.panelSize.width, this.panelSize.height);
52514         }
52515         if(this.closeBtn){
52516             this.closeBtn.setVisible(!this.config.closeOnTab && !this.isSlid && panel.isClosable());
52517         }
52518         this.updateTitle(panel.getTitle());
52519         if(this.tabs){
52520             this.fireEvent("invalidated", this);
52521         }
52522         this.fireEvent("panelactivated", this, panel);
52523     },
52524
52525     /**
52526      * Shows the specified panel.
52527      * @param {Number/String/ContentPanel} panelId The panel's index, id or the panel itself
52528      * @return {Roo.ContentPanel} The shown panel, or null if a panel could not be found from panelId
52529      */
52530     showPanel : function(panel)
52531     {
52532         panel = this.getPanel(panel);
52533         if(panel){
52534             if(this.tabs){
52535                 var tab = this.tabs.getTab(panel.getEl().id);
52536                 if(tab.isHidden()){
52537                     this.tabs.unhideTab(tab.id);
52538                 }
52539                 tab.activate();
52540             }else{
52541                 this.setActivePanel(panel);
52542             }
52543         }
52544         return panel;
52545     },
52546
52547     /**
52548      * Get the active panel for this region.
52549      * @return {Roo.ContentPanel} The active panel or null
52550      */
52551     getActivePanel : function(){
52552         return this.activePanel;
52553     },
52554
52555     validateVisibility : function(){
52556         if(this.panels.getCount() < 1){
52557             this.updateTitle("&#160;");
52558             this.closeBtn.hide();
52559             this.hide();
52560         }else{
52561             if(!this.isVisible()){
52562                 this.show();
52563             }
52564         }
52565     },
52566
52567     /**
52568      * Adds the passed ContentPanel(s) to this region.
52569      * @param {ContentPanel...} panel The ContentPanel(s) to add (you can pass more than one)
52570      * @return {Roo.ContentPanel} The panel added (if only one was added; null otherwise)
52571      */
52572     add : function(panel){
52573         if(arguments.length > 1){
52574             for(var i = 0, len = arguments.length; i < len; i++) {
52575                 this.add(arguments[i]);
52576             }
52577             return null;
52578         }
52579         if(this.hasPanel(panel)){
52580             this.showPanel(panel);
52581             return panel;
52582         }
52583         panel.setRegion(this);
52584         this.panels.add(panel);
52585         if(this.panels.getCount() == 1 && !this.config.alwaysShowTabs){
52586             this.bodyEl.dom.appendChild(panel.getEl().dom);
52587             if(panel.background !== true){
52588                 this.setActivePanel(panel);
52589             }
52590             this.fireEvent("paneladded", this, panel);
52591             return panel;
52592         }
52593         if(!this.tabs){
52594             this.initTabs();
52595         }else{
52596             this.initPanelAsTab(panel);
52597         }
52598         if(panel.background !== true){
52599             this.tabs.activate(panel.getEl().id);
52600         }
52601         this.fireEvent("paneladded", this, panel);
52602         return panel;
52603     },
52604
52605     /**
52606      * Hides the tab for the specified panel.
52607      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
52608      */
52609     hidePanel : function(panel){
52610         if(this.tabs && (panel = this.getPanel(panel))){
52611             this.tabs.hideTab(panel.getEl().id);
52612         }
52613     },
52614
52615     /**
52616      * Unhides the tab for a previously hidden panel.
52617      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
52618      */
52619     unhidePanel : function(panel){
52620         if(this.tabs && (panel = this.getPanel(panel))){
52621             this.tabs.unhideTab(panel.getEl().id);
52622         }
52623     },
52624
52625     clearPanels : function(){
52626         while(this.panels.getCount() > 0){
52627              this.remove(this.panels.first());
52628         }
52629     },
52630
52631     /**
52632      * Removes the specified panel. If preservePanel is not true (either here or in the config), the panel is destroyed.
52633      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
52634      * @param {Boolean} preservePanel Overrides the config preservePanel option
52635      * @return {Roo.ContentPanel} The panel that was removed
52636      */
52637     remove : function(panel, preservePanel){
52638         panel = this.getPanel(panel);
52639         if(!panel){
52640             return null;
52641         }
52642         var e = {};
52643         this.fireEvent("beforeremove", this, panel, e);
52644         if(e.cancel === true){
52645             return null;
52646         }
52647         preservePanel = (typeof preservePanel != "undefined" ? preservePanel : (this.config.preservePanels === true || panel.preserve === true));
52648         var panelId = panel.getId();
52649         this.panels.removeKey(panelId);
52650         if(preservePanel){
52651             document.body.appendChild(panel.getEl().dom);
52652         }
52653         if(this.tabs){
52654             this.tabs.removeTab(panel.getEl().id);
52655         }else if (!preservePanel){
52656             this.bodyEl.dom.removeChild(panel.getEl().dom);
52657         }
52658         if(this.panels.getCount() == 1 && this.tabs && !this.config.alwaysShowTabs){
52659             var p = this.panels.first();
52660             var tempEl = document.createElement("div"); // temp holder to keep IE from deleting the node
52661             tempEl.appendChild(p.getEl().dom);
52662             this.bodyEl.update("");
52663             this.bodyEl.dom.appendChild(p.getEl().dom);
52664             tempEl = null;
52665             this.updateTitle(p.getTitle());
52666             this.tabs = null;
52667             this.bodyEl.setStyle("overflow", this.config.autoScroll ? "auto" : "hidden");
52668             this.setActivePanel(p);
52669         }
52670         panel.setRegion(null);
52671         if(this.activePanel == panel){
52672             this.activePanel = null;
52673         }
52674         if(this.config.autoDestroy !== false && preservePanel !== true){
52675             try{panel.destroy();}catch(e){}
52676         }
52677         this.fireEvent("panelremoved", this, panel);
52678         return panel;
52679     },
52680
52681     /**
52682      * Returns the TabPanel component used by this region
52683      * @return {Roo.TabPanel}
52684      */
52685     getTabs : function(){
52686         return this.tabs;
52687     },
52688
52689     createTool : function(parentEl, className){
52690         var btn = Roo.DomHelper.append(parentEl, {tag: "div", cls: "x-layout-tools-button",
52691             children: [{tag: "div", cls: "x-layout-tools-button-inner " + className, html: "&#160;"}]}, true);
52692         btn.addClassOnOver("x-layout-tools-button-over");
52693         return btn;
52694     }
52695 });/*
52696  * Based on:
52697  * Ext JS Library 1.1.1
52698  * Copyright(c) 2006-2007, Ext JS, LLC.
52699  *
52700  * Originally Released Under LGPL - original licence link has changed is not relivant.
52701  *
52702  * Fork - LGPL
52703  * <script type="text/javascript">
52704  */
52705  
52706
52707
52708 /**
52709  * @class Roo.SplitLayoutRegion
52710  * @extends Roo.LayoutRegion
52711  * Adds a splitbar and other (private) useful functionality to a {@link Roo.LayoutRegion}.
52712  */
52713 Roo.SplitLayoutRegion = function(mgr, config, pos, cursor){
52714     this.cursor = cursor;
52715     Roo.SplitLayoutRegion.superclass.constructor.call(this, mgr, config, pos);
52716 };
52717
52718 Roo.extend(Roo.SplitLayoutRegion, Roo.LayoutRegion, {
52719     splitTip : "Drag to resize.",
52720     collapsibleSplitTip : "Drag to resize. Double click to hide.",
52721     useSplitTips : false,
52722
52723     applyConfig : function(config){
52724         Roo.SplitLayoutRegion.superclass.applyConfig.call(this, config);
52725         if(config.split){
52726             if(!this.split){
52727                 var splitEl = Roo.DomHelper.append(this.mgr.el.dom, 
52728                         {tag: "div", id: this.el.id + "-split", cls: "x-layout-split x-layout-split-"+this.position, html: "&#160;"});
52729                 /** The SplitBar for this region 
52730                 * @type Roo.SplitBar */
52731                 this.split = new Roo.SplitBar(splitEl, this.el, this.orientation);
52732                 this.split.on("moved", this.onSplitMove, this);
52733                 this.split.useShim = config.useShim === true;
52734                 this.split.getMaximumSize = this[this.position == 'north' || this.position == 'south' ? 'getVMaxSize' : 'getHMaxSize'].createDelegate(this);
52735                 if(this.useSplitTips){
52736                     this.split.el.dom.title = config.collapsible ? this.collapsibleSplitTip : this.splitTip;
52737                 }
52738                 if(config.collapsible){
52739                     this.split.el.on("dblclick", this.collapse,  this);
52740                 }
52741             }
52742             if(typeof config.minSize != "undefined"){
52743                 this.split.minSize = config.minSize;
52744             }
52745             if(typeof config.maxSize != "undefined"){
52746                 this.split.maxSize = config.maxSize;
52747             }
52748             if(config.hideWhenEmpty || config.hidden || config.collapsed){
52749                 this.hideSplitter();
52750             }
52751         }
52752     },
52753
52754     getHMaxSize : function(){
52755          var cmax = this.config.maxSize || 10000;
52756          var center = this.mgr.getRegion("center");
52757          return Math.min(cmax, (this.el.getWidth()+center.getEl().getWidth())-center.getMinWidth());
52758     },
52759
52760     getVMaxSize : function(){
52761          var cmax = this.config.maxSize || 10000;
52762          var center = this.mgr.getRegion("center");
52763          return Math.min(cmax, (this.el.getHeight()+center.getEl().getHeight())-center.getMinHeight());
52764     },
52765
52766     onSplitMove : function(split, newSize){
52767         this.fireEvent("resized", this, newSize);
52768     },
52769     
52770     /** 
52771      * Returns the {@link Roo.SplitBar} for this region.
52772      * @return {Roo.SplitBar}
52773      */
52774     getSplitBar : function(){
52775         return this.split;
52776     },
52777     
52778     hide : function(){
52779         this.hideSplitter();
52780         Roo.SplitLayoutRegion.superclass.hide.call(this);
52781     },
52782
52783     hideSplitter : function(){
52784         if(this.split){
52785             this.split.el.setLocation(-2000,-2000);
52786             this.split.el.hide();
52787         }
52788     },
52789
52790     show : function(){
52791         if(this.split){
52792             this.split.el.show();
52793         }
52794         Roo.SplitLayoutRegion.superclass.show.call(this);
52795     },
52796     
52797     beforeSlide: function(){
52798         if(Roo.isGecko){// firefox overflow auto bug workaround
52799             this.bodyEl.clip();
52800             if(this.tabs) {
52801                 this.tabs.bodyEl.clip();
52802             }
52803             if(this.activePanel){
52804                 this.activePanel.getEl().clip();
52805                 
52806                 if(this.activePanel.beforeSlide){
52807                     this.activePanel.beforeSlide();
52808                 }
52809             }
52810         }
52811     },
52812     
52813     afterSlide : function(){
52814         if(Roo.isGecko){// firefox overflow auto bug workaround
52815             this.bodyEl.unclip();
52816             if(this.tabs) {
52817                 this.tabs.bodyEl.unclip();
52818             }
52819             if(this.activePanel){
52820                 this.activePanel.getEl().unclip();
52821                 if(this.activePanel.afterSlide){
52822                     this.activePanel.afterSlide();
52823                 }
52824             }
52825         }
52826     },
52827
52828     initAutoHide : function(){
52829         if(this.autoHide !== false){
52830             if(!this.autoHideHd){
52831                 var st = new Roo.util.DelayedTask(this.slideIn, this);
52832                 this.autoHideHd = {
52833                     "mouseout": function(e){
52834                         if(!e.within(this.el, true)){
52835                             st.delay(500);
52836                         }
52837                     },
52838                     "mouseover" : function(e){
52839                         st.cancel();
52840                     },
52841                     scope : this
52842                 };
52843             }
52844             this.el.on(this.autoHideHd);
52845         }
52846     },
52847
52848     clearAutoHide : function(){
52849         if(this.autoHide !== false){
52850             this.el.un("mouseout", this.autoHideHd.mouseout);
52851             this.el.un("mouseover", this.autoHideHd.mouseover);
52852         }
52853     },
52854
52855     clearMonitor : function(){
52856         Roo.get(document).un("click", this.slideInIf, this);
52857     },
52858
52859     // these names are backwards but not changed for compat
52860     slideOut : function(){
52861         if(this.isSlid || this.el.hasActiveFx()){
52862             return;
52863         }
52864         this.isSlid = true;
52865         if(this.collapseBtn){
52866             this.collapseBtn.hide();
52867         }
52868         this.closeBtnState = this.closeBtn.getStyle('display');
52869         this.closeBtn.hide();
52870         if(this.stickBtn){
52871             this.stickBtn.show();
52872         }
52873         this.el.show();
52874         this.el.alignTo(this.collapsedEl, this.getCollapseAnchor());
52875         this.beforeSlide();
52876         this.el.setStyle("z-index", 10001);
52877         this.el.slideIn(this.getSlideAnchor(), {
52878             callback: function(){
52879                 this.afterSlide();
52880                 this.initAutoHide();
52881                 Roo.get(document).on("click", this.slideInIf, this);
52882                 this.fireEvent("slideshow", this);
52883             },
52884             scope: this,
52885             block: true
52886         });
52887     },
52888
52889     afterSlideIn : function(){
52890         this.clearAutoHide();
52891         this.isSlid = false;
52892         this.clearMonitor();
52893         this.el.setStyle("z-index", "");
52894         if(this.collapseBtn){
52895             this.collapseBtn.show();
52896         }
52897         this.closeBtn.setStyle('display', this.closeBtnState);
52898         if(this.stickBtn){
52899             this.stickBtn.hide();
52900         }
52901         this.fireEvent("slidehide", this);
52902     },
52903
52904     slideIn : function(cb){
52905         if(!this.isSlid || this.el.hasActiveFx()){
52906             Roo.callback(cb);
52907             return;
52908         }
52909         this.isSlid = false;
52910         this.beforeSlide();
52911         this.el.slideOut(this.getSlideAnchor(), {
52912             callback: function(){
52913                 this.el.setLeftTop(-10000, -10000);
52914                 this.afterSlide();
52915                 this.afterSlideIn();
52916                 Roo.callback(cb);
52917             },
52918             scope: this,
52919             block: true
52920         });
52921     },
52922     
52923     slideInIf : function(e){
52924         if(!e.within(this.el)){
52925             this.slideIn();
52926         }
52927     },
52928
52929     animateCollapse : function(){
52930         this.beforeSlide();
52931         this.el.setStyle("z-index", 20000);
52932         var anchor = this.getSlideAnchor();
52933         this.el.slideOut(anchor, {
52934             callback : function(){
52935                 this.el.setStyle("z-index", "");
52936                 this.collapsedEl.slideIn(anchor, {duration:.3});
52937                 this.afterSlide();
52938                 this.el.setLocation(-10000,-10000);
52939                 this.el.hide();
52940                 this.fireEvent("collapsed", this);
52941             },
52942             scope: this,
52943             block: true
52944         });
52945     },
52946
52947     animateExpand : function(){
52948         this.beforeSlide();
52949         this.el.alignTo(this.collapsedEl, this.getCollapseAnchor(), this.getExpandAdj());
52950         this.el.setStyle("z-index", 20000);
52951         this.collapsedEl.hide({
52952             duration:.1
52953         });
52954         this.el.slideIn(this.getSlideAnchor(), {
52955             callback : function(){
52956                 this.el.setStyle("z-index", "");
52957                 this.afterSlide();
52958                 if(this.split){
52959                     this.split.el.show();
52960                 }
52961                 this.fireEvent("invalidated", this);
52962                 this.fireEvent("expanded", this);
52963             },
52964             scope: this,
52965             block: true
52966         });
52967     },
52968
52969     anchors : {
52970         "west" : "left",
52971         "east" : "right",
52972         "north" : "top",
52973         "south" : "bottom"
52974     },
52975
52976     sanchors : {
52977         "west" : "l",
52978         "east" : "r",
52979         "north" : "t",
52980         "south" : "b"
52981     },
52982
52983     canchors : {
52984         "west" : "tl-tr",
52985         "east" : "tr-tl",
52986         "north" : "tl-bl",
52987         "south" : "bl-tl"
52988     },
52989
52990     getAnchor : function(){
52991         return this.anchors[this.position];
52992     },
52993
52994     getCollapseAnchor : function(){
52995         return this.canchors[this.position];
52996     },
52997
52998     getSlideAnchor : function(){
52999         return this.sanchors[this.position];
53000     },
53001
53002     getAlignAdj : function(){
53003         var cm = this.cmargins;
53004         switch(this.position){
53005             case "west":
53006                 return [0, 0];
53007             break;
53008             case "east":
53009                 return [0, 0];
53010             break;
53011             case "north":
53012                 return [0, 0];
53013             break;
53014             case "south":
53015                 return [0, 0];
53016             break;
53017         }
53018     },
53019
53020     getExpandAdj : function(){
53021         var c = this.collapsedEl, cm = this.cmargins;
53022         switch(this.position){
53023             case "west":
53024                 return [-(cm.right+c.getWidth()+cm.left), 0];
53025             break;
53026             case "east":
53027                 return [cm.right+c.getWidth()+cm.left, 0];
53028             break;
53029             case "north":
53030                 return [0, -(cm.top+cm.bottom+c.getHeight())];
53031             break;
53032             case "south":
53033                 return [0, cm.top+cm.bottom+c.getHeight()];
53034             break;
53035         }
53036     }
53037 });/*
53038  * Based on:
53039  * Ext JS Library 1.1.1
53040  * Copyright(c) 2006-2007, Ext JS, LLC.
53041  *
53042  * Originally Released Under LGPL - original licence link has changed is not relivant.
53043  *
53044  * Fork - LGPL
53045  * <script type="text/javascript">
53046  */
53047 /*
53048  * These classes are private internal classes
53049  */
53050 Roo.CenterLayoutRegion = function(mgr, config){
53051     Roo.LayoutRegion.call(this, mgr, config, "center");
53052     this.visible = true;
53053     this.minWidth = config.minWidth || 20;
53054     this.minHeight = config.minHeight || 20;
53055 };
53056
53057 Roo.extend(Roo.CenterLayoutRegion, Roo.LayoutRegion, {
53058     hide : function(){
53059         // center panel can't be hidden
53060     },
53061     
53062     show : function(){
53063         // center panel can't be hidden
53064     },
53065     
53066     getMinWidth: function(){
53067         return this.minWidth;
53068     },
53069     
53070     getMinHeight: function(){
53071         return this.minHeight;
53072     }
53073 });
53074
53075
53076 Roo.NorthLayoutRegion = function(mgr, config){
53077     Roo.LayoutRegion.call(this, mgr, config, "north", "n-resize");
53078     if(this.split){
53079         this.split.placement = Roo.SplitBar.TOP;
53080         this.split.orientation = Roo.SplitBar.VERTICAL;
53081         this.split.el.addClass("x-layout-split-v");
53082     }
53083     var size = config.initialSize || config.height;
53084     if(typeof size != "undefined"){
53085         this.el.setHeight(size);
53086     }
53087 };
53088 Roo.extend(Roo.NorthLayoutRegion, Roo.SplitLayoutRegion, {
53089     orientation: Roo.SplitBar.VERTICAL,
53090     getBox : function(){
53091         if(this.collapsed){
53092             return this.collapsedEl.getBox();
53093         }
53094         var box = this.el.getBox();
53095         if(this.split){
53096             box.height += this.split.el.getHeight();
53097         }
53098         return box;
53099     },
53100     
53101     updateBox : function(box){
53102         if(this.split && !this.collapsed){
53103             box.height -= this.split.el.getHeight();
53104             this.split.el.setLeft(box.x);
53105             this.split.el.setTop(box.y+box.height);
53106             this.split.el.setWidth(box.width);
53107         }
53108         if(this.collapsed){
53109             this.updateBody(box.width, null);
53110         }
53111         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53112     }
53113 });
53114
53115 Roo.SouthLayoutRegion = function(mgr, config){
53116     Roo.SplitLayoutRegion.call(this, mgr, config, "south", "s-resize");
53117     if(this.split){
53118         this.split.placement = Roo.SplitBar.BOTTOM;
53119         this.split.orientation = Roo.SplitBar.VERTICAL;
53120         this.split.el.addClass("x-layout-split-v");
53121     }
53122     var size = config.initialSize || config.height;
53123     if(typeof size != "undefined"){
53124         this.el.setHeight(size);
53125     }
53126 };
53127 Roo.extend(Roo.SouthLayoutRegion, Roo.SplitLayoutRegion, {
53128     orientation: Roo.SplitBar.VERTICAL,
53129     getBox : function(){
53130         if(this.collapsed){
53131             return this.collapsedEl.getBox();
53132         }
53133         var box = this.el.getBox();
53134         if(this.split){
53135             var sh = this.split.el.getHeight();
53136             box.height += sh;
53137             box.y -= sh;
53138         }
53139         return box;
53140     },
53141     
53142     updateBox : function(box){
53143         if(this.split && !this.collapsed){
53144             var sh = this.split.el.getHeight();
53145             box.height -= sh;
53146             box.y += sh;
53147             this.split.el.setLeft(box.x);
53148             this.split.el.setTop(box.y-sh);
53149             this.split.el.setWidth(box.width);
53150         }
53151         if(this.collapsed){
53152             this.updateBody(box.width, null);
53153         }
53154         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53155     }
53156 });
53157
53158 Roo.EastLayoutRegion = function(mgr, config){
53159     Roo.SplitLayoutRegion.call(this, mgr, config, "east", "e-resize");
53160     if(this.split){
53161         this.split.placement = Roo.SplitBar.RIGHT;
53162         this.split.orientation = Roo.SplitBar.HORIZONTAL;
53163         this.split.el.addClass("x-layout-split-h");
53164     }
53165     var size = config.initialSize || config.width;
53166     if(typeof size != "undefined"){
53167         this.el.setWidth(size);
53168     }
53169 };
53170 Roo.extend(Roo.EastLayoutRegion, Roo.SplitLayoutRegion, {
53171     orientation: Roo.SplitBar.HORIZONTAL,
53172     getBox : function(){
53173         if(this.collapsed){
53174             return this.collapsedEl.getBox();
53175         }
53176         var box = this.el.getBox();
53177         if(this.split){
53178             var sw = this.split.el.getWidth();
53179             box.width += sw;
53180             box.x -= sw;
53181         }
53182         return box;
53183     },
53184
53185     updateBox : function(box){
53186         if(this.split && !this.collapsed){
53187             var sw = this.split.el.getWidth();
53188             box.width -= sw;
53189             this.split.el.setLeft(box.x);
53190             this.split.el.setTop(box.y);
53191             this.split.el.setHeight(box.height);
53192             box.x += sw;
53193         }
53194         if(this.collapsed){
53195             this.updateBody(null, box.height);
53196         }
53197         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53198     }
53199 });
53200
53201 Roo.WestLayoutRegion = function(mgr, config){
53202     Roo.SplitLayoutRegion.call(this, mgr, config, "west", "w-resize");
53203     if(this.split){
53204         this.split.placement = Roo.SplitBar.LEFT;
53205         this.split.orientation = Roo.SplitBar.HORIZONTAL;
53206         this.split.el.addClass("x-layout-split-h");
53207     }
53208     var size = config.initialSize || config.width;
53209     if(typeof size != "undefined"){
53210         this.el.setWidth(size);
53211     }
53212 };
53213 Roo.extend(Roo.WestLayoutRegion, Roo.SplitLayoutRegion, {
53214     orientation: Roo.SplitBar.HORIZONTAL,
53215     getBox : function(){
53216         if(this.collapsed){
53217             return this.collapsedEl.getBox();
53218         }
53219         var box = this.el.getBox();
53220         if(this.split){
53221             box.width += this.split.el.getWidth();
53222         }
53223         return box;
53224     },
53225     
53226     updateBox : function(box){
53227         if(this.split && !this.collapsed){
53228             var sw = this.split.el.getWidth();
53229             box.width -= sw;
53230             this.split.el.setLeft(box.x+box.width);
53231             this.split.el.setTop(box.y);
53232             this.split.el.setHeight(box.height);
53233         }
53234         if(this.collapsed){
53235             this.updateBody(null, box.height);
53236         }
53237         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53238     }
53239 });
53240 /*
53241  * Based on:
53242  * Ext JS Library 1.1.1
53243  * Copyright(c) 2006-2007, Ext JS, LLC.
53244  *
53245  * Originally Released Under LGPL - original licence link has changed is not relivant.
53246  *
53247  * Fork - LGPL
53248  * <script type="text/javascript">
53249  */
53250  
53251  
53252 /*
53253  * Private internal class for reading and applying state
53254  */
53255 Roo.LayoutStateManager = function(layout){
53256      // default empty state
53257      this.state = {
53258         north: {},
53259         south: {},
53260         east: {},
53261         west: {}       
53262     };
53263 };
53264
53265 Roo.LayoutStateManager.prototype = {
53266     init : function(layout, provider){
53267         this.provider = provider;
53268         var state = provider.get(layout.id+"-layout-state");
53269         if(state){
53270             var wasUpdating = layout.isUpdating();
53271             if(!wasUpdating){
53272                 layout.beginUpdate();
53273             }
53274             for(var key in state){
53275                 if(typeof state[key] != "function"){
53276                     var rstate = state[key];
53277                     var r = layout.getRegion(key);
53278                     if(r && rstate){
53279                         if(rstate.size){
53280                             r.resizeTo(rstate.size);
53281                         }
53282                         if(rstate.collapsed == true){
53283                             r.collapse(true);
53284                         }else{
53285                             r.expand(null, true);
53286                         }
53287                     }
53288                 }
53289             }
53290             if(!wasUpdating){
53291                 layout.endUpdate();
53292             }
53293             this.state = state; 
53294         }
53295         this.layout = layout;
53296         layout.on("regionresized", this.onRegionResized, this);
53297         layout.on("regioncollapsed", this.onRegionCollapsed, this);
53298         layout.on("regionexpanded", this.onRegionExpanded, this);
53299     },
53300     
53301     storeState : function(){
53302         this.provider.set(this.layout.id+"-layout-state", this.state);
53303     },
53304     
53305     onRegionResized : function(region, newSize){
53306         this.state[region.getPosition()].size = newSize;
53307         this.storeState();
53308     },
53309     
53310     onRegionCollapsed : function(region){
53311         this.state[region.getPosition()].collapsed = true;
53312         this.storeState();
53313     },
53314     
53315     onRegionExpanded : function(region){
53316         this.state[region.getPosition()].collapsed = false;
53317         this.storeState();
53318     }
53319 };/*
53320  * Based on:
53321  * Ext JS Library 1.1.1
53322  * Copyright(c) 2006-2007, Ext JS, LLC.
53323  *
53324  * Originally Released Under LGPL - original licence link has changed is not relivant.
53325  *
53326  * Fork - LGPL
53327  * <script type="text/javascript">
53328  */
53329 /**
53330  * @class Roo.ContentPanel
53331  * @extends Roo.util.Observable
53332  * A basic ContentPanel element.
53333  * @cfg {Boolean}   fitToFrame    True for this panel to adjust its size to fit when the region resizes  (defaults to false)
53334  * @cfg {Boolean}   fitContainer   When using {@link #fitToFrame} and {@link #resizeEl}, you can also fit the parent container  (defaults to false)
53335  * @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
53336  * @cfg {Boolean}   closable      True if the panel can be closed/removed
53337  * @cfg {Boolean}   background    True if the panel should not be activated when it is added (defaults to false)
53338  * @cfg {String/HTMLElement/Element} resizeEl An element to resize if {@link #fitToFrame} is true (instead of this panel's element)
53339  * @cfg {Toolbar}   toolbar       A toolbar for this panel
53340  * @cfg {Boolean} autoScroll    True to scroll overflow in this panel (use with {@link #fitToFrame})
53341  * @cfg {String} title          The title for this panel
53342  * @cfg {Array} adjustments     Values to <b>add</b> to the width/height when doing a {@link #fitToFrame} (default is [0, 0])
53343  * @cfg {String} url            Calls {@link #setUrl} with this value
53344  * @cfg {String} region         (center|north|south|east|west) which region to put this panel on (when used with xtype constructors)
53345  * @cfg {String/Object} params  When used with {@link #url}, calls {@link #setUrl} with this value
53346  * @cfg {Boolean} loadOnce      When used with {@link #url}, calls {@link #setUrl} with this value
53347  * @cfg {String}    content        Raw content to fill content panel with (uses setContent on construction.)
53348
53349  * @constructor
53350  * Create a new ContentPanel.
53351  * @param {String/HTMLElement/Roo.Element} el The container element for this panel
53352  * @param {String/Object} config A string to set only the title or a config object
53353  * @param {String} content (optional) Set the HTML content for this panel
53354  * @param {String} region (optional) Used by xtype constructors to add to regions. (values center,east,west,south,north)
53355  */
53356 Roo.ContentPanel = function(el, config, content){
53357     
53358      
53359     /*
53360     if(el.autoCreate || el.xtype){ // xtype is available if this is called from factory
53361         config = el;
53362         el = Roo.id();
53363     }
53364     if (config && config.parentLayout) { 
53365         el = config.parentLayout.el.createChild(); 
53366     }
53367     */
53368     if(el.autoCreate){ // xtype is available if this is called from factory
53369         config = el;
53370         el = Roo.id();
53371     }
53372     this.el = Roo.get(el);
53373     if(!this.el && config && config.autoCreate){
53374         if(typeof config.autoCreate == "object"){
53375             if(!config.autoCreate.id){
53376                 config.autoCreate.id = config.id||el;
53377             }
53378             this.el = Roo.DomHelper.append(document.body,
53379                         config.autoCreate, true);
53380         }else{
53381             this.el = Roo.DomHelper.append(document.body,
53382                         {tag: "div", cls: "x-layout-inactive-content", id: config.id||el}, true);
53383         }
53384     }
53385     this.closable = false;
53386     this.loaded = false;
53387     this.active = false;
53388     if(typeof config == "string"){
53389         this.title = config;
53390     }else{
53391         Roo.apply(this, config);
53392     }
53393     
53394     if (this.toolbar && !this.toolbar.el && this.toolbar.xtype) {
53395         this.wrapEl = this.el.wrap();
53396         this.toolbar.container = this.el.insertSibling(false, 'before');
53397         this.toolbar = new Roo.Toolbar(this.toolbar);
53398     }
53399     
53400     // xtype created footer. - not sure if will work as we normally have to render first..
53401     if (this.footer && !this.footer.el && this.footer.xtype) {
53402         if (!this.wrapEl) {
53403             this.wrapEl = this.el.wrap();
53404         }
53405     
53406         this.footer.container = this.wrapEl.createChild();
53407          
53408         this.footer = Roo.factory(this.footer, Roo);
53409         
53410     }
53411     
53412     if(this.resizeEl){
53413         this.resizeEl = Roo.get(this.resizeEl, true);
53414     }else{
53415         this.resizeEl = this.el;
53416     }
53417     // handle view.xtype
53418     
53419  
53420     
53421     
53422     this.addEvents({
53423         /**
53424          * @event activate
53425          * Fires when this panel is activated. 
53426          * @param {Roo.ContentPanel} this
53427          */
53428         "activate" : true,
53429         /**
53430          * @event deactivate
53431          * Fires when this panel is activated. 
53432          * @param {Roo.ContentPanel} this
53433          */
53434         "deactivate" : true,
53435
53436         /**
53437          * @event resize
53438          * Fires when this panel is resized if fitToFrame is true.
53439          * @param {Roo.ContentPanel} this
53440          * @param {Number} width The width after any component adjustments
53441          * @param {Number} height The height after any component adjustments
53442          */
53443         "resize" : true,
53444         
53445          /**
53446          * @event render
53447          * Fires when this tab is created
53448          * @param {Roo.ContentPanel} this
53449          */
53450         "render" : true
53451          
53452         
53453     });
53454     
53455
53456     
53457     
53458     if(this.autoScroll){
53459         this.resizeEl.setStyle("overflow", "auto");
53460     } else {
53461         // fix randome scrolling
53462         this.el.on('scroll', function() {
53463             Roo.log('fix random scolling');
53464             this.scrollTo('top',0); 
53465         });
53466     }
53467     content = content || this.content;
53468     if(content){
53469         this.setContent(content);
53470     }
53471     if(config && config.url){
53472         this.setUrl(this.url, this.params, this.loadOnce);
53473     }
53474     
53475     
53476     
53477     Roo.ContentPanel.superclass.constructor.call(this);
53478     
53479     if (this.view && typeof(this.view.xtype) != 'undefined') {
53480         this.view.el = this.el.appendChild(document.createElement("div"));
53481         this.view = Roo.factory(this.view); 
53482         this.view.render  &&  this.view.render(false, '');  
53483     }
53484     
53485     
53486     this.fireEvent('render', this);
53487 };
53488
53489 Roo.extend(Roo.ContentPanel, Roo.util.Observable, {
53490     tabTip:'',
53491     setRegion : function(region){
53492         this.region = region;
53493         if(region){
53494            this.el.replaceClass("x-layout-inactive-content", "x-layout-active-content");
53495         }else{
53496            this.el.replaceClass("x-layout-active-content", "x-layout-inactive-content");
53497         } 
53498     },
53499     
53500     /**
53501      * Returns the toolbar for this Panel if one was configured. 
53502      * @return {Roo.Toolbar} 
53503      */
53504     getToolbar : function(){
53505         return this.toolbar;
53506     },
53507     
53508     setActiveState : function(active){
53509         this.active = active;
53510         if(!active){
53511             this.fireEvent("deactivate", this);
53512         }else{
53513             this.fireEvent("activate", this);
53514         }
53515     },
53516     /**
53517      * Updates this panel's element
53518      * @param {String} content The new content
53519      * @param {Boolean} loadScripts (optional) true to look for and process scripts
53520     */
53521     setContent : function(content, loadScripts){
53522         this.el.update(content, loadScripts);
53523     },
53524
53525     ignoreResize : function(w, h){
53526         if(this.lastSize && this.lastSize.width == w && this.lastSize.height == h){
53527             return true;
53528         }else{
53529             this.lastSize = {width: w, height: h};
53530             return false;
53531         }
53532     },
53533     /**
53534      * Get the {@link Roo.UpdateManager} for this panel. Enables you to perform Ajax updates.
53535      * @return {Roo.UpdateManager} The UpdateManager
53536      */
53537     getUpdateManager : function(){
53538         return this.el.getUpdateManager();
53539     },
53540      /**
53541      * Loads this content panel immediately with content from XHR. Note: to delay loading until the panel is activated, use {@link #setUrl}.
53542      * @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:
53543 <pre><code>
53544 panel.load({
53545     url: "your-url.php",
53546     params: {param1: "foo", param2: "bar"}, // or a URL encoded string
53547     callback: yourFunction,
53548     scope: yourObject, //(optional scope)
53549     discardUrl: false,
53550     nocache: false,
53551     text: "Loading...",
53552     timeout: 30,
53553     scripts: false
53554 });
53555 </code></pre>
53556      * The only required property is <i>url</i>. The optional properties <i>nocache</i>, <i>text</i> and <i>scripts</i>
53557      * 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.
53558      * @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}
53559      * @param {Function} callback (optional) Callback when transaction is complete -- called with signature (oElement, bSuccess, oResponse)
53560      * @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.
53561      * @return {Roo.ContentPanel} this
53562      */
53563     load : function(){
53564         var um = this.el.getUpdateManager();
53565         um.update.apply(um, arguments);
53566         return this;
53567     },
53568
53569
53570     /**
53571      * 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.
53572      * @param {String/Function} url The URL to load the content from or a function to call to get the URL
53573      * @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)
53574      * @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)
53575      * @return {Roo.UpdateManager} The UpdateManager
53576      */
53577     setUrl : function(url, params, loadOnce){
53578         if(this.refreshDelegate){
53579             this.removeListener("activate", this.refreshDelegate);
53580         }
53581         this.refreshDelegate = this._handleRefresh.createDelegate(this, [url, params, loadOnce]);
53582         this.on("activate", this.refreshDelegate);
53583         return this.el.getUpdateManager();
53584     },
53585     
53586     _handleRefresh : function(url, params, loadOnce){
53587         if(!loadOnce || !this.loaded){
53588             var updater = this.el.getUpdateManager();
53589             updater.update(url, params, this._setLoaded.createDelegate(this));
53590         }
53591     },
53592     
53593     _setLoaded : function(){
53594         this.loaded = true;
53595     }, 
53596     
53597     /**
53598      * Returns this panel's id
53599      * @return {String} 
53600      */
53601     getId : function(){
53602         return this.el.id;
53603     },
53604     
53605     /** 
53606      * Returns this panel's element - used by regiosn to add.
53607      * @return {Roo.Element} 
53608      */
53609     getEl : function(){
53610         return this.wrapEl || this.el;
53611     },
53612     
53613     adjustForComponents : function(width, height)
53614     {
53615         //Roo.log('adjustForComponents ');
53616         if(this.resizeEl != this.el){
53617             width -= this.el.getFrameWidth('lr');
53618             height -= this.el.getFrameWidth('tb');
53619         }
53620         if(this.toolbar){
53621             var te = this.toolbar.getEl();
53622             height -= te.getHeight();
53623             te.setWidth(width);
53624         }
53625         if(this.footer){
53626             var te = this.footer.getEl();
53627             //Roo.log("footer:" + te.getHeight());
53628             
53629             height -= te.getHeight();
53630             te.setWidth(width);
53631         }
53632         
53633         
53634         if(this.adjustments){
53635             width += this.adjustments[0];
53636             height += this.adjustments[1];
53637         }
53638         return {"width": width, "height": height};
53639     },
53640     
53641     setSize : function(width, height){
53642         if(this.fitToFrame && !this.ignoreResize(width, height)){
53643             if(this.fitContainer && this.resizeEl != this.el){
53644                 this.el.setSize(width, height);
53645             }
53646             var size = this.adjustForComponents(width, height);
53647             this.resizeEl.setSize(this.autoWidth ? "auto" : size.width, this.autoHeight ? "auto" : size.height);
53648             this.fireEvent('resize', this, size.width, size.height);
53649         }
53650     },
53651     
53652     /**
53653      * Returns this panel's title
53654      * @return {String} 
53655      */
53656     getTitle : function(){
53657         return this.title;
53658     },
53659     
53660     /**
53661      * Set this panel's title
53662      * @param {String} title
53663      */
53664     setTitle : function(title){
53665         this.title = title;
53666         if(this.region){
53667             this.region.updatePanelTitle(this, title);
53668         }
53669     },
53670     
53671     /**
53672      * Returns true is this panel was configured to be closable
53673      * @return {Boolean} 
53674      */
53675     isClosable : function(){
53676         return this.closable;
53677     },
53678     
53679     beforeSlide : function(){
53680         this.el.clip();
53681         this.resizeEl.clip();
53682     },
53683     
53684     afterSlide : function(){
53685         this.el.unclip();
53686         this.resizeEl.unclip();
53687     },
53688     
53689     /**
53690      *   Force a content refresh from the URL specified in the {@link #setUrl} method.
53691      *   Will fail silently if the {@link #setUrl} method has not been called.
53692      *   This does not activate the panel, just updates its content.
53693      */
53694     refresh : function(){
53695         if(this.refreshDelegate){
53696            this.loaded = false;
53697            this.refreshDelegate();
53698         }
53699     },
53700     
53701     /**
53702      * Destroys this panel
53703      */
53704     destroy : function(){
53705         this.el.removeAllListeners();
53706         var tempEl = document.createElement("span");
53707         tempEl.appendChild(this.el.dom);
53708         tempEl.innerHTML = "";
53709         this.el.remove();
53710         this.el = null;
53711     },
53712     
53713     /**
53714      * form - if the content panel contains a form - this is a reference to it.
53715      * @type {Roo.form.Form}
53716      */
53717     form : false,
53718     /**
53719      * view - if the content panel contains a view (Roo.DatePicker / Roo.View / Roo.JsonView)
53720      *    This contains a reference to it.
53721      * @type {Roo.View}
53722      */
53723     view : false,
53724     
53725       /**
53726      * Adds a xtype elements to the panel - currently only supports Forms, View, JsonView.
53727      * <pre><code>
53728
53729 layout.addxtype({
53730        xtype : 'Form',
53731        items: [ .... ]
53732    }
53733 );
53734
53735 </code></pre>
53736      * @param {Object} cfg Xtype definition of item to add.
53737      */
53738     
53739     addxtype : function(cfg) {
53740         // add form..
53741         if (cfg.xtype.match(/^Form$/)) {
53742             
53743             var el;
53744             //if (this.footer) {
53745             //    el = this.footer.container.insertSibling(false, 'before');
53746             //} else {
53747                 el = this.el.createChild();
53748             //}
53749
53750             this.form = new  Roo.form.Form(cfg);
53751             
53752             
53753             if ( this.form.allItems.length) {
53754                 this.form.render(el.dom);
53755             }
53756             return this.form;
53757         }
53758         // should only have one of theses..
53759         if ([ 'View', 'JsonView', 'DatePicker'].indexOf(cfg.xtype) > -1) {
53760             // views.. should not be just added - used named prop 'view''
53761             
53762             cfg.el = this.el.appendChild(document.createElement("div"));
53763             // factory?
53764             
53765             var ret = new Roo.factory(cfg);
53766              
53767              ret.render && ret.render(false, ''); // render blank..
53768             this.view = ret;
53769             return ret;
53770         }
53771         return false;
53772     }
53773 });
53774
53775 /**
53776  * @class Roo.GridPanel
53777  * @extends Roo.ContentPanel
53778  * @constructor
53779  * Create a new GridPanel.
53780  * @param {Roo.grid.Grid} grid The grid for this panel
53781  * @param {String/Object} config A string to set only the panel's title, or a config object
53782  */
53783 Roo.GridPanel = function(grid, config){
53784     
53785   
53786     this.wrapper = Roo.DomHelper.append(document.body, // wrapper for IE7 strict & safari scroll issue
53787         {tag: "div", cls: "x-layout-grid-wrapper x-layout-inactive-content"}, true);
53788         
53789     this.wrapper.dom.appendChild(grid.getGridEl().dom);
53790     
53791     Roo.GridPanel.superclass.constructor.call(this, this.wrapper, config);
53792     
53793     if(this.toolbar){
53794         this.toolbar.el.insertBefore(this.wrapper.dom.firstChild);
53795     }
53796     // xtype created footer. - not sure if will work as we normally have to render first..
53797     if (this.footer && !this.footer.el && this.footer.xtype) {
53798         
53799         this.footer.container = this.grid.getView().getFooterPanel(true);
53800         this.footer.dataSource = this.grid.dataSource;
53801         this.footer = Roo.factory(this.footer, Roo);
53802         
53803     }
53804     
53805     grid.monitorWindowResize = false; // turn off autosizing
53806     grid.autoHeight = false;
53807     grid.autoWidth = false;
53808     this.grid = grid;
53809     this.grid.getGridEl().replaceClass("x-layout-inactive-content", "x-layout-component-panel");
53810 };
53811
53812 Roo.extend(Roo.GridPanel, Roo.ContentPanel, {
53813     getId : function(){
53814         return this.grid.id;
53815     },
53816     
53817     /**
53818      * Returns the grid for this panel
53819      * @return {Roo.grid.Grid} 
53820      */
53821     getGrid : function(){
53822         return this.grid;    
53823     },
53824     
53825     setSize : function(width, height){
53826         if(!this.ignoreResize(width, height)){
53827             var grid = this.grid;
53828             var size = this.adjustForComponents(width, height);
53829             grid.getGridEl().setSize(size.width, size.height);
53830             grid.autoSize();
53831         }
53832     },
53833     
53834     beforeSlide : function(){
53835         this.grid.getView().scroller.clip();
53836     },
53837     
53838     afterSlide : function(){
53839         this.grid.getView().scroller.unclip();
53840     },
53841     
53842     destroy : function(){
53843         this.grid.destroy();
53844         delete this.grid;
53845         Roo.GridPanel.superclass.destroy.call(this); 
53846     }
53847 });
53848
53849
53850 /**
53851  * @class Roo.NestedLayoutPanel
53852  * @extends Roo.ContentPanel
53853  * @constructor
53854  * Create a new NestedLayoutPanel.
53855  * 
53856  * 
53857  * @param {Roo.BorderLayout} layout The layout for this panel
53858  * @param {String/Object} config A string to set only the title or a config object
53859  */
53860 Roo.NestedLayoutPanel = function(layout, config)
53861 {
53862     // construct with only one argument..
53863     /* FIXME - implement nicer consturctors
53864     if (layout.layout) {
53865         config = layout;
53866         layout = config.layout;
53867         delete config.layout;
53868     }
53869     if (layout.xtype && !layout.getEl) {
53870         // then layout needs constructing..
53871         layout = Roo.factory(layout, Roo);
53872     }
53873     */
53874     
53875     
53876     Roo.NestedLayoutPanel.superclass.constructor.call(this, layout.getEl(), config);
53877     
53878     layout.monitorWindowResize = false; // turn off autosizing
53879     this.layout = layout;
53880     this.layout.getEl().addClass("x-layout-nested-layout");
53881     
53882     
53883     
53884     
53885 };
53886
53887 Roo.extend(Roo.NestedLayoutPanel, Roo.ContentPanel, {
53888
53889     setSize : function(width, height){
53890         if(!this.ignoreResize(width, height)){
53891             var size = this.adjustForComponents(width, height);
53892             var el = this.layout.getEl();
53893             el.setSize(size.width, size.height);
53894             var touch = el.dom.offsetWidth;
53895             this.layout.layout();
53896             // ie requires a double layout on the first pass
53897             if(Roo.isIE && !this.initialized){
53898                 this.initialized = true;
53899                 this.layout.layout();
53900             }
53901         }
53902     },
53903     
53904     // activate all subpanels if not currently active..
53905     
53906     setActiveState : function(active){
53907         this.active = active;
53908         if(!active){
53909             this.fireEvent("deactivate", this);
53910             return;
53911         }
53912         
53913         this.fireEvent("activate", this);
53914         // not sure if this should happen before or after..
53915         if (!this.layout) {
53916             return; // should not happen..
53917         }
53918         var reg = false;
53919         for (var r in this.layout.regions) {
53920             reg = this.layout.getRegion(r);
53921             if (reg.getActivePanel()) {
53922                 //reg.showPanel(reg.getActivePanel()); // force it to activate.. 
53923                 reg.setActivePanel(reg.getActivePanel());
53924                 continue;
53925             }
53926             if (!reg.panels.length) {
53927                 continue;
53928             }
53929             reg.showPanel(reg.getPanel(0));
53930         }
53931         
53932         
53933         
53934         
53935     },
53936     
53937     /**
53938      * Returns the nested BorderLayout for this panel
53939      * @return {Roo.BorderLayout} 
53940      */
53941     getLayout : function(){
53942         return this.layout;
53943     },
53944     
53945      /**
53946      * Adds a xtype elements to the layout of the nested panel
53947      * <pre><code>
53948
53949 panel.addxtype({
53950        xtype : 'ContentPanel',
53951        region: 'west',
53952        items: [ .... ]
53953    }
53954 );
53955
53956 panel.addxtype({
53957         xtype : 'NestedLayoutPanel',
53958         region: 'west',
53959         layout: {
53960            center: { },
53961            west: { }   
53962         },
53963         items : [ ... list of content panels or nested layout panels.. ]
53964    }
53965 );
53966 </code></pre>
53967      * @param {Object} cfg Xtype definition of item to add.
53968      */
53969     addxtype : function(cfg) {
53970         return this.layout.addxtype(cfg);
53971     
53972     }
53973 });
53974
53975 Roo.ScrollPanel = function(el, config, content){
53976     config = config || {};
53977     config.fitToFrame = true;
53978     Roo.ScrollPanel.superclass.constructor.call(this, el, config, content);
53979     
53980     this.el.dom.style.overflow = "hidden";
53981     var wrap = this.el.wrap({cls: "x-scroller x-layout-inactive-content"});
53982     this.el.removeClass("x-layout-inactive-content");
53983     this.el.on("mousewheel", this.onWheel, this);
53984
53985     var up = wrap.createChild({cls: "x-scroller-up", html: "&#160;"}, this.el.dom);
53986     var down = wrap.createChild({cls: "x-scroller-down", html: "&#160;"});
53987     up.unselectable(); down.unselectable();
53988     up.on("click", this.scrollUp, this);
53989     down.on("click", this.scrollDown, this);
53990     up.addClassOnOver("x-scroller-btn-over");
53991     down.addClassOnOver("x-scroller-btn-over");
53992     up.addClassOnClick("x-scroller-btn-click");
53993     down.addClassOnClick("x-scroller-btn-click");
53994     this.adjustments = [0, -(up.getHeight() + down.getHeight())];
53995
53996     this.resizeEl = this.el;
53997     this.el = wrap; this.up = up; this.down = down;
53998 };
53999
54000 Roo.extend(Roo.ScrollPanel, Roo.ContentPanel, {
54001     increment : 100,
54002     wheelIncrement : 5,
54003     scrollUp : function(){
54004         this.resizeEl.scroll("up", this.increment, {callback: this.afterScroll, scope: this});
54005     },
54006
54007     scrollDown : function(){
54008         this.resizeEl.scroll("down", this.increment, {callback: this.afterScroll, scope: this});
54009     },
54010
54011     afterScroll : function(){
54012         var el = this.resizeEl;
54013         var t = el.dom.scrollTop, h = el.dom.scrollHeight, ch = el.dom.clientHeight;
54014         this.up[t == 0 ? "addClass" : "removeClass"]("x-scroller-btn-disabled");
54015         this.down[h - t <= ch ? "addClass" : "removeClass"]("x-scroller-btn-disabled");
54016     },
54017
54018     setSize : function(){
54019         Roo.ScrollPanel.superclass.setSize.apply(this, arguments);
54020         this.afterScroll();
54021     },
54022
54023     onWheel : function(e){
54024         var d = e.getWheelDelta();
54025         this.resizeEl.dom.scrollTop -= (d*this.wheelIncrement);
54026         this.afterScroll();
54027         e.stopEvent();
54028     },
54029
54030     setContent : function(content, loadScripts){
54031         this.resizeEl.update(content, loadScripts);
54032     }
54033
54034 });
54035
54036
54037
54038
54039
54040
54041
54042
54043
54044 /**
54045  * @class Roo.TreePanel
54046  * @extends Roo.ContentPanel
54047  * @constructor
54048  * Create a new TreePanel. - defaults to fit/scoll contents.
54049  * @param {String/Object} config A string to set only the panel's title, or a config object
54050  * @cfg {Roo.tree.TreePanel} tree The tree TreePanel, with config etc.
54051  */
54052 Roo.TreePanel = function(config){
54053     var el = config.el;
54054     var tree = config.tree;
54055     delete config.tree; 
54056     delete config.el; // hopefull!
54057     
54058     // wrapper for IE7 strict & safari scroll issue
54059     
54060     var treeEl = el.createChild();
54061     config.resizeEl = treeEl;
54062     
54063     
54064     
54065     Roo.TreePanel.superclass.constructor.call(this, el, config);
54066  
54067  
54068     this.tree = new Roo.tree.TreePanel(treeEl , tree);
54069     //console.log(tree);
54070     this.on('activate', function()
54071     {
54072         if (this.tree.rendered) {
54073             return;
54074         }
54075         //console.log('render tree');
54076         this.tree.render();
54077     });
54078     // this should not be needed.. - it's actually the 'el' that resizes?
54079     // actuall it breaks the containerScroll - dragging nodes auto scroll at top
54080     
54081     //this.on('resize',  function (cp, w, h) {
54082     //        this.tree.innerCt.setWidth(w);
54083     //        this.tree.innerCt.setHeight(h);
54084     //        //this.tree.innerCt.setStyle('overflow-y', 'auto');
54085     //});
54086
54087         
54088     
54089 };
54090
54091 Roo.extend(Roo.TreePanel, Roo.ContentPanel, {   
54092     fitToFrame : true,
54093     autoScroll : true
54094 });
54095
54096
54097
54098
54099
54100
54101
54102
54103
54104
54105
54106 /*
54107  * Based on:
54108  * Ext JS Library 1.1.1
54109  * Copyright(c) 2006-2007, Ext JS, LLC.
54110  *
54111  * Originally Released Under LGPL - original licence link has changed is not relivant.
54112  *
54113  * Fork - LGPL
54114  * <script type="text/javascript">
54115  */
54116  
54117
54118 /**
54119  * @class Roo.ReaderLayout
54120  * @extends Roo.BorderLayout
54121  * This is a pre-built layout that represents a classic, 5-pane application.  It consists of a header, a primary
54122  * center region containing two nested regions (a top one for a list view and one for item preview below),
54123  * and regions on either side that can be used for navigation, application commands, informational displays, etc.
54124  * The setup and configuration work exactly the same as it does for a {@link Roo.BorderLayout} - this class simply
54125  * expedites the setup of the overall layout and regions for this common application style.
54126  * Example:
54127  <pre><code>
54128 var reader = new Roo.ReaderLayout();
54129 var CP = Roo.ContentPanel;  // shortcut for adding
54130
54131 reader.beginUpdate();
54132 reader.add("north", new CP("north", "North"));
54133 reader.add("west", new CP("west", {title: "West"}));
54134 reader.add("east", new CP("east", {title: "East"}));
54135
54136 reader.regions.listView.add(new CP("listView", "List"));
54137 reader.regions.preview.add(new CP("preview", "Preview"));
54138 reader.endUpdate();
54139 </code></pre>
54140 * @constructor
54141 * Create a new ReaderLayout
54142 * @param {Object} config Configuration options
54143 * @param {String/HTMLElement/Element} container (optional) The container this layout is bound to (defaults to
54144 * document.body if omitted)
54145 */
54146 Roo.ReaderLayout = function(config, renderTo){
54147     var c = config || {size:{}};
54148     Roo.ReaderLayout.superclass.constructor.call(this, renderTo || document.body, {
54149         north: c.north !== false ? Roo.apply({
54150             split:false,
54151             initialSize: 32,
54152             titlebar: false
54153         }, c.north) : false,
54154         west: c.west !== false ? Roo.apply({
54155             split:true,
54156             initialSize: 200,
54157             minSize: 175,
54158             maxSize: 400,
54159             titlebar: true,
54160             collapsible: true,
54161             animate: true,
54162             margins:{left:5,right:0,bottom:5,top:5},
54163             cmargins:{left:5,right:5,bottom:5,top:5}
54164         }, c.west) : false,
54165         east: c.east !== false ? Roo.apply({
54166             split:true,
54167             initialSize: 200,
54168             minSize: 175,
54169             maxSize: 400,
54170             titlebar: true,
54171             collapsible: true,
54172             animate: true,
54173             margins:{left:0,right:5,bottom:5,top:5},
54174             cmargins:{left:5,right:5,bottom:5,top:5}
54175         }, c.east) : false,
54176         center: Roo.apply({
54177             tabPosition: 'top',
54178             autoScroll:false,
54179             closeOnTab: true,
54180             titlebar:false,
54181             margins:{left:c.west!==false ? 0 : 5,right:c.east!==false ? 0 : 5,bottom:5,top:2}
54182         }, c.center)
54183     });
54184
54185     this.el.addClass('x-reader');
54186
54187     this.beginUpdate();
54188
54189     var inner = new Roo.BorderLayout(Roo.get(document.body).createChild(), {
54190         south: c.preview !== false ? Roo.apply({
54191             split:true,
54192             initialSize: 200,
54193             minSize: 100,
54194             autoScroll:true,
54195             collapsible:true,
54196             titlebar: true,
54197             cmargins:{top:5,left:0, right:0, bottom:0}
54198         }, c.preview) : false,
54199         center: Roo.apply({
54200             autoScroll:false,
54201             titlebar:false,
54202             minHeight:200
54203         }, c.listView)
54204     });
54205     this.add('center', new Roo.NestedLayoutPanel(inner,
54206             Roo.apply({title: c.mainTitle || '',tabTip:''},c.innerPanelCfg)));
54207
54208     this.endUpdate();
54209
54210     this.regions.preview = inner.getRegion('south');
54211     this.regions.listView = inner.getRegion('center');
54212 };
54213
54214 Roo.extend(Roo.ReaderLayout, Roo.BorderLayout);/*
54215  * Based on:
54216  * Ext JS Library 1.1.1
54217  * Copyright(c) 2006-2007, Ext JS, LLC.
54218  *
54219  * Originally Released Under LGPL - original licence link has changed is not relivant.
54220  *
54221  * Fork - LGPL
54222  * <script type="text/javascript">
54223  */
54224  
54225 /**
54226  * @class Roo.grid.Grid
54227  * @extends Roo.util.Observable
54228  * This class represents the primary interface of a component based grid control.
54229  * <br><br>Usage:<pre><code>
54230  var grid = new Roo.grid.Grid("my-container-id", {
54231      ds: myDataStore,
54232      cm: myColModel,
54233      selModel: mySelectionModel,
54234      autoSizeColumns: true,
54235      monitorWindowResize: false,
54236      trackMouseOver: true
54237  });
54238  // set any options
54239  grid.render();
54240  * </code></pre>
54241  * <b>Common Problems:</b><br/>
54242  * - Grid does not resize properly when going smaller: Setting overflow hidden on the container
54243  * element will correct this<br/>
54244  * - If you get el.style[camel]= NaNpx or -2px or something related, be certain you have given your container element
54245  * dimensions. The grid adapts to your container's size, if your container has no size defined then the results
54246  * are unpredictable.<br/>
54247  * - Do not render the grid into an element with display:none. Try using visibility:hidden. Otherwise there is no way for the
54248  * grid to calculate dimensions/offsets.<br/>
54249   * @constructor
54250  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
54251  * The container MUST have some type of size defined for the grid to fill. The container will be
54252  * automatically set to position relative if it isn't already.
54253  * @param {Object} config A config object that sets properties on this grid.
54254  */
54255 Roo.grid.Grid = function(container, config){
54256         // initialize the container
54257         this.container = Roo.get(container);
54258         this.container.update("");
54259         this.container.setStyle("overflow", "hidden");
54260     this.container.addClass('x-grid-container');
54261
54262     this.id = this.container.id;
54263
54264     Roo.apply(this, config);
54265     // check and correct shorthanded configs
54266     if(this.ds){
54267         this.dataSource = this.ds;
54268         delete this.ds;
54269     }
54270     if(this.cm){
54271         this.colModel = this.cm;
54272         delete this.cm;
54273     }
54274     if(this.sm){
54275         this.selModel = this.sm;
54276         delete this.sm;
54277     }
54278
54279     if (this.selModel) {
54280         this.selModel = Roo.factory(this.selModel, Roo.grid);
54281         this.sm = this.selModel;
54282         this.sm.xmodule = this.xmodule || false;
54283     }
54284     if (typeof(this.colModel.config) == 'undefined') {
54285         this.colModel = new Roo.grid.ColumnModel(this.colModel);
54286         this.cm = this.colModel;
54287         this.cm.xmodule = this.xmodule || false;
54288     }
54289     if (this.dataSource) {
54290         this.dataSource= Roo.factory(this.dataSource, Roo.data);
54291         this.ds = this.dataSource;
54292         this.ds.xmodule = this.xmodule || false;
54293          
54294     }
54295     
54296     
54297     
54298     if(this.width){
54299         this.container.setWidth(this.width);
54300     }
54301
54302     if(this.height){
54303         this.container.setHeight(this.height);
54304     }
54305     /** @private */
54306         this.addEvents({
54307         // raw events
54308         /**
54309          * @event click
54310          * The raw click event for the entire grid.
54311          * @param {Roo.EventObject} e
54312          */
54313         "click" : true,
54314         /**
54315          * @event dblclick
54316          * The raw dblclick event for the entire grid.
54317          * @param {Roo.EventObject} e
54318          */
54319         "dblclick" : true,
54320         /**
54321          * @event contextmenu
54322          * The raw contextmenu event for the entire grid.
54323          * @param {Roo.EventObject} e
54324          */
54325         "contextmenu" : true,
54326         /**
54327          * @event mousedown
54328          * The raw mousedown event for the entire grid.
54329          * @param {Roo.EventObject} e
54330          */
54331         "mousedown" : true,
54332         /**
54333          * @event mouseup
54334          * The raw mouseup event for the entire grid.
54335          * @param {Roo.EventObject} e
54336          */
54337         "mouseup" : true,
54338         /**
54339          * @event mouseover
54340          * The raw mouseover event for the entire grid.
54341          * @param {Roo.EventObject} e
54342          */
54343         "mouseover" : true,
54344         /**
54345          * @event mouseout
54346          * The raw mouseout event for the entire grid.
54347          * @param {Roo.EventObject} e
54348          */
54349         "mouseout" : true,
54350         /**
54351          * @event keypress
54352          * The raw keypress event for the entire grid.
54353          * @param {Roo.EventObject} e
54354          */
54355         "keypress" : true,
54356         /**
54357          * @event keydown
54358          * The raw keydown event for the entire grid.
54359          * @param {Roo.EventObject} e
54360          */
54361         "keydown" : true,
54362
54363         // custom events
54364
54365         /**
54366          * @event cellclick
54367          * Fires when a cell is clicked
54368          * @param {Grid} this
54369          * @param {Number} rowIndex
54370          * @param {Number} columnIndex
54371          * @param {Roo.EventObject} e
54372          */
54373         "cellclick" : true,
54374         /**
54375          * @event celldblclick
54376          * Fires when a cell is double clicked
54377          * @param {Grid} this
54378          * @param {Number} rowIndex
54379          * @param {Number} columnIndex
54380          * @param {Roo.EventObject} e
54381          */
54382         "celldblclick" : true,
54383         /**
54384          * @event rowclick
54385          * Fires when a row is clicked
54386          * @param {Grid} this
54387          * @param {Number} rowIndex
54388          * @param {Roo.EventObject} e
54389          */
54390         "rowclick" : true,
54391         /**
54392          * @event rowdblclick
54393          * Fires when a row is double clicked
54394          * @param {Grid} this
54395          * @param {Number} rowIndex
54396          * @param {Roo.EventObject} e
54397          */
54398         "rowdblclick" : true,
54399         /**
54400          * @event headerclick
54401          * Fires when a header is clicked
54402          * @param {Grid} this
54403          * @param {Number} columnIndex
54404          * @param {Roo.EventObject} e
54405          */
54406         "headerclick" : true,
54407         /**
54408          * @event headerdblclick
54409          * Fires when a header cell is double clicked
54410          * @param {Grid} this
54411          * @param {Number} columnIndex
54412          * @param {Roo.EventObject} e
54413          */
54414         "headerdblclick" : true,
54415         /**
54416          * @event rowcontextmenu
54417          * Fires when a row is right clicked
54418          * @param {Grid} this
54419          * @param {Number} rowIndex
54420          * @param {Roo.EventObject} e
54421          */
54422         "rowcontextmenu" : true,
54423         /**
54424          * @event cellcontextmenu
54425          * Fires when a cell is right clicked
54426          * @param {Grid} this
54427          * @param {Number} rowIndex
54428          * @param {Number} cellIndex
54429          * @param {Roo.EventObject} e
54430          */
54431          "cellcontextmenu" : true,
54432         /**
54433          * @event headercontextmenu
54434          * Fires when a header is right clicked
54435          * @param {Grid} this
54436          * @param {Number} columnIndex
54437          * @param {Roo.EventObject} e
54438          */
54439         "headercontextmenu" : true,
54440         /**
54441          * @event bodyscroll
54442          * Fires when the body element is scrolled
54443          * @param {Number} scrollLeft
54444          * @param {Number} scrollTop
54445          */
54446         "bodyscroll" : true,
54447         /**
54448          * @event columnresize
54449          * Fires when the user resizes a column
54450          * @param {Number} columnIndex
54451          * @param {Number} newSize
54452          */
54453         "columnresize" : true,
54454         /**
54455          * @event columnmove
54456          * Fires when the user moves a column
54457          * @param {Number} oldIndex
54458          * @param {Number} newIndex
54459          */
54460         "columnmove" : true,
54461         /**
54462          * @event startdrag
54463          * Fires when row(s) start being dragged
54464          * @param {Grid} this
54465          * @param {Roo.GridDD} dd The drag drop object
54466          * @param {event} e The raw browser event
54467          */
54468         "startdrag" : true,
54469         /**
54470          * @event enddrag
54471          * Fires when a drag operation is complete
54472          * @param {Grid} this
54473          * @param {Roo.GridDD} dd The drag drop object
54474          * @param {event} e The raw browser event
54475          */
54476         "enddrag" : true,
54477         /**
54478          * @event dragdrop
54479          * Fires when dragged row(s) are dropped on a valid DD target
54480          * @param {Grid} this
54481          * @param {Roo.GridDD} dd The drag drop object
54482          * @param {String} targetId The target drag drop object
54483          * @param {event} e The raw browser event
54484          */
54485         "dragdrop" : true,
54486         /**
54487          * @event dragover
54488          * Fires while row(s) are being dragged. "targetId" is the id of the Yahoo.util.DD object the selected rows are being dragged over.
54489          * @param {Grid} this
54490          * @param {Roo.GridDD} dd The drag drop object
54491          * @param {String} targetId The target drag drop object
54492          * @param {event} e The raw browser event
54493          */
54494         "dragover" : true,
54495         /**
54496          * @event dragenter
54497          *  Fires when the dragged row(s) first cross another DD target while being dragged
54498          * @param {Grid} this
54499          * @param {Roo.GridDD} dd The drag drop object
54500          * @param {String} targetId The target drag drop object
54501          * @param {event} e The raw browser event
54502          */
54503         "dragenter" : true,
54504         /**
54505          * @event dragout
54506          * Fires when the dragged row(s) leave another DD target while being dragged
54507          * @param {Grid} this
54508          * @param {Roo.GridDD} dd The drag drop object
54509          * @param {String} targetId The target drag drop object
54510          * @param {event} e The raw browser event
54511          */
54512         "dragout" : true,
54513         /**
54514          * @event rowclass
54515          * Fires when a row is rendered, so you can change add a style to it.
54516          * @param {GridView} gridview   The grid view
54517          * @param {Object} rowcfg   contains record  rowIndex and rowClass - set rowClass to add a style.
54518          */
54519         'rowclass' : true,
54520
54521         /**
54522          * @event render
54523          * Fires when the grid is rendered
54524          * @param {Grid} grid
54525          */
54526         'render' : true
54527     });
54528
54529     Roo.grid.Grid.superclass.constructor.call(this);
54530 };
54531 Roo.extend(Roo.grid.Grid, Roo.util.Observable, {
54532     
54533     /**
54534      * @cfg {String} ddGroup - drag drop group.
54535      */
54536
54537     /**
54538      * @cfg {Number} minColumnWidth The minimum width a column can be resized to. Default is 25.
54539      */
54540     minColumnWidth : 25,
54541
54542     /**
54543      * @cfg {Boolean} autoSizeColumns True to automatically resize the columns to fit their content
54544      * <b>on initial render.</b> It is more efficient to explicitly size the columns
54545      * through the ColumnModel's {@link Roo.grid.ColumnModel#width} config option.  Default is false.
54546      */
54547     autoSizeColumns : false,
54548
54549     /**
54550      * @cfg {Boolean} autoSizeHeaders True to measure headers with column data when auto sizing columns. Default is true.
54551      */
54552     autoSizeHeaders : true,
54553
54554     /**
54555      * @cfg {Boolean} monitorWindowResize True to autoSize the grid when the window resizes. Default is true.
54556      */
54557     monitorWindowResize : true,
54558
54559     /**
54560      * @cfg {Boolean} maxRowsToMeasure If autoSizeColumns is on, maxRowsToMeasure can be used to limit the number of
54561      * rows measured to get a columns size. Default is 0 (all rows).
54562      */
54563     maxRowsToMeasure : 0,
54564
54565     /**
54566      * @cfg {Boolean} trackMouseOver True to highlight rows when the mouse is over. Default is true.
54567      */
54568     trackMouseOver : true,
54569
54570     /**
54571     * @cfg {Boolean} enableDrag  True to enable drag of rows. Default is false. (double check if this is needed?)
54572     */
54573     
54574     /**
54575     * @cfg {Boolean} enableDragDrop True to enable drag and drop of rows. Default is false.
54576     */
54577     enableDragDrop : false,
54578     
54579     /**
54580     * @cfg {Boolean} enableColumnMove True to enable drag and drop reorder of columns. Default is true.
54581     */
54582     enableColumnMove : true,
54583     
54584     /**
54585     * @cfg {Boolean} enableColumnHide True to enable hiding of columns with the header context menu. Default is true.
54586     */
54587     enableColumnHide : true,
54588     
54589     /**
54590     * @cfg {Boolean} enableRowHeightSync True to manually sync row heights across locked and not locked rows. Default is false.
54591     */
54592     enableRowHeightSync : false,
54593     
54594     /**
54595     * @cfg {Boolean} stripeRows True to stripe the rows.  Default is true.
54596     */
54597     stripeRows : true,
54598     
54599     /**
54600     * @cfg {Boolean} autoHeight True to fit the height of the grid container to the height of the data. Default is false.
54601     */
54602     autoHeight : false,
54603
54604     /**
54605      * @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.
54606      */
54607     autoExpandColumn : false,
54608
54609     /**
54610     * @cfg {Number} autoExpandMin The minimum width the autoExpandColumn can have (if enabled).
54611     * Default is 50.
54612     */
54613     autoExpandMin : 50,
54614
54615     /**
54616     * @cfg {Number} autoExpandMax The maximum width the autoExpandColumn can have (if enabled). Default is 1000.
54617     */
54618     autoExpandMax : 1000,
54619
54620     /**
54621     * @cfg {Object} view The {@link Roo.grid.GridView} used by the grid. This can be set before a call to render().
54622     */
54623     view : null,
54624
54625     /**
54626     * @cfg {Object} loadMask An {@link Roo.LoadMask} config or true to mask the grid while loading. Default is false.
54627     */
54628     loadMask : false,
54629     /**
54630     * @cfg {Roo.dd.DropTarget} dropTarget An {@link Roo.dd.DropTarget} config
54631     */
54632     dropTarget: false,
54633     
54634    
54635     
54636     // private
54637     rendered : false,
54638
54639     /**
54640     * @cfg {Boolean} autoWidth True to set the grid's width to the default total width of the grid's columns instead
54641     * of a fixed width. Default is false.
54642     */
54643     /**
54644     * @cfg {Number} maxHeight Sets the maximum height of the grid - ignored if autoHeight is not on.
54645     */
54646     /**
54647      * Called once after all setup has been completed and the grid is ready to be rendered.
54648      * @return {Roo.grid.Grid} this
54649      */
54650     render : function()
54651     {
54652         var c = this.container;
54653         // try to detect autoHeight/width mode
54654         if((!c.dom.offsetHeight || c.dom.offsetHeight < 20) || c.getStyle("height") == "auto"){
54655             this.autoHeight = true;
54656         }
54657         var view = this.getView();
54658         view.init(this);
54659
54660         c.on("click", this.onClick, this);
54661         c.on("dblclick", this.onDblClick, this);
54662         c.on("contextmenu", this.onContextMenu, this);
54663         c.on("keydown", this.onKeyDown, this);
54664         if (Roo.isTouch) {
54665             c.on("touchstart", this.onTouchStart, this);
54666         }
54667
54668         this.relayEvents(c, ["mousedown","mouseup","mouseover","mouseout","keypress"]);
54669
54670         this.getSelectionModel().init(this);
54671
54672         view.render();
54673
54674         if(this.loadMask){
54675             this.loadMask = new Roo.LoadMask(this.container,
54676                     Roo.apply({store:this.dataSource}, this.loadMask));
54677         }
54678         
54679         
54680         if (this.toolbar && this.toolbar.xtype) {
54681             this.toolbar.container = this.getView().getHeaderPanel(true);
54682             this.toolbar = new Roo.Toolbar(this.toolbar);
54683         }
54684         if (this.footer && this.footer.xtype) {
54685             this.footer.dataSource = this.getDataSource();
54686             this.footer.container = this.getView().getFooterPanel(true);
54687             this.footer = Roo.factory(this.footer, Roo);
54688         }
54689         if (this.dropTarget && this.dropTarget.xtype) {
54690             delete this.dropTarget.xtype;
54691             this.dropTarget =  new Roo.dd.DropTarget(this.getView().mainBody, this.dropTarget);
54692         }
54693         
54694         
54695         this.rendered = true;
54696         this.fireEvent('render', this);
54697         return this;
54698     },
54699
54700         /**
54701          * Reconfigures the grid to use a different Store and Column Model.
54702          * The View will be bound to the new objects and refreshed.
54703          * @param {Roo.data.Store} dataSource The new {@link Roo.data.Store} object
54704          * @param {Roo.grid.ColumnModel} The new {@link Roo.grid.ColumnModel} object
54705          */
54706     reconfigure : function(dataSource, colModel){
54707         if(this.loadMask){
54708             this.loadMask.destroy();
54709             this.loadMask = new Roo.LoadMask(this.container,
54710                     Roo.apply({store:dataSource}, this.loadMask));
54711         }
54712         this.view.bind(dataSource, colModel);
54713         this.dataSource = dataSource;
54714         this.colModel = colModel;
54715         this.view.refresh(true);
54716     },
54717
54718     // private
54719     onKeyDown : function(e){
54720         this.fireEvent("keydown", e);
54721     },
54722
54723     /**
54724      * Destroy this grid.
54725      * @param {Boolean} removeEl True to remove the element
54726      */
54727     destroy : function(removeEl, keepListeners){
54728         if(this.loadMask){
54729             this.loadMask.destroy();
54730         }
54731         var c = this.container;
54732         c.removeAllListeners();
54733         this.view.destroy();
54734         this.colModel.purgeListeners();
54735         if(!keepListeners){
54736             this.purgeListeners();
54737         }
54738         c.update("");
54739         if(removeEl === true){
54740             c.remove();
54741         }
54742     },
54743
54744     // private
54745     processEvent : function(name, e){
54746         // does this fire select???
54747         //Roo.log('grid:processEvent '  + name);
54748         
54749         if (name != 'touchstart' ) {
54750             this.fireEvent(name, e);    
54751         }
54752         
54753         var t = e.getTarget();
54754         var v = this.view;
54755         var header = v.findHeaderIndex(t);
54756         if(header !== false){
54757             var ename = name == 'touchstart' ? 'click' : name;
54758              
54759             this.fireEvent("header" + ename, this, header, e);
54760         }else{
54761             var row = v.findRowIndex(t);
54762             var cell = v.findCellIndex(t);
54763             if (name == 'touchstart') {
54764                 // first touch is always a click.
54765                 // hopefull this happens after selection is updated.?
54766                 name = false;
54767                 
54768                 if (typeof(this.selModel.getSelectedCell) != 'undefined') {
54769                     var cs = this.selModel.getSelectedCell();
54770                     if (row == cs[0] && cell == cs[1]){
54771                         name = 'dblclick';
54772                     }
54773                 }
54774                 if (typeof(this.selModel.getSelections) != 'undefined') {
54775                     var cs = this.selModel.getSelections();
54776                     var ds = this.dataSource;
54777                     if (cs.length == 1 && ds.getAt(row) == cs[0]){
54778                         name = 'dblclick';
54779                     }
54780                 }
54781                 if (!name) {
54782                     return;
54783                 }
54784             }
54785             
54786             
54787             if(row !== false){
54788                 this.fireEvent("row" + name, this, row, e);
54789                 if(cell !== false){
54790                     this.fireEvent("cell" + name, this, row, cell, e);
54791                 }
54792             }
54793         }
54794     },
54795
54796     // private
54797     onClick : function(e){
54798         this.processEvent("click", e);
54799     },
54800    // private
54801     onTouchStart : function(e){
54802         this.processEvent("touchstart", e);
54803     },
54804
54805     // private
54806     onContextMenu : function(e, t){
54807         this.processEvent("contextmenu", e);
54808     },
54809
54810     // private
54811     onDblClick : function(e){
54812         this.processEvent("dblclick", e);
54813     },
54814
54815     // private
54816     walkCells : function(row, col, step, fn, scope){
54817         var cm = this.colModel, clen = cm.getColumnCount();
54818         var ds = this.dataSource, rlen = ds.getCount(), first = true;
54819         if(step < 0){
54820             if(col < 0){
54821                 row--;
54822                 first = false;
54823             }
54824             while(row >= 0){
54825                 if(!first){
54826                     col = clen-1;
54827                 }
54828                 first = false;
54829                 while(col >= 0){
54830                     if(fn.call(scope || this, row, col, cm) === true){
54831                         return [row, col];
54832                     }
54833                     col--;
54834                 }
54835                 row--;
54836             }
54837         } else {
54838             if(col >= clen){
54839                 row++;
54840                 first = false;
54841             }
54842             while(row < rlen){
54843                 if(!first){
54844                     col = 0;
54845                 }
54846                 first = false;
54847                 while(col < clen){
54848                     if(fn.call(scope || this, row, col, cm) === true){
54849                         return [row, col];
54850                     }
54851                     col++;
54852                 }
54853                 row++;
54854             }
54855         }
54856         return null;
54857     },
54858
54859     // private
54860     getSelections : function(){
54861         return this.selModel.getSelections();
54862     },
54863
54864     /**
54865      * Causes the grid to manually recalculate its dimensions. Generally this is done automatically,
54866      * but if manual update is required this method will initiate it.
54867      */
54868     autoSize : function(){
54869         if(this.rendered){
54870             this.view.layout();
54871             if(this.view.adjustForScroll){
54872                 this.view.adjustForScroll();
54873             }
54874         }
54875     },
54876
54877     /**
54878      * Returns the grid's underlying element.
54879      * @return {Element} The element
54880      */
54881     getGridEl : function(){
54882         return this.container;
54883     },
54884
54885     // private for compatibility, overridden by editor grid
54886     stopEditing : function(){},
54887
54888     /**
54889      * Returns the grid's SelectionModel.
54890      * @return {SelectionModel}
54891      */
54892     getSelectionModel : function(){
54893         if(!this.selModel){
54894             this.selModel = new Roo.grid.RowSelectionModel();
54895         }
54896         return this.selModel;
54897     },
54898
54899     /**
54900      * Returns the grid's DataSource.
54901      * @return {DataSource}
54902      */
54903     getDataSource : function(){
54904         return this.dataSource;
54905     },
54906
54907     /**
54908      * Returns the grid's ColumnModel.
54909      * @return {ColumnModel}
54910      */
54911     getColumnModel : function(){
54912         return this.colModel;
54913     },
54914
54915     /**
54916      * Returns the grid's GridView object.
54917      * @return {GridView}
54918      */
54919     getView : function(){
54920         if(!this.view){
54921             this.view = new Roo.grid.GridView(this.viewConfig);
54922         }
54923         return this.view;
54924     },
54925     /**
54926      * Called to get grid's drag proxy text, by default returns this.ddText.
54927      * @return {String}
54928      */
54929     getDragDropText : function(){
54930         var count = this.selModel.getCount();
54931         return String.format(this.ddText, count, count == 1 ? '' : 's');
54932     }
54933 });
54934 /**
54935  * Configures the text is the drag proxy (defaults to "%0 selected row(s)").
54936  * %0 is replaced with the number of selected rows.
54937  * @type String
54938  */
54939 Roo.grid.Grid.prototype.ddText = "{0} selected row{1}";/*
54940  * Based on:
54941  * Ext JS Library 1.1.1
54942  * Copyright(c) 2006-2007, Ext JS, LLC.
54943  *
54944  * Originally Released Under LGPL - original licence link has changed is not relivant.
54945  *
54946  * Fork - LGPL
54947  * <script type="text/javascript">
54948  */
54949  
54950 Roo.grid.AbstractGridView = function(){
54951         this.grid = null;
54952         
54953         this.events = {
54954             "beforerowremoved" : true,
54955             "beforerowsinserted" : true,
54956             "beforerefresh" : true,
54957             "rowremoved" : true,
54958             "rowsinserted" : true,
54959             "rowupdated" : true,
54960             "refresh" : true
54961         };
54962     Roo.grid.AbstractGridView.superclass.constructor.call(this);
54963 };
54964
54965 Roo.extend(Roo.grid.AbstractGridView, Roo.util.Observable, {
54966     rowClass : "x-grid-row",
54967     cellClass : "x-grid-cell",
54968     tdClass : "x-grid-td",
54969     hdClass : "x-grid-hd",
54970     splitClass : "x-grid-hd-split",
54971     
54972     init: function(grid){
54973         this.grid = grid;
54974                 var cid = this.grid.getGridEl().id;
54975         this.colSelector = "#" + cid + " ." + this.cellClass + "-";
54976         this.tdSelector = "#" + cid + " ." + this.tdClass + "-";
54977         this.hdSelector = "#" + cid + " ." + this.hdClass + "-";
54978         this.splitSelector = "#" + cid + " ." + this.splitClass + "-";
54979         },
54980         
54981     getColumnRenderers : function(){
54982         var renderers = [];
54983         var cm = this.grid.colModel;
54984         var colCount = cm.getColumnCount();
54985         for(var i = 0; i < colCount; i++){
54986             renderers[i] = cm.getRenderer(i);
54987         }
54988         return renderers;
54989     },
54990     
54991     getColumnIds : function(){
54992         var ids = [];
54993         var cm = this.grid.colModel;
54994         var colCount = cm.getColumnCount();
54995         for(var i = 0; i < colCount; i++){
54996             ids[i] = cm.getColumnId(i);
54997         }
54998         return ids;
54999     },
55000     
55001     getDataIndexes : function(){
55002         if(!this.indexMap){
55003             this.indexMap = this.buildIndexMap();
55004         }
55005         return this.indexMap.colToData;
55006     },
55007     
55008     getColumnIndexByDataIndex : function(dataIndex){
55009         if(!this.indexMap){
55010             this.indexMap = this.buildIndexMap();
55011         }
55012         return this.indexMap.dataToCol[dataIndex];
55013     },
55014     
55015     /**
55016      * Set a css style for a column dynamically. 
55017      * @param {Number} colIndex The index of the column
55018      * @param {String} name The css property name
55019      * @param {String} value The css value
55020      */
55021     setCSSStyle : function(colIndex, name, value){
55022         var selector = "#" + this.grid.id + " .x-grid-col-" + colIndex;
55023         Roo.util.CSS.updateRule(selector, name, value);
55024     },
55025     
55026     generateRules : function(cm){
55027         var ruleBuf = [], rulesId = this.grid.id + '-cssrules';
55028         Roo.util.CSS.removeStyleSheet(rulesId);
55029         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
55030             var cid = cm.getColumnId(i);
55031             ruleBuf.push(this.colSelector, cid, " {\n", cm.config[i].css, "}\n",
55032                          this.tdSelector, cid, " {\n}\n",
55033                          this.hdSelector, cid, " {\n}\n",
55034                          this.splitSelector, cid, " {\n}\n");
55035         }
55036         return Roo.util.CSS.createStyleSheet(ruleBuf.join(""), rulesId);
55037     }
55038 });/*
55039  * Based on:
55040  * Ext JS Library 1.1.1
55041  * Copyright(c) 2006-2007, Ext JS, LLC.
55042  *
55043  * Originally Released Under LGPL - original licence link has changed is not relivant.
55044  *
55045  * Fork - LGPL
55046  * <script type="text/javascript">
55047  */
55048
55049 // private
55050 // This is a support class used internally by the Grid components
55051 Roo.grid.HeaderDragZone = function(grid, hd, hd2){
55052     this.grid = grid;
55053     this.view = grid.getView();
55054     this.ddGroup = "gridHeader" + this.grid.getGridEl().id;
55055     Roo.grid.HeaderDragZone.superclass.constructor.call(this, hd);
55056     if(hd2){
55057         this.setHandleElId(Roo.id(hd));
55058         this.setOuterHandleElId(Roo.id(hd2));
55059     }
55060     this.scroll = false;
55061 };
55062 Roo.extend(Roo.grid.HeaderDragZone, Roo.dd.DragZone, {
55063     maxDragWidth: 120,
55064     getDragData : function(e){
55065         var t = Roo.lib.Event.getTarget(e);
55066         var h = this.view.findHeaderCell(t);
55067         if(h){
55068             return {ddel: h.firstChild, header:h};
55069         }
55070         return false;
55071     },
55072
55073     onInitDrag : function(e){
55074         this.view.headersDisabled = true;
55075         var clone = this.dragData.ddel.cloneNode(true);
55076         clone.id = Roo.id();
55077         clone.style.width = Math.min(this.dragData.header.offsetWidth,this.maxDragWidth) + "px";
55078         this.proxy.update(clone);
55079         return true;
55080     },
55081
55082     afterValidDrop : function(){
55083         var v = this.view;
55084         setTimeout(function(){
55085             v.headersDisabled = false;
55086         }, 50);
55087     },
55088
55089     afterInvalidDrop : function(){
55090         var v = this.view;
55091         setTimeout(function(){
55092             v.headersDisabled = false;
55093         }, 50);
55094     }
55095 });
55096 /*
55097  * Based on:
55098  * Ext JS Library 1.1.1
55099  * Copyright(c) 2006-2007, Ext JS, LLC.
55100  *
55101  * Originally Released Under LGPL - original licence link has changed is not relivant.
55102  *
55103  * Fork - LGPL
55104  * <script type="text/javascript">
55105  */
55106 // private
55107 // This is a support class used internally by the Grid components
55108 Roo.grid.HeaderDropZone = function(grid, hd, hd2){
55109     this.grid = grid;
55110     this.view = grid.getView();
55111     // split the proxies so they don't interfere with mouse events
55112     this.proxyTop = Roo.DomHelper.append(document.body, {
55113         cls:"col-move-top", html:"&#160;"
55114     }, true);
55115     this.proxyBottom = Roo.DomHelper.append(document.body, {
55116         cls:"col-move-bottom", html:"&#160;"
55117     }, true);
55118     this.proxyTop.hide = this.proxyBottom.hide = function(){
55119         this.setLeftTop(-100,-100);
55120         this.setStyle("visibility", "hidden");
55121     };
55122     this.ddGroup = "gridHeader" + this.grid.getGridEl().id;
55123     // temporarily disabled
55124     //Roo.dd.ScrollManager.register(this.view.scroller.dom);
55125     Roo.grid.HeaderDropZone.superclass.constructor.call(this, grid.getGridEl().dom);
55126 };
55127 Roo.extend(Roo.grid.HeaderDropZone, Roo.dd.DropZone, {
55128     proxyOffsets : [-4, -9],
55129     fly: Roo.Element.fly,
55130
55131     getTargetFromEvent : function(e){
55132         var t = Roo.lib.Event.getTarget(e);
55133         var cindex = this.view.findCellIndex(t);
55134         if(cindex !== false){
55135             return this.view.getHeaderCell(cindex);
55136         }
55137         return null;
55138     },
55139
55140     nextVisible : function(h){
55141         var v = this.view, cm = this.grid.colModel;
55142         h = h.nextSibling;
55143         while(h){
55144             if(!cm.isHidden(v.getCellIndex(h))){
55145                 return h;
55146             }
55147             h = h.nextSibling;
55148         }
55149         return null;
55150     },
55151
55152     prevVisible : function(h){
55153         var v = this.view, cm = this.grid.colModel;
55154         h = h.prevSibling;
55155         while(h){
55156             if(!cm.isHidden(v.getCellIndex(h))){
55157                 return h;
55158             }
55159             h = h.prevSibling;
55160         }
55161         return null;
55162     },
55163
55164     positionIndicator : function(h, n, e){
55165         var x = Roo.lib.Event.getPageX(e);
55166         var r = Roo.lib.Dom.getRegion(n.firstChild);
55167         var px, pt, py = r.top + this.proxyOffsets[1];
55168         if((r.right - x) <= (r.right-r.left)/2){
55169             px = r.right+this.view.borderWidth;
55170             pt = "after";
55171         }else{
55172             px = r.left;
55173             pt = "before";
55174         }
55175         var oldIndex = this.view.getCellIndex(h);
55176         var newIndex = this.view.getCellIndex(n);
55177
55178         if(this.grid.colModel.isFixed(newIndex)){
55179             return false;
55180         }
55181
55182         var locked = this.grid.colModel.isLocked(newIndex);
55183
55184         if(pt == "after"){
55185             newIndex++;
55186         }
55187         if(oldIndex < newIndex){
55188             newIndex--;
55189         }
55190         if(oldIndex == newIndex && (locked == this.grid.colModel.isLocked(oldIndex))){
55191             return false;
55192         }
55193         px +=  this.proxyOffsets[0];
55194         this.proxyTop.setLeftTop(px, py);
55195         this.proxyTop.show();
55196         if(!this.bottomOffset){
55197             this.bottomOffset = this.view.mainHd.getHeight();
55198         }
55199         this.proxyBottom.setLeftTop(px, py+this.proxyTop.dom.offsetHeight+this.bottomOffset);
55200         this.proxyBottom.show();
55201         return pt;
55202     },
55203
55204     onNodeEnter : function(n, dd, e, data){
55205         if(data.header != n){
55206             this.positionIndicator(data.header, n, e);
55207         }
55208     },
55209
55210     onNodeOver : function(n, dd, e, data){
55211         var result = false;
55212         if(data.header != n){
55213             result = this.positionIndicator(data.header, n, e);
55214         }
55215         if(!result){
55216             this.proxyTop.hide();
55217             this.proxyBottom.hide();
55218         }
55219         return result ? this.dropAllowed : this.dropNotAllowed;
55220     },
55221
55222     onNodeOut : function(n, dd, e, data){
55223         this.proxyTop.hide();
55224         this.proxyBottom.hide();
55225     },
55226
55227     onNodeDrop : function(n, dd, e, data){
55228         var h = data.header;
55229         if(h != n){
55230             var cm = this.grid.colModel;
55231             var x = Roo.lib.Event.getPageX(e);
55232             var r = Roo.lib.Dom.getRegion(n.firstChild);
55233             var pt = (r.right - x) <= ((r.right-r.left)/2) ? "after" : "before";
55234             var oldIndex = this.view.getCellIndex(h);
55235             var newIndex = this.view.getCellIndex(n);
55236             var locked = cm.isLocked(newIndex);
55237             if(pt == "after"){
55238                 newIndex++;
55239             }
55240             if(oldIndex < newIndex){
55241                 newIndex--;
55242             }
55243             if(oldIndex == newIndex && (locked == cm.isLocked(oldIndex))){
55244                 return false;
55245             }
55246             cm.setLocked(oldIndex, locked, true);
55247             cm.moveColumn(oldIndex, newIndex);
55248             this.grid.fireEvent("columnmove", oldIndex, newIndex);
55249             return true;
55250         }
55251         return false;
55252     }
55253 });
55254 /*
55255  * Based on:
55256  * Ext JS Library 1.1.1
55257  * Copyright(c) 2006-2007, Ext JS, LLC.
55258  *
55259  * Originally Released Under LGPL - original licence link has changed is not relivant.
55260  *
55261  * Fork - LGPL
55262  * <script type="text/javascript">
55263  */
55264   
55265 /**
55266  * @class Roo.grid.GridView
55267  * @extends Roo.util.Observable
55268  *
55269  * @constructor
55270  * @param {Object} config
55271  */
55272 Roo.grid.GridView = function(config){
55273     Roo.grid.GridView.superclass.constructor.call(this);
55274     this.el = null;
55275
55276     Roo.apply(this, config);
55277 };
55278
55279 Roo.extend(Roo.grid.GridView, Roo.grid.AbstractGridView, {
55280
55281     unselectable :  'unselectable="on"',
55282     unselectableCls :  'x-unselectable',
55283     
55284     
55285     rowClass : "x-grid-row",
55286
55287     cellClass : "x-grid-col",
55288
55289     tdClass : "x-grid-td",
55290
55291     hdClass : "x-grid-hd",
55292
55293     splitClass : "x-grid-split",
55294
55295     sortClasses : ["sort-asc", "sort-desc"],
55296
55297     enableMoveAnim : false,
55298
55299     hlColor: "C3DAF9",
55300
55301     dh : Roo.DomHelper,
55302
55303     fly : Roo.Element.fly,
55304
55305     css : Roo.util.CSS,
55306
55307     borderWidth: 1,
55308
55309     splitOffset: 3,
55310
55311     scrollIncrement : 22,
55312
55313     cellRE: /(?:.*?)x-grid-(?:hd|cell|csplit)-(?:[\d]+)-([\d]+)(?:.*?)/,
55314
55315     findRE: /\s?(?:x-grid-hd|x-grid-col|x-grid-csplit)\s/,
55316
55317     bind : function(ds, cm){
55318         if(this.ds){
55319             this.ds.un("load", this.onLoad, this);
55320             this.ds.un("datachanged", this.onDataChange, this);
55321             this.ds.un("add", this.onAdd, this);
55322             this.ds.un("remove", this.onRemove, this);
55323             this.ds.un("update", this.onUpdate, this);
55324             this.ds.un("clear", this.onClear, this);
55325         }
55326         if(ds){
55327             ds.on("load", this.onLoad, this);
55328             ds.on("datachanged", this.onDataChange, this);
55329             ds.on("add", this.onAdd, this);
55330             ds.on("remove", this.onRemove, this);
55331             ds.on("update", this.onUpdate, this);
55332             ds.on("clear", this.onClear, this);
55333         }
55334         this.ds = ds;
55335
55336         if(this.cm){
55337             this.cm.un("widthchange", this.onColWidthChange, this);
55338             this.cm.un("headerchange", this.onHeaderChange, this);
55339             this.cm.un("hiddenchange", this.onHiddenChange, this);
55340             this.cm.un("columnmoved", this.onColumnMove, this);
55341             this.cm.un("columnlockchange", this.onColumnLock, this);
55342         }
55343         if(cm){
55344             this.generateRules(cm);
55345             cm.on("widthchange", this.onColWidthChange, this);
55346             cm.on("headerchange", this.onHeaderChange, this);
55347             cm.on("hiddenchange", this.onHiddenChange, this);
55348             cm.on("columnmoved", this.onColumnMove, this);
55349             cm.on("columnlockchange", this.onColumnLock, this);
55350         }
55351         this.cm = cm;
55352     },
55353
55354     init: function(grid){
55355         Roo.grid.GridView.superclass.init.call(this, grid);
55356
55357         this.bind(grid.dataSource, grid.colModel);
55358
55359         grid.on("headerclick", this.handleHeaderClick, this);
55360
55361         if(grid.trackMouseOver){
55362             grid.on("mouseover", this.onRowOver, this);
55363             grid.on("mouseout", this.onRowOut, this);
55364         }
55365         grid.cancelTextSelection = function(){};
55366         this.gridId = grid.id;
55367
55368         var tpls = this.templates || {};
55369
55370         if(!tpls.master){
55371             tpls.master = new Roo.Template(
55372                '<div class="x-grid" hidefocus="true">',
55373                 '<a href="#" class="x-grid-focus" tabIndex="-1"></a>',
55374                   '<div class="x-grid-topbar"></div>',
55375                   '<div class="x-grid-scroller"><div></div></div>',
55376                   '<div class="x-grid-locked">',
55377                       '<div class="x-grid-header">{lockedHeader}</div>',
55378                       '<div class="x-grid-body">{lockedBody}</div>',
55379                   "</div>",
55380                   '<div class="x-grid-viewport">',
55381                       '<div class="x-grid-header">{header}</div>',
55382                       '<div class="x-grid-body">{body}</div>',
55383                   "</div>",
55384                   '<div class="x-grid-bottombar"></div>',
55385                  
55386                   '<div class="x-grid-resize-proxy">&#160;</div>',
55387                "</div>"
55388             );
55389             tpls.master.disableformats = true;
55390         }
55391
55392         if(!tpls.header){
55393             tpls.header = new Roo.Template(
55394                '<table border="0" cellspacing="0" cellpadding="0">',
55395                '<tbody><tr class="x-grid-hd-row">{cells}</tr></tbody>',
55396                "</table>{splits}"
55397             );
55398             tpls.header.disableformats = true;
55399         }
55400         tpls.header.compile();
55401
55402         if(!tpls.hcell){
55403             tpls.hcell = new Roo.Template(
55404                 '<td class="x-grid-hd x-grid-td-{id} {cellId}"><div title="{title}" class="x-grid-hd-inner x-grid-hd-{id}">',
55405                 '<div class="x-grid-hd-text ' + this.unselectableCls +  '" ' + this.unselectable +'>{value}<img class="x-grid-sort-icon" src="', Roo.BLANK_IMAGE_URL, '" /></div>',
55406                 "</div></td>"
55407              );
55408              tpls.hcell.disableFormats = true;
55409         }
55410         tpls.hcell.compile();
55411
55412         if(!tpls.hsplit){
55413             tpls.hsplit = new Roo.Template('<div class="x-grid-split {splitId} x-grid-split-{id}" style="{style} ' +
55414                                             this.unselectableCls +  '" ' + this.unselectable +'>&#160;</div>');
55415             tpls.hsplit.disableFormats = true;
55416         }
55417         tpls.hsplit.compile();
55418
55419         if(!tpls.body){
55420             tpls.body = new Roo.Template(
55421                '<table border="0" cellspacing="0" cellpadding="0">',
55422                "<tbody>{rows}</tbody>",
55423                "</table>"
55424             );
55425             tpls.body.disableFormats = true;
55426         }
55427         tpls.body.compile();
55428
55429         if(!tpls.row){
55430             tpls.row = new Roo.Template('<tr class="x-grid-row {alt}">{cells}</tr>');
55431             tpls.row.disableFormats = true;
55432         }
55433         tpls.row.compile();
55434
55435         if(!tpls.cell){
55436             tpls.cell = new Roo.Template(
55437                 '<td class="x-grid-col x-grid-td-{id} {cellId} {css}" tabIndex="0">',
55438                 '<div class="x-grid-col-{id} x-grid-cell-inner"><div class="x-grid-cell-text ' +
55439                     this.unselectableCls +  '" ' + this.unselectable +'" {attr}>{value}</div></div>',
55440                 "</td>"
55441             );
55442             tpls.cell.disableFormats = true;
55443         }
55444         tpls.cell.compile();
55445
55446         this.templates = tpls;
55447     },
55448
55449     // remap these for backwards compat
55450     onColWidthChange : function(){
55451         this.updateColumns.apply(this, arguments);
55452     },
55453     onHeaderChange : function(){
55454         this.updateHeaders.apply(this, arguments);
55455     }, 
55456     onHiddenChange : function(){
55457         this.handleHiddenChange.apply(this, arguments);
55458     },
55459     onColumnMove : function(){
55460         this.handleColumnMove.apply(this, arguments);
55461     },
55462     onColumnLock : function(){
55463         this.handleLockChange.apply(this, arguments);
55464     },
55465
55466     onDataChange : function(){
55467         this.refresh();
55468         this.updateHeaderSortState();
55469     },
55470
55471     onClear : function(){
55472         this.refresh();
55473     },
55474
55475     onUpdate : function(ds, record){
55476         this.refreshRow(record);
55477     },
55478
55479     refreshRow : function(record){
55480         var ds = this.ds, index;
55481         if(typeof record == 'number'){
55482             index = record;
55483             record = ds.getAt(index);
55484         }else{
55485             index = ds.indexOf(record);
55486         }
55487         this.insertRows(ds, index, index, true);
55488         this.onRemove(ds, record, index+1, true);
55489         this.syncRowHeights(index, index);
55490         this.layout();
55491         this.fireEvent("rowupdated", this, index, record);
55492     },
55493
55494     onAdd : function(ds, records, index){
55495         this.insertRows(ds, index, index + (records.length-1));
55496     },
55497
55498     onRemove : function(ds, record, index, isUpdate){
55499         if(isUpdate !== true){
55500             this.fireEvent("beforerowremoved", this, index, record);
55501         }
55502         var bt = this.getBodyTable(), lt = this.getLockedTable();
55503         if(bt.rows[index]){
55504             bt.firstChild.removeChild(bt.rows[index]);
55505         }
55506         if(lt.rows[index]){
55507             lt.firstChild.removeChild(lt.rows[index]);
55508         }
55509         if(isUpdate !== true){
55510             this.stripeRows(index);
55511             this.syncRowHeights(index, index);
55512             this.layout();
55513             this.fireEvent("rowremoved", this, index, record);
55514         }
55515     },
55516
55517     onLoad : function(){
55518         this.scrollToTop();
55519     },
55520
55521     /**
55522      * Scrolls the grid to the top
55523      */
55524     scrollToTop : function(){
55525         if(this.scroller){
55526             this.scroller.dom.scrollTop = 0;
55527             this.syncScroll();
55528         }
55529     },
55530
55531     /**
55532      * Gets a panel in the header of the grid that can be used for toolbars etc.
55533      * After modifying the contents of this panel a call to grid.autoSize() may be
55534      * required to register any changes in size.
55535      * @param {Boolean} doShow By default the header is hidden. Pass true to show the panel
55536      * @return Roo.Element
55537      */
55538     getHeaderPanel : function(doShow){
55539         if(doShow){
55540             this.headerPanel.show();
55541         }
55542         return this.headerPanel;
55543     },
55544
55545     /**
55546      * Gets a panel in the footer of the grid that can be used for toolbars etc.
55547      * After modifying the contents of this panel a call to grid.autoSize() may be
55548      * required to register any changes in size.
55549      * @param {Boolean} doShow By default the footer is hidden. Pass true to show the panel
55550      * @return Roo.Element
55551      */
55552     getFooterPanel : function(doShow){
55553         if(doShow){
55554             this.footerPanel.show();
55555         }
55556         return this.footerPanel;
55557     },
55558
55559     initElements : function(){
55560         var E = Roo.Element;
55561         var el = this.grid.getGridEl().dom.firstChild;
55562         var cs = el.childNodes;
55563
55564         this.el = new E(el);
55565         
55566          this.focusEl = new E(el.firstChild);
55567         this.focusEl.swallowEvent("click", true);
55568         
55569         this.headerPanel = new E(cs[1]);
55570         this.headerPanel.enableDisplayMode("block");
55571
55572         this.scroller = new E(cs[2]);
55573         this.scrollSizer = new E(this.scroller.dom.firstChild);
55574
55575         this.lockedWrap = new E(cs[3]);
55576         this.lockedHd = new E(this.lockedWrap.dom.firstChild);
55577         this.lockedBody = new E(this.lockedWrap.dom.childNodes[1]);
55578
55579         this.mainWrap = new E(cs[4]);
55580         this.mainHd = new E(this.mainWrap.dom.firstChild);
55581         this.mainBody = new E(this.mainWrap.dom.childNodes[1]);
55582
55583         this.footerPanel = new E(cs[5]);
55584         this.footerPanel.enableDisplayMode("block");
55585
55586         this.resizeProxy = new E(cs[6]);
55587
55588         this.headerSelector = String.format(
55589            '#{0} td.x-grid-hd, #{1} td.x-grid-hd',
55590            this.lockedHd.id, this.mainHd.id
55591         );
55592
55593         this.splitterSelector = String.format(
55594            '#{0} div.x-grid-split, #{1} div.x-grid-split',
55595            this.idToCssName(this.lockedHd.id), this.idToCssName(this.mainHd.id)
55596         );
55597     },
55598     idToCssName : function(s)
55599     {
55600         return s.replace(/[^a-z0-9]+/ig, '-');
55601     },
55602
55603     getHeaderCell : function(index){
55604         return Roo.DomQuery.select(this.headerSelector)[index];
55605     },
55606
55607     getHeaderCellMeasure : function(index){
55608         return this.getHeaderCell(index).firstChild;
55609     },
55610
55611     getHeaderCellText : function(index){
55612         return this.getHeaderCell(index).firstChild.firstChild;
55613     },
55614
55615     getLockedTable : function(){
55616         return this.lockedBody.dom.firstChild;
55617     },
55618
55619     getBodyTable : function(){
55620         return this.mainBody.dom.firstChild;
55621     },
55622
55623     getLockedRow : function(index){
55624         return this.getLockedTable().rows[index];
55625     },
55626
55627     getRow : function(index){
55628         return this.getBodyTable().rows[index];
55629     },
55630
55631     getRowComposite : function(index){
55632         if(!this.rowEl){
55633             this.rowEl = new Roo.CompositeElementLite();
55634         }
55635         var els = [], lrow, mrow;
55636         if(lrow = this.getLockedRow(index)){
55637             els.push(lrow);
55638         }
55639         if(mrow = this.getRow(index)){
55640             els.push(mrow);
55641         }
55642         this.rowEl.elements = els;
55643         return this.rowEl;
55644     },
55645     /**
55646      * Gets the 'td' of the cell
55647      * 
55648      * @param {Integer} rowIndex row to select
55649      * @param {Integer} colIndex column to select
55650      * 
55651      * @return {Object} 
55652      */
55653     getCell : function(rowIndex, colIndex){
55654         var locked = this.cm.getLockedCount();
55655         var source;
55656         if(colIndex < locked){
55657             source = this.lockedBody.dom.firstChild;
55658         }else{
55659             source = this.mainBody.dom.firstChild;
55660             colIndex -= locked;
55661         }
55662         return source.rows[rowIndex].childNodes[colIndex];
55663     },
55664
55665     getCellText : function(rowIndex, colIndex){
55666         return this.getCell(rowIndex, colIndex).firstChild.firstChild;
55667     },
55668
55669     getCellBox : function(cell){
55670         var b = this.fly(cell).getBox();
55671         if(Roo.isOpera){ // opera fails to report the Y
55672             b.y = cell.offsetTop + this.mainBody.getY();
55673         }
55674         return b;
55675     },
55676
55677     getCellIndex : function(cell){
55678         var id = String(cell.className).match(this.cellRE);
55679         if(id){
55680             return parseInt(id[1], 10);
55681         }
55682         return 0;
55683     },
55684
55685     findHeaderIndex : function(n){
55686         var r = Roo.fly(n).findParent("td." + this.hdClass, 6);
55687         return r ? this.getCellIndex(r) : false;
55688     },
55689
55690     findHeaderCell : function(n){
55691         var r = Roo.fly(n).findParent("td." + this.hdClass, 6);
55692         return r ? r : false;
55693     },
55694
55695     findRowIndex : function(n){
55696         if(!n){
55697             return false;
55698         }
55699         var r = Roo.fly(n).findParent("tr." + this.rowClass, 6);
55700         return r ? r.rowIndex : false;
55701     },
55702
55703     findCellIndex : function(node){
55704         var stop = this.el.dom;
55705         while(node && node != stop){
55706             if(this.findRE.test(node.className)){
55707                 return this.getCellIndex(node);
55708             }
55709             node = node.parentNode;
55710         }
55711         return false;
55712     },
55713
55714     getColumnId : function(index){
55715         return this.cm.getColumnId(index);
55716     },
55717
55718     getSplitters : function()
55719     {
55720         if(this.splitterSelector){
55721            return Roo.DomQuery.select(this.splitterSelector);
55722         }else{
55723             return null;
55724       }
55725     },
55726
55727     getSplitter : function(index){
55728         return this.getSplitters()[index];
55729     },
55730
55731     onRowOver : function(e, t){
55732         var row;
55733         if((row = this.findRowIndex(t)) !== false){
55734             this.getRowComposite(row).addClass("x-grid-row-over");
55735         }
55736     },
55737
55738     onRowOut : function(e, t){
55739         var row;
55740         if((row = this.findRowIndex(t)) !== false && row !== this.findRowIndex(e.getRelatedTarget())){
55741             this.getRowComposite(row).removeClass("x-grid-row-over");
55742         }
55743     },
55744
55745     renderHeaders : function(){
55746         var cm = this.cm;
55747         var ct = this.templates.hcell, ht = this.templates.header, st = this.templates.hsplit;
55748         var cb = [], lb = [], sb = [], lsb = [], p = {};
55749         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
55750             p.cellId = "x-grid-hd-0-" + i;
55751             p.splitId = "x-grid-csplit-0-" + i;
55752             p.id = cm.getColumnId(i);
55753             p.value = cm.getColumnHeader(i) || "";
55754             p.title = cm.getColumnTooltip(i) || (''+p.value).match(/\</)  ? '' :  p.value  || "";
55755             p.style = (this.grid.enableColumnResize === false || !cm.isResizable(i) || cm.isFixed(i)) ? 'cursor:default' : '';
55756             if(!cm.isLocked(i)){
55757                 cb[cb.length] = ct.apply(p);
55758                 sb[sb.length] = st.apply(p);
55759             }else{
55760                 lb[lb.length] = ct.apply(p);
55761                 lsb[lsb.length] = st.apply(p);
55762             }
55763         }
55764         return [ht.apply({cells: lb.join(""), splits:lsb.join("")}),
55765                 ht.apply({cells: cb.join(""), splits:sb.join("")})];
55766     },
55767
55768     updateHeaders : function(){
55769         var html = this.renderHeaders();
55770         this.lockedHd.update(html[0]);
55771         this.mainHd.update(html[1]);
55772     },
55773
55774     /**
55775      * Focuses the specified row.
55776      * @param {Number} row The row index
55777      */
55778     focusRow : function(row)
55779     {
55780         //Roo.log('GridView.focusRow');
55781         var x = this.scroller.dom.scrollLeft;
55782         this.focusCell(row, 0, false);
55783         this.scroller.dom.scrollLeft = x;
55784     },
55785
55786     /**
55787      * Focuses the specified cell.
55788      * @param {Number} row The row index
55789      * @param {Number} col The column index
55790      * @param {Boolean} hscroll false to disable horizontal scrolling
55791      */
55792     focusCell : function(row, col, hscroll)
55793     {
55794         //Roo.log('GridView.focusCell');
55795         var el = this.ensureVisible(row, col, hscroll);
55796         this.focusEl.alignTo(el, "tl-tl");
55797         if(Roo.isGecko){
55798             this.focusEl.focus();
55799         }else{
55800             this.focusEl.focus.defer(1, this.focusEl);
55801         }
55802     },
55803
55804     /**
55805      * Scrolls the specified cell into view
55806      * @param {Number} row The row index
55807      * @param {Number} col The column index
55808      * @param {Boolean} hscroll false to disable horizontal scrolling
55809      */
55810     ensureVisible : function(row, col, hscroll)
55811     {
55812         //Roo.log('GridView.ensureVisible,' + row + ',' + col);
55813         //return null; //disable for testing.
55814         if(typeof row != "number"){
55815             row = row.rowIndex;
55816         }
55817         if(row < 0 && row >= this.ds.getCount()){
55818             return  null;
55819         }
55820         col = (col !== undefined ? col : 0);
55821         var cm = this.grid.colModel;
55822         while(cm.isHidden(col)){
55823             col++;
55824         }
55825
55826         var el = this.getCell(row, col);
55827         if(!el){
55828             return null;
55829         }
55830         var c = this.scroller.dom;
55831
55832         var ctop = parseInt(el.offsetTop, 10);
55833         var cleft = parseInt(el.offsetLeft, 10);
55834         var cbot = ctop + el.offsetHeight;
55835         var cright = cleft + el.offsetWidth;
55836         
55837         var ch = c.clientHeight - this.mainHd.dom.offsetHeight;
55838         var stop = parseInt(c.scrollTop, 10);
55839         var sleft = parseInt(c.scrollLeft, 10);
55840         var sbot = stop + ch;
55841         var sright = sleft + c.clientWidth;
55842         /*
55843         Roo.log('GridView.ensureVisible:' +
55844                 ' ctop:' + ctop +
55845                 ' c.clientHeight:' + c.clientHeight +
55846                 ' this.mainHd.dom.offsetHeight:' + this.mainHd.dom.offsetHeight +
55847                 ' stop:' + stop +
55848                 ' cbot:' + cbot +
55849                 ' sbot:' + sbot +
55850                 ' ch:' + ch  
55851                 );
55852         */
55853         if(ctop < stop){
55854              c.scrollTop = ctop;
55855             //Roo.log("set scrolltop to ctop DISABLE?");
55856         }else if(cbot > sbot){
55857             //Roo.log("set scrolltop to cbot-ch");
55858             c.scrollTop = cbot-ch;
55859         }
55860         
55861         if(hscroll !== false){
55862             if(cleft < sleft){
55863                 c.scrollLeft = cleft;
55864             }else if(cright > sright){
55865                 c.scrollLeft = cright-c.clientWidth;
55866             }
55867         }
55868          
55869         return el;
55870     },
55871
55872     updateColumns : function(){
55873         this.grid.stopEditing();
55874         var cm = this.grid.colModel, colIds = this.getColumnIds();
55875         //var totalWidth = cm.getTotalWidth();
55876         var pos = 0;
55877         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
55878             //if(cm.isHidden(i)) continue;
55879             var w = cm.getColumnWidth(i);
55880             this.css.updateRule(this.colSelector+this.idToCssName(colIds[i]), "width", (w - this.borderWidth) + "px");
55881             this.css.updateRule(this.hdSelector+this.idToCssName(colIds[i]), "width", (w - this.borderWidth) + "px");
55882         }
55883         this.updateSplitters();
55884     },
55885
55886     generateRules : function(cm){
55887         var ruleBuf = [], rulesId = this.idToCssName(this.grid.id)+ '-cssrules';
55888         Roo.util.CSS.removeStyleSheet(rulesId);
55889         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
55890             var cid = cm.getColumnId(i);
55891             var align = '';
55892             if(cm.config[i].align){
55893                 align = 'text-align:'+cm.config[i].align+';';
55894             }
55895             var hidden = '';
55896             if(cm.isHidden(i)){
55897                 hidden = 'display:none;';
55898             }
55899             var width = "width:" + (cm.getColumnWidth(i) - this.borderWidth) + "px;";
55900             ruleBuf.push(
55901                     this.colSelector, cid, " {\n", cm.config[i].css, align, width, "\n}\n",
55902                     this.hdSelector, cid, " {\n", align, width, "}\n",
55903                     this.tdSelector, cid, " {\n",hidden,"\n}\n",
55904                     this.splitSelector, cid, " {\n", hidden , "\n}\n");
55905         }
55906         return Roo.util.CSS.createStyleSheet(ruleBuf.join(""), rulesId);
55907     },
55908
55909     updateSplitters : function(){
55910         var cm = this.cm, s = this.getSplitters();
55911         if(s){ // splitters not created yet
55912             var pos = 0, locked = true;
55913             for(var i = 0, len = cm.getColumnCount(); i < len; i++){
55914                 if(cm.isHidden(i)) {
55915                     continue;
55916                 }
55917                 var w = cm.getColumnWidth(i); // make sure it's a number
55918                 if(!cm.isLocked(i) && locked){
55919                     pos = 0;
55920                     locked = false;
55921                 }
55922                 pos += w;
55923                 s[i].style.left = (pos-this.splitOffset) + "px";
55924             }
55925         }
55926     },
55927
55928     handleHiddenChange : function(colModel, colIndex, hidden){
55929         if(hidden){
55930             this.hideColumn(colIndex);
55931         }else{
55932             this.unhideColumn(colIndex);
55933         }
55934     },
55935
55936     hideColumn : function(colIndex){
55937         var cid = this.getColumnId(colIndex);
55938         this.css.updateRule(this.tdSelector+this.idToCssName(cid), "display", "none");
55939         this.css.updateRule(this.splitSelector+this.idToCssName(cid), "display", "none");
55940         if(Roo.isSafari){
55941             this.updateHeaders();
55942         }
55943         this.updateSplitters();
55944         this.layout();
55945     },
55946
55947     unhideColumn : function(colIndex){
55948         var cid = this.getColumnId(colIndex);
55949         this.css.updateRule(this.tdSelector+this.idToCssName(cid), "display", "");
55950         this.css.updateRule(this.splitSelector+this.idToCssName(cid), "display", "");
55951
55952         if(Roo.isSafari){
55953             this.updateHeaders();
55954         }
55955         this.updateSplitters();
55956         this.layout();
55957     },
55958
55959     insertRows : function(dm, firstRow, lastRow, isUpdate){
55960         if(firstRow == 0 && lastRow == dm.getCount()-1){
55961             this.refresh();
55962         }else{
55963             if(!isUpdate){
55964                 this.fireEvent("beforerowsinserted", this, firstRow, lastRow);
55965             }
55966             var s = this.getScrollState();
55967             var markup = this.renderRows(firstRow, lastRow);
55968             this.bufferRows(markup[0], this.getLockedTable(), firstRow);
55969             this.bufferRows(markup[1], this.getBodyTable(), firstRow);
55970             this.restoreScroll(s);
55971             if(!isUpdate){
55972                 this.fireEvent("rowsinserted", this, firstRow, lastRow);
55973                 this.syncRowHeights(firstRow, lastRow);
55974                 this.stripeRows(firstRow);
55975                 this.layout();
55976             }
55977         }
55978     },
55979
55980     bufferRows : function(markup, target, index){
55981         var before = null, trows = target.rows, tbody = target.tBodies[0];
55982         if(index < trows.length){
55983             before = trows[index];
55984         }
55985         var b = document.createElement("div");
55986         b.innerHTML = "<table><tbody>"+markup+"</tbody></table>";
55987         var rows = b.firstChild.rows;
55988         for(var i = 0, len = rows.length; i < len; i++){
55989             if(before){
55990                 tbody.insertBefore(rows[0], before);
55991             }else{
55992                 tbody.appendChild(rows[0]);
55993             }
55994         }
55995         b.innerHTML = "";
55996         b = null;
55997     },
55998
55999     deleteRows : function(dm, firstRow, lastRow){
56000         if(dm.getRowCount()<1){
56001             this.fireEvent("beforerefresh", this);
56002             this.mainBody.update("");
56003             this.lockedBody.update("");
56004             this.fireEvent("refresh", this);
56005         }else{
56006             this.fireEvent("beforerowsdeleted", this, firstRow, lastRow);
56007             var bt = this.getBodyTable();
56008             var tbody = bt.firstChild;
56009             var rows = bt.rows;
56010             for(var rowIndex = firstRow; rowIndex <= lastRow; rowIndex++){
56011                 tbody.removeChild(rows[firstRow]);
56012             }
56013             this.stripeRows(firstRow);
56014             this.fireEvent("rowsdeleted", this, firstRow, lastRow);
56015         }
56016     },
56017
56018     updateRows : function(dataSource, firstRow, lastRow){
56019         var s = this.getScrollState();
56020         this.refresh();
56021         this.restoreScroll(s);
56022     },
56023
56024     handleSort : function(dataSource, sortColumnIndex, sortDir, noRefresh){
56025         if(!noRefresh){
56026            this.refresh();
56027         }
56028         this.updateHeaderSortState();
56029     },
56030
56031     getScrollState : function(){
56032         
56033         var sb = this.scroller.dom;
56034         return {left: sb.scrollLeft, top: sb.scrollTop};
56035     },
56036
56037     stripeRows : function(startRow){
56038         if(!this.grid.stripeRows || this.ds.getCount() < 1){
56039             return;
56040         }
56041         startRow = startRow || 0;
56042         var rows = this.getBodyTable().rows;
56043         var lrows = this.getLockedTable().rows;
56044         var cls = ' x-grid-row-alt ';
56045         for(var i = startRow, len = rows.length; i < len; i++){
56046             var row = rows[i], lrow = lrows[i];
56047             var isAlt = ((i+1) % 2 == 0);
56048             var hasAlt = (' '+row.className + ' ').indexOf(cls) != -1;
56049             if(isAlt == hasAlt){
56050                 continue;
56051             }
56052             if(isAlt){
56053                 row.className += " x-grid-row-alt";
56054             }else{
56055                 row.className = row.className.replace("x-grid-row-alt", "");
56056             }
56057             if(lrow){
56058                 lrow.className = row.className;
56059             }
56060         }
56061     },
56062
56063     restoreScroll : function(state){
56064         //Roo.log('GridView.restoreScroll');
56065         var sb = this.scroller.dom;
56066         sb.scrollLeft = state.left;
56067         sb.scrollTop = state.top;
56068         this.syncScroll();
56069     },
56070
56071     syncScroll : function(){
56072         //Roo.log('GridView.syncScroll');
56073         var sb = this.scroller.dom;
56074         var sh = this.mainHd.dom;
56075         var bs = this.mainBody.dom;
56076         var lv = this.lockedBody.dom;
56077         sh.scrollLeft = bs.scrollLeft = sb.scrollLeft;
56078         lv.scrollTop = bs.scrollTop = sb.scrollTop;
56079     },
56080
56081     handleScroll : function(e){
56082         this.syncScroll();
56083         var sb = this.scroller.dom;
56084         this.grid.fireEvent("bodyscroll", sb.scrollLeft, sb.scrollTop);
56085         e.stopEvent();
56086     },
56087
56088     handleWheel : function(e){
56089         var d = e.getWheelDelta();
56090         this.scroller.dom.scrollTop -= d*22;
56091         // set this here to prevent jumpy scrolling on large tables
56092         this.lockedBody.dom.scrollTop = this.mainBody.dom.scrollTop = this.scroller.dom.scrollTop;
56093         e.stopEvent();
56094     },
56095
56096     renderRows : function(startRow, endRow){
56097         // pull in all the crap needed to render rows
56098         var g = this.grid, cm = g.colModel, ds = g.dataSource, stripe = g.stripeRows;
56099         var colCount = cm.getColumnCount();
56100
56101         if(ds.getCount() < 1){
56102             return ["", ""];
56103         }
56104
56105         // build a map for all the columns
56106         var cs = [];
56107         for(var i = 0; i < colCount; i++){
56108             var name = cm.getDataIndex(i);
56109             cs[i] = {
56110                 name : typeof name == 'undefined' ? ds.fields.get(i).name : name,
56111                 renderer : cm.getRenderer(i),
56112                 id : cm.getColumnId(i),
56113                 locked : cm.isLocked(i),
56114                 has_editor : cm.isCellEditable(i)
56115             };
56116         }
56117
56118         startRow = startRow || 0;
56119         endRow = typeof endRow == "undefined"? ds.getCount()-1 : endRow;
56120
56121         // records to render
56122         var rs = ds.getRange(startRow, endRow);
56123
56124         return this.doRender(cs, rs, ds, startRow, colCount, stripe);
56125     },
56126
56127     // As much as I hate to duplicate code, this was branched because FireFox really hates
56128     // [].join("") on strings. The performance difference was substantial enough to
56129     // branch this function
56130     doRender : Roo.isGecko ?
56131             function(cs, rs, ds, startRow, colCount, stripe){
56132                 var ts = this.templates, ct = ts.cell, rt = ts.row;
56133                 // buffers
56134                 var buf = "", lbuf = "", cb, lcb, c, p = {}, rp = {}, r, rowIndex;
56135                 
56136                 var hasListener = this.grid.hasListener('rowclass');
56137                 var rowcfg = {};
56138                 for(var j = 0, len = rs.length; j < len; j++){
56139                     r = rs[j]; cb = ""; lcb = ""; rowIndex = (j+startRow);
56140                     for(var i = 0; i < colCount; i++){
56141                         c = cs[i];
56142                         p.cellId = "x-grid-cell-" + rowIndex + "-" + i;
56143                         p.id = c.id;
56144                         p.css = p.attr = "";
56145                         p.value = c.renderer(r.data[c.name], p, r, rowIndex, i, ds);
56146                         if(p.value == undefined || p.value === "") {
56147                             p.value = "&#160;";
56148                         }
56149                         if(c.has_editor){
56150                             p.css += ' x-grid-editable-cell';
56151                         }
56152                         if(c.dirty && typeof r.modified[c.name] !== 'undefined'){
56153                             p.css +=  ' x-grid-dirty-cell';
56154                         }
56155                         var markup = ct.apply(p);
56156                         if(!c.locked){
56157                             cb+= markup;
56158                         }else{
56159                             lcb+= markup;
56160                         }
56161                     }
56162                     var alt = [];
56163                     if(stripe && ((rowIndex+1) % 2 == 0)){
56164                         alt.push("x-grid-row-alt")
56165                     }
56166                     if(r.dirty){
56167                         alt.push(  " x-grid-dirty-row");
56168                     }
56169                     rp.cells = lcb;
56170                     if(this.getRowClass){
56171                         alt.push(this.getRowClass(r, rowIndex));
56172                     }
56173                     if (hasListener) {
56174                         rowcfg = {
56175                              
56176                             record: r,
56177                             rowIndex : rowIndex,
56178                             rowClass : ''
56179                         };
56180                         this.grid.fireEvent('rowclass', this, rowcfg);
56181                         alt.push(rowcfg.rowClass);
56182                     }
56183                     rp.alt = alt.join(" ");
56184                     lbuf+= rt.apply(rp);
56185                     rp.cells = cb;
56186                     buf+=  rt.apply(rp);
56187                 }
56188                 return [lbuf, buf];
56189             } :
56190             function(cs, rs, ds, startRow, colCount, stripe){
56191                 var ts = this.templates, ct = ts.cell, rt = ts.row;
56192                 // buffers
56193                 var buf = [], lbuf = [], cb, lcb, c, p = {}, rp = {}, r, rowIndex;
56194                 var hasListener = this.grid.hasListener('rowclass');
56195  
56196                 var rowcfg = {};
56197                 for(var j = 0, len = rs.length; j < len; j++){
56198                     r = rs[j]; cb = []; lcb = []; rowIndex = (j+startRow);
56199                     for(var i = 0; i < colCount; i++){
56200                         c = cs[i];
56201                         p.cellId = "x-grid-cell-" + rowIndex + "-" + i;
56202                         p.id = c.id;
56203                         p.css = p.attr = "";
56204                         p.value = c.renderer(r.data[c.name], p, r, rowIndex, i, ds);
56205                         if(p.value == undefined || p.value === "") {
56206                             p.value = "&#160;";
56207                         }
56208                         //Roo.log(c);
56209                          if(c.has_editor){
56210                             p.css += ' x-grid-editable-cell';
56211                         }
56212                         if(r.dirty && typeof r.modified[c.name] !== 'undefined'){
56213                             p.css += ' x-grid-dirty-cell' 
56214                         }
56215                         
56216                         var markup = ct.apply(p);
56217                         if(!c.locked){
56218                             cb[cb.length] = markup;
56219                         }else{
56220                             lcb[lcb.length] = markup;
56221                         }
56222                     }
56223                     var alt = [];
56224                     if(stripe && ((rowIndex+1) % 2 == 0)){
56225                         alt.push( "x-grid-row-alt");
56226                     }
56227                     if(r.dirty){
56228                         alt.push(" x-grid-dirty-row");
56229                     }
56230                     rp.cells = lcb;
56231                     if(this.getRowClass){
56232                         alt.push( this.getRowClass(r, rowIndex));
56233                     }
56234                     if (hasListener) {
56235                         rowcfg = {
56236                              
56237                             record: r,
56238                             rowIndex : rowIndex,
56239                             rowClass : ''
56240                         };
56241                         this.grid.fireEvent('rowclass', this, rowcfg);
56242                         alt.push(rowcfg.rowClass);
56243                     }
56244                     
56245                     rp.alt = alt.join(" ");
56246                     rp.cells = lcb.join("");
56247                     lbuf[lbuf.length] = rt.apply(rp);
56248                     rp.cells = cb.join("");
56249                     buf[buf.length] =  rt.apply(rp);
56250                 }
56251                 return [lbuf.join(""), buf.join("")];
56252             },
56253
56254     renderBody : function(){
56255         var markup = this.renderRows();
56256         var bt = this.templates.body;
56257         return [bt.apply({rows: markup[0]}), bt.apply({rows: markup[1]})];
56258     },
56259
56260     /**
56261      * Refreshes the grid
56262      * @param {Boolean} headersToo
56263      */
56264     refresh : function(headersToo){
56265         this.fireEvent("beforerefresh", this);
56266         this.grid.stopEditing();
56267         var result = this.renderBody();
56268         this.lockedBody.update(result[0]);
56269         this.mainBody.update(result[1]);
56270         if(headersToo === true){
56271             this.updateHeaders();
56272             this.updateColumns();
56273             this.updateSplitters();
56274             this.updateHeaderSortState();
56275         }
56276         this.syncRowHeights();
56277         this.layout();
56278         this.fireEvent("refresh", this);
56279     },
56280
56281     handleColumnMove : function(cm, oldIndex, newIndex){
56282         this.indexMap = null;
56283         var s = this.getScrollState();
56284         this.refresh(true);
56285         this.restoreScroll(s);
56286         this.afterMove(newIndex);
56287     },
56288
56289     afterMove : function(colIndex){
56290         if(this.enableMoveAnim && Roo.enableFx){
56291             this.fly(this.getHeaderCell(colIndex).firstChild).highlight(this.hlColor);
56292         }
56293         // if multisort - fix sortOrder, and reload..
56294         if (this.grid.dataSource.multiSort) {
56295             // the we can call sort again..
56296             var dm = this.grid.dataSource;
56297             var cm = this.grid.colModel;
56298             var so = [];
56299             for(var i = 0; i < cm.config.length; i++ ) {
56300                 
56301                 if ((typeof(dm.sortToggle[cm.config[i].dataIndex]) == 'undefined')) {
56302                     continue; // dont' bother, it's not in sort list or being set.
56303                 }
56304                 
56305                 so.push(cm.config[i].dataIndex);
56306             };
56307             dm.sortOrder = so;
56308             dm.load(dm.lastOptions);
56309             
56310             
56311         }
56312         
56313     },
56314
56315     updateCell : function(dm, rowIndex, dataIndex){
56316         var colIndex = this.getColumnIndexByDataIndex(dataIndex);
56317         if(typeof colIndex == "undefined"){ // not present in grid
56318             return;
56319         }
56320         var cm = this.grid.colModel;
56321         var cell = this.getCell(rowIndex, colIndex);
56322         var cellText = this.getCellText(rowIndex, colIndex);
56323
56324         var p = {
56325             cellId : "x-grid-cell-" + rowIndex + "-" + colIndex,
56326             id : cm.getColumnId(colIndex),
56327             css: colIndex == cm.getColumnCount()-1 ? "x-grid-col-last" : ""
56328         };
56329         var renderer = cm.getRenderer(colIndex);
56330         var val = renderer(dm.getValueAt(rowIndex, dataIndex), p, rowIndex, colIndex, dm);
56331         if(typeof val == "undefined" || val === "") {
56332             val = "&#160;";
56333         }
56334         cellText.innerHTML = val;
56335         cell.className = this.cellClass + " " + this.idToCssName(p.cellId) + " " + p.css;
56336         this.syncRowHeights(rowIndex, rowIndex);
56337     },
56338
56339     calcColumnWidth : function(colIndex, maxRowsToMeasure){
56340         var maxWidth = 0;
56341         if(this.grid.autoSizeHeaders){
56342             var h = this.getHeaderCellMeasure(colIndex);
56343             maxWidth = Math.max(maxWidth, h.scrollWidth);
56344         }
56345         var tb, index;
56346         if(this.cm.isLocked(colIndex)){
56347             tb = this.getLockedTable();
56348             index = colIndex;
56349         }else{
56350             tb = this.getBodyTable();
56351             index = colIndex - this.cm.getLockedCount();
56352         }
56353         if(tb && tb.rows){
56354             var rows = tb.rows;
56355             var stopIndex = Math.min(maxRowsToMeasure || rows.length, rows.length);
56356             for(var i = 0; i < stopIndex; i++){
56357                 var cell = rows[i].childNodes[index].firstChild;
56358                 maxWidth = Math.max(maxWidth, cell.scrollWidth);
56359             }
56360         }
56361         return maxWidth + /*margin for error in IE*/ 5;
56362     },
56363     /**
56364      * Autofit a column to its content.
56365      * @param {Number} colIndex
56366      * @param {Boolean} forceMinSize true to force the column to go smaller if possible
56367      */
56368      autoSizeColumn : function(colIndex, forceMinSize, suppressEvent){
56369          if(this.cm.isHidden(colIndex)){
56370              return; // can't calc a hidden column
56371          }
56372         if(forceMinSize){
56373             var cid = this.cm.getColumnId(colIndex);
56374             this.css.updateRule(this.colSelector +this.idToCssName( cid), "width", this.grid.minColumnWidth + "px");
56375            if(this.grid.autoSizeHeaders){
56376                this.css.updateRule(this.hdSelector + this.idToCssName(cid), "width", this.grid.minColumnWidth + "px");
56377            }
56378         }
56379         var newWidth = this.calcColumnWidth(colIndex);
56380         this.cm.setColumnWidth(colIndex,
56381             Math.max(this.grid.minColumnWidth, newWidth), suppressEvent);
56382         if(!suppressEvent){
56383             this.grid.fireEvent("columnresize", colIndex, newWidth);
56384         }
56385     },
56386
56387     /**
56388      * Autofits all columns to their content and then expands to fit any extra space in the grid
56389      */
56390      autoSizeColumns : function(){
56391         var cm = this.grid.colModel;
56392         var colCount = cm.getColumnCount();
56393         for(var i = 0; i < colCount; i++){
56394             this.autoSizeColumn(i, true, true);
56395         }
56396         if(cm.getTotalWidth() < this.scroller.dom.clientWidth){
56397             this.fitColumns();
56398         }else{
56399             this.updateColumns();
56400             this.layout();
56401         }
56402     },
56403
56404     /**
56405      * Autofits all columns to the grid's width proportionate with their current size
56406      * @param {Boolean} reserveScrollSpace Reserve space for a scrollbar
56407      */
56408     fitColumns : function(reserveScrollSpace){
56409         var cm = this.grid.colModel;
56410         var colCount = cm.getColumnCount();
56411         var cols = [];
56412         var width = 0;
56413         var i, w;
56414         for (i = 0; i < colCount; i++){
56415             if(!cm.isHidden(i) && !cm.isFixed(i)){
56416                 w = cm.getColumnWidth(i);
56417                 cols.push(i);
56418                 cols.push(w);
56419                 width += w;
56420             }
56421         }
56422         var avail = Math.min(this.scroller.dom.clientWidth, this.el.getWidth());
56423         if(reserveScrollSpace){
56424             avail -= 17;
56425         }
56426         var frac = (avail - cm.getTotalWidth())/width;
56427         while (cols.length){
56428             w = cols.pop();
56429             i = cols.pop();
56430             cm.setColumnWidth(i, Math.floor(w + w*frac), true);
56431         }
56432         this.updateColumns();
56433         this.layout();
56434     },
56435
56436     onRowSelect : function(rowIndex){
56437         var row = this.getRowComposite(rowIndex);
56438         row.addClass("x-grid-row-selected");
56439     },
56440
56441     onRowDeselect : function(rowIndex){
56442         var row = this.getRowComposite(rowIndex);
56443         row.removeClass("x-grid-row-selected");
56444     },
56445
56446     onCellSelect : function(row, col){
56447         var cell = this.getCell(row, col);
56448         if(cell){
56449             Roo.fly(cell).addClass("x-grid-cell-selected");
56450         }
56451     },
56452
56453     onCellDeselect : function(row, col){
56454         var cell = this.getCell(row, col);
56455         if(cell){
56456             Roo.fly(cell).removeClass("x-grid-cell-selected");
56457         }
56458     },
56459
56460     updateHeaderSortState : function(){
56461         
56462         // sort state can be single { field: xxx, direction : yyy}
56463         // or   { xxx=>ASC , yyy : DESC ..... }
56464         
56465         var mstate = {};
56466         if (!this.ds.multiSort) { 
56467             var state = this.ds.getSortState();
56468             if(!state){
56469                 return;
56470             }
56471             mstate[state.field] = state.direction;
56472             // FIXME... - this is not used here.. but might be elsewhere..
56473             this.sortState = state;
56474             
56475         } else {
56476             mstate = this.ds.sortToggle;
56477         }
56478         //remove existing sort classes..
56479         
56480         var sc = this.sortClasses;
56481         var hds = this.el.select(this.headerSelector).removeClass(sc);
56482         
56483         for(var f in mstate) {
56484         
56485             var sortColumn = this.cm.findColumnIndex(f);
56486             
56487             if(sortColumn != -1){
56488                 var sortDir = mstate[f];        
56489                 hds.item(sortColumn).addClass(sc[sortDir == "DESC" ? 1 : 0]);
56490             }
56491         }
56492         
56493          
56494         
56495     },
56496
56497
56498     handleHeaderClick : function(g, index,e){
56499         
56500         Roo.log("header click");
56501         
56502         if (Roo.isTouch) {
56503             // touch events on header are handled by context
56504             this.handleHdCtx(g,index,e);
56505             return;
56506         }
56507         
56508         
56509         if(this.headersDisabled){
56510             return;
56511         }
56512         var dm = g.dataSource, cm = g.colModel;
56513         if(!cm.isSortable(index)){
56514             return;
56515         }
56516         g.stopEditing();
56517         
56518         if (dm.multiSort) {
56519             // update the sortOrder
56520             var so = [];
56521             for(var i = 0; i < cm.config.length; i++ ) {
56522                 
56523                 if ((typeof(dm.sortToggle[cm.config[i].dataIndex]) == 'undefined') && (index != i)) {
56524                     continue; // dont' bother, it's not in sort list or being set.
56525                 }
56526                 
56527                 so.push(cm.config[i].dataIndex);
56528             };
56529             dm.sortOrder = so;
56530         }
56531         
56532         
56533         dm.sort(cm.getDataIndex(index));
56534     },
56535
56536
56537     destroy : function(){
56538         if(this.colMenu){
56539             this.colMenu.removeAll();
56540             Roo.menu.MenuMgr.unregister(this.colMenu);
56541             this.colMenu.getEl().remove();
56542             delete this.colMenu;
56543         }
56544         if(this.hmenu){
56545             this.hmenu.removeAll();
56546             Roo.menu.MenuMgr.unregister(this.hmenu);
56547             this.hmenu.getEl().remove();
56548             delete this.hmenu;
56549         }
56550         if(this.grid.enableColumnMove){
56551             var dds = Roo.dd.DDM.ids['gridHeader' + this.grid.getGridEl().id];
56552             if(dds){
56553                 for(var dd in dds){
56554                     if(!dds[dd].config.isTarget && dds[dd].dragElId){
56555                         var elid = dds[dd].dragElId;
56556                         dds[dd].unreg();
56557                         Roo.get(elid).remove();
56558                     } else if(dds[dd].config.isTarget){
56559                         dds[dd].proxyTop.remove();
56560                         dds[dd].proxyBottom.remove();
56561                         dds[dd].unreg();
56562                     }
56563                     if(Roo.dd.DDM.locationCache[dd]){
56564                         delete Roo.dd.DDM.locationCache[dd];
56565                     }
56566                 }
56567                 delete Roo.dd.DDM.ids['gridHeader' + this.grid.getGridEl().id];
56568             }
56569         }
56570         Roo.util.CSS.removeStyleSheet(this.idToCssName(this.grid.id) + '-cssrules');
56571         this.bind(null, null);
56572         Roo.EventManager.removeResizeListener(this.onWindowResize, this);
56573     },
56574
56575     handleLockChange : function(){
56576         this.refresh(true);
56577     },
56578
56579     onDenyColumnLock : function(){
56580
56581     },
56582
56583     onDenyColumnHide : function(){
56584
56585     },
56586
56587     handleHdMenuClick : function(item){
56588         var index = this.hdCtxIndex;
56589         var cm = this.cm, ds = this.ds;
56590         switch(item.id){
56591             case "asc":
56592                 ds.sort(cm.getDataIndex(index), "ASC");
56593                 break;
56594             case "desc":
56595                 ds.sort(cm.getDataIndex(index), "DESC");
56596                 break;
56597             case "lock":
56598                 var lc = cm.getLockedCount();
56599                 if(cm.getColumnCount(true) <= lc+1){
56600                     this.onDenyColumnLock();
56601                     return;
56602                 }
56603                 if(lc != index){
56604                     cm.setLocked(index, true, true);
56605                     cm.moveColumn(index, lc);
56606                     this.grid.fireEvent("columnmove", index, lc);
56607                 }else{
56608                     cm.setLocked(index, true);
56609                 }
56610             break;
56611             case "unlock":
56612                 var lc = cm.getLockedCount();
56613                 if((lc-1) != index){
56614                     cm.setLocked(index, false, true);
56615                     cm.moveColumn(index, lc-1);
56616                     this.grid.fireEvent("columnmove", index, lc-1);
56617                 }else{
56618                     cm.setLocked(index, false);
56619                 }
56620             break;
56621             case 'wider': // used to expand cols on touch..
56622             case 'narrow':
56623                 var cw = cm.getColumnWidth(index);
56624                 cw += (item.id == 'wider' ? 1 : -1) * 50;
56625                 cw = Math.max(0, cw);
56626                 cw = Math.min(cw,4000);
56627                 cm.setColumnWidth(index, cw);
56628                 break;
56629                 
56630             default:
56631                 index = cm.getIndexById(item.id.substr(4));
56632                 if(index != -1){
56633                     if(item.checked && cm.getColumnCount(true) <= 1){
56634                         this.onDenyColumnHide();
56635                         return false;
56636                     }
56637                     cm.setHidden(index, item.checked);
56638                 }
56639         }
56640         return true;
56641     },
56642
56643     beforeColMenuShow : function(){
56644         var cm = this.cm,  colCount = cm.getColumnCount();
56645         this.colMenu.removeAll();
56646         for(var i = 0; i < colCount; i++){
56647             this.colMenu.add(new Roo.menu.CheckItem({
56648                 id: "col-"+cm.getColumnId(i),
56649                 text: cm.getColumnHeader(i),
56650                 checked: !cm.isHidden(i),
56651                 hideOnClick:false
56652             }));
56653         }
56654     },
56655
56656     handleHdCtx : function(g, index, e){
56657         e.stopEvent();
56658         var hd = this.getHeaderCell(index);
56659         this.hdCtxIndex = index;
56660         var ms = this.hmenu.items, cm = this.cm;
56661         ms.get("asc").setDisabled(!cm.isSortable(index));
56662         ms.get("desc").setDisabled(!cm.isSortable(index));
56663         if(this.grid.enableColLock !== false){
56664             ms.get("lock").setDisabled(cm.isLocked(index));
56665             ms.get("unlock").setDisabled(!cm.isLocked(index));
56666         }
56667         this.hmenu.show(hd, "tl-bl");
56668     },
56669
56670     handleHdOver : function(e){
56671         var hd = this.findHeaderCell(e.getTarget());
56672         if(hd && !this.headersDisabled){
56673             if(this.grid.colModel.isSortable(this.getCellIndex(hd))){
56674                this.fly(hd).addClass("x-grid-hd-over");
56675             }
56676         }
56677     },
56678
56679     handleHdOut : function(e){
56680         var hd = this.findHeaderCell(e.getTarget());
56681         if(hd){
56682             this.fly(hd).removeClass("x-grid-hd-over");
56683         }
56684     },
56685
56686     handleSplitDblClick : function(e, t){
56687         var i = this.getCellIndex(t);
56688         if(this.grid.enableColumnResize !== false && this.cm.isResizable(i) && !this.cm.isFixed(i)){
56689             this.autoSizeColumn(i, true);
56690             this.layout();
56691         }
56692     },
56693
56694     render : function(){
56695
56696         var cm = this.cm;
56697         var colCount = cm.getColumnCount();
56698
56699         if(this.grid.monitorWindowResize === true){
56700             Roo.EventManager.onWindowResize(this.onWindowResize, this, true);
56701         }
56702         var header = this.renderHeaders();
56703         var body = this.templates.body.apply({rows:""});
56704         var html = this.templates.master.apply({
56705             lockedBody: body,
56706             body: body,
56707             lockedHeader: header[0],
56708             header: header[1]
56709         });
56710
56711         //this.updateColumns();
56712
56713         this.grid.getGridEl().dom.innerHTML = html;
56714
56715         this.initElements();
56716         
56717         // a kludge to fix the random scolling effect in webkit
56718         this.el.on("scroll", function() {
56719             this.el.dom.scrollTop=0; // hopefully not recursive..
56720         },this);
56721
56722         this.scroller.on("scroll", this.handleScroll, this);
56723         this.lockedBody.on("mousewheel", this.handleWheel, this);
56724         this.mainBody.on("mousewheel", this.handleWheel, this);
56725
56726         this.mainHd.on("mouseover", this.handleHdOver, this);
56727         this.mainHd.on("mouseout", this.handleHdOut, this);
56728         this.mainHd.on("dblclick", this.handleSplitDblClick, this,
56729                 {delegate: "."+this.splitClass});
56730
56731         this.lockedHd.on("mouseover", this.handleHdOver, this);
56732         this.lockedHd.on("mouseout", this.handleHdOut, this);
56733         this.lockedHd.on("dblclick", this.handleSplitDblClick, this,
56734                 {delegate: "."+this.splitClass});
56735
56736         if(this.grid.enableColumnResize !== false && Roo.grid.SplitDragZone){
56737             new Roo.grid.SplitDragZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
56738         }
56739
56740         this.updateSplitters();
56741
56742         if(this.grid.enableColumnMove && Roo.grid.HeaderDragZone){
56743             new Roo.grid.HeaderDragZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
56744             new Roo.grid.HeaderDropZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
56745         }
56746
56747         if(this.grid.enableCtxMenu !== false && Roo.menu.Menu){
56748             this.hmenu = new Roo.menu.Menu({id: this.grid.id + "-hctx"});
56749             this.hmenu.add(
56750                 {id:"asc", text: this.sortAscText, cls: "xg-hmenu-sort-asc"},
56751                 {id:"desc", text: this.sortDescText, cls: "xg-hmenu-sort-desc"}
56752             );
56753             if(this.grid.enableColLock !== false){
56754                 this.hmenu.add('-',
56755                     {id:"lock", text: this.lockText, cls: "xg-hmenu-lock"},
56756                     {id:"unlock", text: this.unlockText, cls: "xg-hmenu-unlock"}
56757                 );
56758             }
56759             if (Roo.isTouch) {
56760                  this.hmenu.add('-',
56761                     {id:"wider", text: this.columnsWiderText},
56762                     {id:"narrow", text: this.columnsNarrowText }
56763                 );
56764                 
56765                  
56766             }
56767             
56768             if(this.grid.enableColumnHide !== false){
56769
56770                 this.colMenu = new Roo.menu.Menu({id:this.grid.id + "-hcols-menu"});
56771                 this.colMenu.on("beforeshow", this.beforeColMenuShow, this);
56772                 this.colMenu.on("itemclick", this.handleHdMenuClick, this);
56773
56774                 this.hmenu.add('-',
56775                     {id:"columns", text: this.columnsText, menu: this.colMenu}
56776                 );
56777             }
56778             this.hmenu.on("itemclick", this.handleHdMenuClick, this);
56779
56780             this.grid.on("headercontextmenu", this.handleHdCtx, this);
56781         }
56782
56783         if((this.grid.enableDragDrop || this.grid.enableDrag) && Roo.grid.GridDragZone){
56784             this.dd = new Roo.grid.GridDragZone(this.grid, {
56785                 ddGroup : this.grid.ddGroup || 'GridDD'
56786             });
56787             
56788         }
56789
56790         /*
56791         for(var i = 0; i < colCount; i++){
56792             if(cm.isHidden(i)){
56793                 this.hideColumn(i);
56794             }
56795             if(cm.config[i].align){
56796                 this.css.updateRule(this.colSelector + i, "textAlign", cm.config[i].align);
56797                 this.css.updateRule(this.hdSelector + i, "textAlign", cm.config[i].align);
56798             }
56799         }*/
56800         
56801         this.updateHeaderSortState();
56802
56803         this.beforeInitialResize();
56804         this.layout(true);
56805
56806         // two part rendering gives faster view to the user
56807         this.renderPhase2.defer(1, this);
56808     },
56809
56810     renderPhase2 : function(){
56811         // render the rows now
56812         this.refresh();
56813         if(this.grid.autoSizeColumns){
56814             this.autoSizeColumns();
56815         }
56816     },
56817
56818     beforeInitialResize : function(){
56819
56820     },
56821
56822     onColumnSplitterMoved : function(i, w){
56823         this.userResized = true;
56824         var cm = this.grid.colModel;
56825         cm.setColumnWidth(i, w, true);
56826         var cid = cm.getColumnId(i);
56827         this.css.updateRule(this.colSelector + this.idToCssName(cid), "width", (w-this.borderWidth) + "px");
56828         this.css.updateRule(this.hdSelector + this.idToCssName(cid), "width", (w-this.borderWidth) + "px");
56829         this.updateSplitters();
56830         this.layout();
56831         this.grid.fireEvent("columnresize", i, w);
56832     },
56833
56834     syncRowHeights : function(startIndex, endIndex){
56835         if(this.grid.enableRowHeightSync === true && this.cm.getLockedCount() > 0){
56836             startIndex = startIndex || 0;
56837             var mrows = this.getBodyTable().rows;
56838             var lrows = this.getLockedTable().rows;
56839             var len = mrows.length-1;
56840             endIndex = Math.min(endIndex || len, len);
56841             for(var i = startIndex; i <= endIndex; i++){
56842                 var m = mrows[i], l = lrows[i];
56843                 var h = Math.max(m.offsetHeight, l.offsetHeight);
56844                 m.style.height = l.style.height = h + "px";
56845             }
56846         }
56847     },
56848
56849     layout : function(initialRender, is2ndPass){
56850         var g = this.grid;
56851         var auto = g.autoHeight;
56852         var scrollOffset = 16;
56853         var c = g.getGridEl(), cm = this.cm,
56854                 expandCol = g.autoExpandColumn,
56855                 gv = this;
56856         //c.beginMeasure();
56857
56858         if(!c.dom.offsetWidth){ // display:none?
56859             if(initialRender){
56860                 this.lockedWrap.show();
56861                 this.mainWrap.show();
56862             }
56863             return;
56864         }
56865
56866         var hasLock = this.cm.isLocked(0);
56867
56868         var tbh = this.headerPanel.getHeight();
56869         var bbh = this.footerPanel.getHeight();
56870
56871         if(auto){
56872             var ch = this.getBodyTable().offsetHeight + tbh + bbh + this.mainHd.getHeight();
56873             var newHeight = ch + c.getBorderWidth("tb");
56874             if(g.maxHeight){
56875                 newHeight = Math.min(g.maxHeight, newHeight);
56876             }
56877             c.setHeight(newHeight);
56878         }
56879
56880         if(g.autoWidth){
56881             c.setWidth(cm.getTotalWidth()+c.getBorderWidth('lr'));
56882         }
56883
56884         var s = this.scroller;
56885
56886         var csize = c.getSize(true);
56887
56888         this.el.setSize(csize.width, csize.height);
56889
56890         this.headerPanel.setWidth(csize.width);
56891         this.footerPanel.setWidth(csize.width);
56892
56893         var hdHeight = this.mainHd.getHeight();
56894         var vw = csize.width;
56895         var vh = csize.height - (tbh + bbh);
56896
56897         s.setSize(vw, vh);
56898
56899         var bt = this.getBodyTable();
56900         
56901         if(cm.getLockedCount() == cm.config.length){
56902             bt = this.getLockedTable();
56903         }
56904         
56905         var ltWidth = hasLock ?
56906                       Math.max(this.getLockedTable().offsetWidth, this.lockedHd.dom.firstChild.offsetWidth) : 0;
56907
56908         var scrollHeight = bt.offsetHeight;
56909         var scrollWidth = ltWidth + bt.offsetWidth;
56910         var vscroll = false, hscroll = false;
56911
56912         this.scrollSizer.setSize(scrollWidth, scrollHeight+hdHeight);
56913
56914         var lw = this.lockedWrap, mw = this.mainWrap;
56915         var lb = this.lockedBody, mb = this.mainBody;
56916
56917         setTimeout(function(){
56918             var t = s.dom.offsetTop;
56919             var w = s.dom.clientWidth,
56920                 h = s.dom.clientHeight;
56921
56922             lw.setTop(t);
56923             lw.setSize(ltWidth, h);
56924
56925             mw.setLeftTop(ltWidth, t);
56926             mw.setSize(w-ltWidth, h);
56927
56928             lb.setHeight(h-hdHeight);
56929             mb.setHeight(h-hdHeight);
56930
56931             if(is2ndPass !== true && !gv.userResized && expandCol){
56932                 // high speed resize without full column calculation
56933                 
56934                 var ci = cm.getIndexById(expandCol);
56935                 if (ci < 0) {
56936                     ci = cm.findColumnIndex(expandCol);
56937                 }
56938                 ci = Math.max(0, ci); // make sure it's got at least the first col.
56939                 var expandId = cm.getColumnId(ci);
56940                 var  tw = cm.getTotalWidth(false);
56941                 var currentWidth = cm.getColumnWidth(ci);
56942                 var cw = Math.min(Math.max(((w-tw)+currentWidth-2)-/*scrollbar*/(w <= s.dom.offsetWidth ? 0 : 18), g.autoExpandMin), g.autoExpandMax);
56943                 if(currentWidth != cw){
56944                     cm.setColumnWidth(ci, cw, true);
56945                     gv.css.updateRule(gv.colSelector+gv.idToCssName(expandId), "width", (cw - gv.borderWidth) + "px");
56946                     gv.css.updateRule(gv.hdSelector+gv.idToCssName(expandId), "width", (cw - gv.borderWidth) + "px");
56947                     gv.updateSplitters();
56948                     gv.layout(false, true);
56949                 }
56950             }
56951
56952             if(initialRender){
56953                 lw.show();
56954                 mw.show();
56955             }
56956             //c.endMeasure();
56957         }, 10);
56958     },
56959
56960     onWindowResize : function(){
56961         if(!this.grid.monitorWindowResize || this.grid.autoHeight){
56962             return;
56963         }
56964         this.layout();
56965     },
56966
56967     appendFooter : function(parentEl){
56968         return null;
56969     },
56970
56971     sortAscText : "Sort Ascending",
56972     sortDescText : "Sort Descending",
56973     lockText : "Lock Column",
56974     unlockText : "Unlock Column",
56975     columnsText : "Columns",
56976  
56977     columnsWiderText : "Wider",
56978     columnsNarrowText : "Thinner"
56979 });
56980
56981
56982 Roo.grid.GridView.ColumnDragZone = function(grid, hd){
56983     Roo.grid.GridView.ColumnDragZone.superclass.constructor.call(this, grid, hd, null);
56984     this.proxy.el.addClass('x-grid3-col-dd');
56985 };
56986
56987 Roo.extend(Roo.grid.GridView.ColumnDragZone, Roo.grid.HeaderDragZone, {
56988     handleMouseDown : function(e){
56989
56990     },
56991
56992     callHandleMouseDown : function(e){
56993         Roo.grid.GridView.ColumnDragZone.superclass.handleMouseDown.call(this, e);
56994     }
56995 });
56996 /*
56997  * Based on:
56998  * Ext JS Library 1.1.1
56999  * Copyright(c) 2006-2007, Ext JS, LLC.
57000  *
57001  * Originally Released Under LGPL - original licence link has changed is not relivant.
57002  *
57003  * Fork - LGPL
57004  * <script type="text/javascript">
57005  */
57006  
57007 // private
57008 // This is a support class used internally by the Grid components
57009 Roo.grid.SplitDragZone = function(grid, hd, hd2){
57010     this.grid = grid;
57011     this.view = grid.getView();
57012     this.proxy = this.view.resizeProxy;
57013     Roo.grid.SplitDragZone.superclass.constructor.call(this, hd,
57014         "gridSplitters" + this.grid.getGridEl().id, {
57015         dragElId : Roo.id(this.proxy.dom), resizeFrame:false
57016     });
57017     this.setHandleElId(Roo.id(hd));
57018     this.setOuterHandleElId(Roo.id(hd2));
57019     this.scroll = false;
57020 };
57021 Roo.extend(Roo.grid.SplitDragZone, Roo.dd.DDProxy, {
57022     fly: Roo.Element.fly,
57023
57024     b4StartDrag : function(x, y){
57025         this.view.headersDisabled = true;
57026         this.proxy.setHeight(this.view.mainWrap.getHeight());
57027         var w = this.cm.getColumnWidth(this.cellIndex);
57028         var minw = Math.max(w-this.grid.minColumnWidth, 0);
57029         this.resetConstraints();
57030         this.setXConstraint(minw, 1000);
57031         this.setYConstraint(0, 0);
57032         this.minX = x - minw;
57033         this.maxX = x + 1000;
57034         this.startPos = x;
57035         Roo.dd.DDProxy.prototype.b4StartDrag.call(this, x, y);
57036     },
57037
57038
57039     handleMouseDown : function(e){
57040         ev = Roo.EventObject.setEvent(e);
57041         var t = this.fly(ev.getTarget());
57042         if(t.hasClass("x-grid-split")){
57043             this.cellIndex = this.view.getCellIndex(t.dom);
57044             this.split = t.dom;
57045             this.cm = this.grid.colModel;
57046             if(this.cm.isResizable(this.cellIndex) && !this.cm.isFixed(this.cellIndex)){
57047                 Roo.grid.SplitDragZone.superclass.handleMouseDown.apply(this, arguments);
57048             }
57049         }
57050     },
57051
57052     endDrag : function(e){
57053         this.view.headersDisabled = false;
57054         var endX = Math.max(this.minX, Roo.lib.Event.getPageX(e));
57055         var diff = endX - this.startPos;
57056         this.view.onColumnSplitterMoved(this.cellIndex, this.cm.getColumnWidth(this.cellIndex)+diff);
57057     },
57058
57059     autoOffset : function(){
57060         this.setDelta(0,0);
57061     }
57062 });/*
57063  * Based on:
57064  * Ext JS Library 1.1.1
57065  * Copyright(c) 2006-2007, Ext JS, LLC.
57066  *
57067  * Originally Released Under LGPL - original licence link has changed is not relivant.
57068  *
57069  * Fork - LGPL
57070  * <script type="text/javascript">
57071  */
57072  
57073 // private
57074 // This is a support class used internally by the Grid components
57075 Roo.grid.GridDragZone = function(grid, config){
57076     this.view = grid.getView();
57077     Roo.grid.GridDragZone.superclass.constructor.call(this, this.view.mainBody.dom, config);
57078     if(this.view.lockedBody){
57079         this.setHandleElId(Roo.id(this.view.mainBody.dom));
57080         this.setOuterHandleElId(Roo.id(this.view.lockedBody.dom));
57081     }
57082     this.scroll = false;
57083     this.grid = grid;
57084     this.ddel = document.createElement('div');
57085     this.ddel.className = 'x-grid-dd-wrap';
57086 };
57087
57088 Roo.extend(Roo.grid.GridDragZone, Roo.dd.DragZone, {
57089     ddGroup : "GridDD",
57090
57091     getDragData : function(e){
57092         var t = Roo.lib.Event.getTarget(e);
57093         var rowIndex = this.view.findRowIndex(t);
57094         var sm = this.grid.selModel;
57095             
57096         //Roo.log(rowIndex);
57097         
57098         if (sm.getSelectedCell) {
57099             // cell selection..
57100             if (!sm.getSelectedCell()) {
57101                 return false;
57102             }
57103             if (rowIndex != sm.getSelectedCell()[0]) {
57104                 return false;
57105             }
57106         
57107         }
57108         
57109         if(rowIndex !== false){
57110             
57111             // if editorgrid.. 
57112             
57113             
57114             //Roo.log([ sm.getSelectedCell() ? sm.getSelectedCell()[0] : 'NO' , rowIndex ]);
57115                
57116             //if(!sm.isSelected(rowIndex) || e.hasModifier()){
57117               //  
57118             //}
57119             if (e.hasModifier()){
57120                 sm.handleMouseDown(e, t); // non modifier buttons are handled by row select.
57121             }
57122             
57123             Roo.log("getDragData");
57124             
57125             return {
57126                 grid: this.grid,
57127                 ddel: this.ddel,
57128                 rowIndex: rowIndex,
57129                 selections:sm.getSelections ? sm.getSelections() : (
57130                     sm.getSelectedCell() ? [ this.grid.ds.getAt(sm.getSelectedCell()[0]) ] : []
57131                 )
57132             };
57133         }
57134         return false;
57135     },
57136
57137     onInitDrag : function(e){
57138         var data = this.dragData;
57139         this.ddel.innerHTML = this.grid.getDragDropText();
57140         this.proxy.update(this.ddel);
57141         // fire start drag?
57142     },
57143
57144     afterRepair : function(){
57145         this.dragging = false;
57146     },
57147
57148     getRepairXY : function(e, data){
57149         return false;
57150     },
57151
57152     onEndDrag : function(data, e){
57153         // fire end drag?
57154     },
57155
57156     onValidDrop : function(dd, e, id){
57157         // fire drag drop?
57158         this.hideProxy();
57159     },
57160
57161     beforeInvalidDrop : function(e, id){
57162
57163     }
57164 });/*
57165  * Based on:
57166  * Ext JS Library 1.1.1
57167  * Copyright(c) 2006-2007, Ext JS, LLC.
57168  *
57169  * Originally Released Under LGPL - original licence link has changed is not relivant.
57170  *
57171  * Fork - LGPL
57172  * <script type="text/javascript">
57173  */
57174  
57175
57176 /**
57177  * @class Roo.grid.ColumnModel
57178  * @extends Roo.util.Observable
57179  * This is the default implementation of a ColumnModel used by the Grid. It defines
57180  * the columns in the grid.
57181  * <br>Usage:<br>
57182  <pre><code>
57183  var colModel = new Roo.grid.ColumnModel([
57184         {header: "Ticker", width: 60, sortable: true, locked: true},
57185         {header: "Company Name", width: 150, sortable: true},
57186         {header: "Market Cap.", width: 100, sortable: true},
57187         {header: "$ Sales", width: 100, sortable: true, renderer: money},
57188         {header: "Employees", width: 100, sortable: true, resizable: false}
57189  ]);
57190  </code></pre>
57191  * <p>
57192  
57193  * The config options listed for this class are options which may appear in each
57194  * individual column definition.
57195  * <br/>RooJS Fix - column id's are not sequential but use Roo.id() - fixes bugs with layouts.
57196  * @constructor
57197  * @param {Object} config An Array of column config objects. See this class's
57198  * config objects for details.
57199 */
57200 Roo.grid.ColumnModel = function(config){
57201         /**
57202      * The config passed into the constructor
57203      */
57204     this.config = config;
57205     this.lookup = {};
57206
57207     // if no id, create one
57208     // if the column does not have a dataIndex mapping,
57209     // map it to the order it is in the config
57210     for(var i = 0, len = config.length; i < len; i++){
57211         var c = config[i];
57212         if(typeof c.dataIndex == "undefined"){
57213             c.dataIndex = i;
57214         }
57215         if(typeof c.renderer == "string"){
57216             c.renderer = Roo.util.Format[c.renderer];
57217         }
57218         if(typeof c.id == "undefined"){
57219             c.id = Roo.id();
57220         }
57221         if(c.editor && c.editor.xtype){
57222             c.editor  = Roo.factory(c.editor, Roo.grid);
57223         }
57224         if(c.editor && c.editor.isFormField){
57225             c.editor = new Roo.grid.GridEditor(c.editor);
57226         }
57227         this.lookup[c.id] = c;
57228     }
57229
57230     /**
57231      * The width of columns which have no width specified (defaults to 100)
57232      * @type Number
57233      */
57234     this.defaultWidth = 100;
57235
57236     /**
57237      * Default sortable of columns which have no sortable specified (defaults to false)
57238      * @type Boolean
57239      */
57240     this.defaultSortable = false;
57241
57242     this.addEvents({
57243         /**
57244              * @event widthchange
57245              * Fires when the width of a column changes.
57246              * @param {ColumnModel} this
57247              * @param {Number} columnIndex The column index
57248              * @param {Number} newWidth The new width
57249              */
57250             "widthchange": true,
57251         /**
57252              * @event headerchange
57253              * Fires when the text of a header changes.
57254              * @param {ColumnModel} this
57255              * @param {Number} columnIndex The column index
57256              * @param {Number} newText The new header text
57257              */
57258             "headerchange": true,
57259         /**
57260              * @event hiddenchange
57261              * Fires when a column is hidden or "unhidden".
57262              * @param {ColumnModel} this
57263              * @param {Number} columnIndex The column index
57264              * @param {Boolean} hidden true if hidden, false otherwise
57265              */
57266             "hiddenchange": true,
57267             /**
57268          * @event columnmoved
57269          * Fires when a column is moved.
57270          * @param {ColumnModel} this
57271          * @param {Number} oldIndex
57272          * @param {Number} newIndex
57273          */
57274         "columnmoved" : true,
57275         /**
57276          * @event columlockchange
57277          * Fires when a column's locked state is changed
57278          * @param {ColumnModel} this
57279          * @param {Number} colIndex
57280          * @param {Boolean} locked true if locked
57281          */
57282         "columnlockchange" : true
57283     });
57284     Roo.grid.ColumnModel.superclass.constructor.call(this);
57285 };
57286 Roo.extend(Roo.grid.ColumnModel, Roo.util.Observable, {
57287     /**
57288      * @cfg {String} header The header text to display in the Grid view.
57289      */
57290     /**
57291      * @cfg {String} dataIndex (Optional) The name of the field in the grid's {@link Roo.data.Store}'s
57292      * {@link Roo.data.Record} definition from which to draw the column's value. If not
57293      * specified, the column's index is used as an index into the Record's data Array.
57294      */
57295     /**
57296      * @cfg {Number} width (Optional) The initial width in pixels of the column. Using this
57297      * instead of {@link Roo.grid.Grid#autoSizeColumns} is more efficient.
57298      */
57299     /**
57300      * @cfg {Boolean} sortable (Optional) True if sorting is to be allowed on this column.
57301      * Defaults to the value of the {@link #defaultSortable} property.
57302      * Whether local/remote sorting is used is specified in {@link Roo.data.Store#remoteSort}.
57303      */
57304     /**
57305      * @cfg {Boolean} locked (Optional) True to lock the column in place while scrolling the Grid.  Defaults to false.
57306      */
57307     /**
57308      * @cfg {Boolean} fixed (Optional) True if the column width cannot be changed.  Defaults to false.
57309      */
57310     /**
57311      * @cfg {Boolean} resizable (Optional) False to disable column resizing. Defaults to true.
57312      */
57313     /**
57314      * @cfg {Boolean} hidden (Optional) True to hide the column. Defaults to false.
57315      */
57316     /**
57317      * @cfg {Function} renderer (Optional) A function used to generate HTML markup for a cell
57318      * given the cell's data value. See {@link #setRenderer}. If not specified, the
57319      * default renderer returns the escaped data value. If an object is returned (bootstrap only)
57320      * then it is treated as a Roo Component object instance, and it is rendered after the initial row is rendered
57321      */
57322        /**
57323      * @cfg {Roo.grid.GridEditor} editor (Optional) For grid editors - returns the grid editor 
57324      */
57325     /**
57326      * @cfg {String} align (Optional) Set the CSS text-align property of the column.  Defaults to undefined.
57327      */
57328     /**
57329      * @cfg {String} valign (Optional) Set the CSS vertical-align property of the column (eg. middle, top, bottom etc).  Defaults to undefined.
57330      */
57331     /**
57332      * @cfg {String} cursor (Optional)
57333      */
57334     /**
57335      * @cfg {String} tooltip (Optional)
57336      */
57337     /**
57338      * @cfg {Number} xs (Optional)
57339      */
57340     /**
57341      * @cfg {Number} sm (Optional)
57342      */
57343     /**
57344      * @cfg {Number} md (Optional)
57345      */
57346     /**
57347      * @cfg {Number} lg (Optional)
57348      */
57349     /**
57350      * Returns the id of the column at the specified index.
57351      * @param {Number} index The column index
57352      * @return {String} the id
57353      */
57354     getColumnId : function(index){
57355         return this.config[index].id;
57356     },
57357
57358     /**
57359      * Returns the column for a specified id.
57360      * @param {String} id The column id
57361      * @return {Object} the column
57362      */
57363     getColumnById : function(id){
57364         return this.lookup[id];
57365     },
57366
57367     
57368     /**
57369      * Returns the column for a specified dataIndex.
57370      * @param {String} dataIndex The column dataIndex
57371      * @return {Object|Boolean} the column or false if not found
57372      */
57373     getColumnByDataIndex: function(dataIndex){
57374         var index = this.findColumnIndex(dataIndex);
57375         return index > -1 ? this.config[index] : false;
57376     },
57377     
57378     /**
57379      * Returns the index for a specified column id.
57380      * @param {String} id The column id
57381      * @return {Number} the index, or -1 if not found
57382      */
57383     getIndexById : function(id){
57384         for(var i = 0, len = this.config.length; i < len; i++){
57385             if(this.config[i].id == id){
57386                 return i;
57387             }
57388         }
57389         return -1;
57390     },
57391     
57392     /**
57393      * Returns the index for a specified column dataIndex.
57394      * @param {String} dataIndex The column dataIndex
57395      * @return {Number} the index, or -1 if not found
57396      */
57397     
57398     findColumnIndex : function(dataIndex){
57399         for(var i = 0, len = this.config.length; i < len; i++){
57400             if(this.config[i].dataIndex == dataIndex){
57401                 return i;
57402             }
57403         }
57404         return -1;
57405     },
57406     
57407     
57408     moveColumn : function(oldIndex, newIndex){
57409         var c = this.config[oldIndex];
57410         this.config.splice(oldIndex, 1);
57411         this.config.splice(newIndex, 0, c);
57412         this.dataMap = null;
57413         this.fireEvent("columnmoved", this, oldIndex, newIndex);
57414     },
57415
57416     isLocked : function(colIndex){
57417         return this.config[colIndex].locked === true;
57418     },
57419
57420     setLocked : function(colIndex, value, suppressEvent){
57421         if(this.isLocked(colIndex) == value){
57422             return;
57423         }
57424         this.config[colIndex].locked = value;
57425         if(!suppressEvent){
57426             this.fireEvent("columnlockchange", this, colIndex, value);
57427         }
57428     },
57429
57430     getTotalLockedWidth : function(){
57431         var totalWidth = 0;
57432         for(var i = 0; i < this.config.length; i++){
57433             if(this.isLocked(i) && !this.isHidden(i)){
57434                 this.totalWidth += this.getColumnWidth(i);
57435             }
57436         }
57437         return totalWidth;
57438     },
57439
57440     getLockedCount : function(){
57441         for(var i = 0, len = this.config.length; i < len; i++){
57442             if(!this.isLocked(i)){
57443                 return i;
57444             }
57445         }
57446         
57447         return this.config.length;
57448     },
57449
57450     /**
57451      * Returns the number of columns.
57452      * @return {Number}
57453      */
57454     getColumnCount : function(visibleOnly){
57455         if(visibleOnly === true){
57456             var c = 0;
57457             for(var i = 0, len = this.config.length; i < len; i++){
57458                 if(!this.isHidden(i)){
57459                     c++;
57460                 }
57461             }
57462             return c;
57463         }
57464         return this.config.length;
57465     },
57466
57467     /**
57468      * Returns the column configs that return true by the passed function that is called with (columnConfig, index)
57469      * @param {Function} fn
57470      * @param {Object} scope (optional)
57471      * @return {Array} result
57472      */
57473     getColumnsBy : function(fn, scope){
57474         var r = [];
57475         for(var i = 0, len = this.config.length; i < len; i++){
57476             var c = this.config[i];
57477             if(fn.call(scope||this, c, i) === true){
57478                 r[r.length] = c;
57479             }
57480         }
57481         return r;
57482     },
57483
57484     /**
57485      * Returns true if the specified column is sortable.
57486      * @param {Number} col The column index
57487      * @return {Boolean}
57488      */
57489     isSortable : function(col){
57490         if(typeof this.config[col].sortable == "undefined"){
57491             return this.defaultSortable;
57492         }
57493         return this.config[col].sortable;
57494     },
57495
57496     /**
57497      * Returns the rendering (formatting) function defined for the column.
57498      * @param {Number} col The column index.
57499      * @return {Function} The function used to render the cell. See {@link #setRenderer}.
57500      */
57501     getRenderer : function(col){
57502         if(!this.config[col].renderer){
57503             return Roo.grid.ColumnModel.defaultRenderer;
57504         }
57505         return this.config[col].renderer;
57506     },
57507
57508     /**
57509      * Sets the rendering (formatting) function for a column.
57510      * @param {Number} col The column index
57511      * @param {Function} fn The function to use to process the cell's raw data
57512      * to return HTML markup for the grid view. The render function is called with
57513      * the following parameters:<ul>
57514      * <li>Data value.</li>
57515      * <li>Cell metadata. An object in which you may set the following attributes:<ul>
57516      * <li>css A CSS style string to apply to the table cell.</li>
57517      * <li>attr An HTML attribute definition string to apply to the data container element <i>within</i> the table cell.</li></ul>
57518      * <li>The {@link Roo.data.Record} from which the data was extracted.</li>
57519      * <li>Row index</li>
57520      * <li>Column index</li>
57521      * <li>The {@link Roo.data.Store} object from which the Record was extracted</li></ul>
57522      */
57523     setRenderer : function(col, fn){
57524         this.config[col].renderer = fn;
57525     },
57526
57527     /**
57528      * Returns the width for the specified column.
57529      * @param {Number} col The column index
57530      * @return {Number}
57531      */
57532     getColumnWidth : function(col){
57533         return this.config[col].width * 1 || this.defaultWidth;
57534     },
57535
57536     /**
57537      * Sets the width for a column.
57538      * @param {Number} col The column index
57539      * @param {Number} width The new width
57540      */
57541     setColumnWidth : function(col, width, suppressEvent){
57542         this.config[col].width = width;
57543         this.totalWidth = null;
57544         if(!suppressEvent){
57545              this.fireEvent("widthchange", this, col, width);
57546         }
57547     },
57548
57549     /**
57550      * Returns the total width of all columns.
57551      * @param {Boolean} includeHidden True to include hidden column widths
57552      * @return {Number}
57553      */
57554     getTotalWidth : function(includeHidden){
57555         if(!this.totalWidth){
57556             this.totalWidth = 0;
57557             for(var i = 0, len = this.config.length; i < len; i++){
57558                 if(includeHidden || !this.isHidden(i)){
57559                     this.totalWidth += this.getColumnWidth(i);
57560                 }
57561             }
57562         }
57563         return this.totalWidth;
57564     },
57565
57566     /**
57567      * Returns the header for the specified column.
57568      * @param {Number} col The column index
57569      * @return {String}
57570      */
57571     getColumnHeader : function(col){
57572         return this.config[col].header;
57573     },
57574
57575     /**
57576      * Sets the header for a column.
57577      * @param {Number} col The column index
57578      * @param {String} header The new header
57579      */
57580     setColumnHeader : function(col, header){
57581         this.config[col].header = header;
57582         this.fireEvent("headerchange", this, col, header);
57583     },
57584
57585     /**
57586      * Returns the tooltip for the specified column.
57587      * @param {Number} col The column index
57588      * @return {String}
57589      */
57590     getColumnTooltip : function(col){
57591             return this.config[col].tooltip;
57592     },
57593     /**
57594      * Sets the tooltip for a column.
57595      * @param {Number} col The column index
57596      * @param {String} tooltip The new tooltip
57597      */
57598     setColumnTooltip : function(col, tooltip){
57599             this.config[col].tooltip = tooltip;
57600     },
57601
57602     /**
57603      * Returns the dataIndex for the specified column.
57604      * @param {Number} col The column index
57605      * @return {Number}
57606      */
57607     getDataIndex : function(col){
57608         return this.config[col].dataIndex;
57609     },
57610
57611     /**
57612      * Sets the dataIndex for a column.
57613      * @param {Number} col The column index
57614      * @param {Number} dataIndex The new dataIndex
57615      */
57616     setDataIndex : function(col, dataIndex){
57617         this.config[col].dataIndex = dataIndex;
57618     },
57619
57620     
57621     
57622     /**
57623      * Returns true if the cell is editable.
57624      * @param {Number} colIndex The column index
57625      * @param {Number} rowIndex The row index - this is nto actually used..?
57626      * @return {Boolean}
57627      */
57628     isCellEditable : function(colIndex, rowIndex){
57629         return (this.config[colIndex].editable || (typeof this.config[colIndex].editable == "undefined" && this.config[colIndex].editor)) ? true : false;
57630     },
57631
57632     /**
57633      * Returns the editor defined for the cell/column.
57634      * return false or null to disable editing.
57635      * @param {Number} colIndex The column index
57636      * @param {Number} rowIndex The row index
57637      * @return {Object}
57638      */
57639     getCellEditor : function(colIndex, rowIndex){
57640         return this.config[colIndex].editor;
57641     },
57642
57643     /**
57644      * Sets if a column is editable.
57645      * @param {Number} col The column index
57646      * @param {Boolean} editable True if the column is editable
57647      */
57648     setEditable : function(col, editable){
57649         this.config[col].editable = editable;
57650     },
57651
57652
57653     /**
57654      * Returns true if the column is hidden.
57655      * @param {Number} colIndex The column index
57656      * @return {Boolean}
57657      */
57658     isHidden : function(colIndex){
57659         return this.config[colIndex].hidden;
57660     },
57661
57662
57663     /**
57664      * Returns true if the column width cannot be changed
57665      */
57666     isFixed : function(colIndex){
57667         return this.config[colIndex].fixed;
57668     },
57669
57670     /**
57671      * Returns true if the column can be resized
57672      * @return {Boolean}
57673      */
57674     isResizable : function(colIndex){
57675         return colIndex >= 0 && this.config[colIndex].resizable !== false && this.config[colIndex].fixed !== true;
57676     },
57677     /**
57678      * Sets if a column is hidden.
57679      * @param {Number} colIndex The column index
57680      * @param {Boolean} hidden True if the column is hidden
57681      */
57682     setHidden : function(colIndex, hidden){
57683         this.config[colIndex].hidden = hidden;
57684         this.totalWidth = null;
57685         this.fireEvent("hiddenchange", this, colIndex, hidden);
57686     },
57687
57688     /**
57689      * Sets the editor for a column.
57690      * @param {Number} col The column index
57691      * @param {Object} editor The editor object
57692      */
57693     setEditor : function(col, editor){
57694         this.config[col].editor = editor;
57695     }
57696 });
57697
57698 Roo.grid.ColumnModel.defaultRenderer = function(value)
57699 {
57700     if(typeof value == "object") {
57701         return value;
57702     }
57703         if(typeof value == "string" && value.length < 1){
57704             return "&#160;";
57705         }
57706     
57707         return String.format("{0}", value);
57708 };
57709
57710 // Alias for backwards compatibility
57711 Roo.grid.DefaultColumnModel = Roo.grid.ColumnModel;
57712 /*
57713  * Based on:
57714  * Ext JS Library 1.1.1
57715  * Copyright(c) 2006-2007, Ext JS, LLC.
57716  *
57717  * Originally Released Under LGPL - original licence link has changed is not relivant.
57718  *
57719  * Fork - LGPL
57720  * <script type="text/javascript">
57721  */
57722
57723 /**
57724  * @class Roo.grid.AbstractSelectionModel
57725  * @extends Roo.util.Observable
57726  * Abstract base class for grid SelectionModels.  It provides the interface that should be
57727  * implemented by descendant classes.  This class should not be directly instantiated.
57728  * @constructor
57729  */
57730 Roo.grid.AbstractSelectionModel = function(){
57731     this.locked = false;
57732     Roo.grid.AbstractSelectionModel.superclass.constructor.call(this);
57733 };
57734
57735 Roo.extend(Roo.grid.AbstractSelectionModel, Roo.util.Observable,  {
57736     /** @ignore Called by the grid automatically. Do not call directly. */
57737     init : function(grid){
57738         this.grid = grid;
57739         this.initEvents();
57740     },
57741
57742     /**
57743      * Locks the selections.
57744      */
57745     lock : function(){
57746         this.locked = true;
57747     },
57748
57749     /**
57750      * Unlocks the selections.
57751      */
57752     unlock : function(){
57753         this.locked = false;
57754     },
57755
57756     /**
57757      * Returns true if the selections are locked.
57758      * @return {Boolean}
57759      */
57760     isLocked : function(){
57761         return this.locked;
57762     }
57763 });/*
57764  * Based on:
57765  * Ext JS Library 1.1.1
57766  * Copyright(c) 2006-2007, Ext JS, LLC.
57767  *
57768  * Originally Released Under LGPL - original licence link has changed is not relivant.
57769  *
57770  * Fork - LGPL
57771  * <script type="text/javascript">
57772  */
57773 /**
57774  * @extends Roo.grid.AbstractSelectionModel
57775  * @class Roo.grid.RowSelectionModel
57776  * The default SelectionModel used by {@link Roo.grid.Grid}.
57777  * It supports multiple selections and keyboard selection/navigation. 
57778  * @constructor
57779  * @param {Object} config
57780  */
57781 Roo.grid.RowSelectionModel = function(config){
57782     Roo.apply(this, config);
57783     this.selections = new Roo.util.MixedCollection(false, function(o){
57784         return o.id;
57785     });
57786
57787     this.last = false;
57788     this.lastActive = false;
57789
57790     this.addEvents({
57791         /**
57792              * @event selectionchange
57793              * Fires when the selection changes
57794              * @param {SelectionModel} this
57795              */
57796             "selectionchange" : true,
57797         /**
57798              * @event afterselectionchange
57799              * Fires after the selection changes (eg. by key press or clicking)
57800              * @param {SelectionModel} this
57801              */
57802             "afterselectionchange" : true,
57803         /**
57804              * @event beforerowselect
57805              * Fires when a row is selected being selected, return false to cancel.
57806              * @param {SelectionModel} this
57807              * @param {Number} rowIndex The selected index
57808              * @param {Boolean} keepExisting False if other selections will be cleared
57809              */
57810             "beforerowselect" : true,
57811         /**
57812              * @event rowselect
57813              * Fires when a row is selected.
57814              * @param {SelectionModel} this
57815              * @param {Number} rowIndex The selected index
57816              * @param {Roo.data.Record} r The record
57817              */
57818             "rowselect" : true,
57819         /**
57820              * @event rowdeselect
57821              * Fires when a row is deselected.
57822              * @param {SelectionModel} this
57823              * @param {Number} rowIndex The selected index
57824              */
57825         "rowdeselect" : true
57826     });
57827     Roo.grid.RowSelectionModel.superclass.constructor.call(this);
57828     this.locked = false;
57829 };
57830
57831 Roo.extend(Roo.grid.RowSelectionModel, Roo.grid.AbstractSelectionModel,  {
57832     /**
57833      * @cfg {Boolean} singleSelect
57834      * True to allow selection of only one row at a time (defaults to false)
57835      */
57836     singleSelect : false,
57837
57838     // private
57839     initEvents : function(){
57840
57841         if(!this.grid.enableDragDrop && !this.grid.enableDrag){
57842             this.grid.on("mousedown", this.handleMouseDown, this);
57843         }else{ // allow click to work like normal
57844             this.grid.on("rowclick", this.handleDragableRowClick, this);
57845         }
57846
57847         this.rowNav = new Roo.KeyNav(this.grid.getGridEl(), {
57848             "up" : function(e){
57849                 if(!e.shiftKey){
57850                     this.selectPrevious(e.shiftKey);
57851                 }else if(this.last !== false && this.lastActive !== false){
57852                     var last = this.last;
57853                     this.selectRange(this.last,  this.lastActive-1);
57854                     this.grid.getView().focusRow(this.lastActive);
57855                     if(last !== false){
57856                         this.last = last;
57857                     }
57858                 }else{
57859                     this.selectFirstRow();
57860                 }
57861                 this.fireEvent("afterselectionchange", this);
57862             },
57863             "down" : function(e){
57864                 if(!e.shiftKey){
57865                     this.selectNext(e.shiftKey);
57866                 }else if(this.last !== false && this.lastActive !== false){
57867                     var last = this.last;
57868                     this.selectRange(this.last,  this.lastActive+1);
57869                     this.grid.getView().focusRow(this.lastActive);
57870                     if(last !== false){
57871                         this.last = last;
57872                     }
57873                 }else{
57874                     this.selectFirstRow();
57875                 }
57876                 this.fireEvent("afterselectionchange", this);
57877             },
57878             scope: this
57879         });
57880
57881         var view = this.grid.view;
57882         view.on("refresh", this.onRefresh, this);
57883         view.on("rowupdated", this.onRowUpdated, this);
57884         view.on("rowremoved", this.onRemove, this);
57885     },
57886
57887     // private
57888     onRefresh : function(){
57889         var ds = this.grid.dataSource, i, v = this.grid.view;
57890         var s = this.selections;
57891         s.each(function(r){
57892             if((i = ds.indexOfId(r.id)) != -1){
57893                 v.onRowSelect(i);
57894                 s.add(ds.getAt(i)); // updating the selection relate data
57895             }else{
57896                 s.remove(r);
57897             }
57898         });
57899     },
57900
57901     // private
57902     onRemove : function(v, index, r){
57903         this.selections.remove(r);
57904     },
57905
57906     // private
57907     onRowUpdated : function(v, index, r){
57908         if(this.isSelected(r)){
57909             v.onRowSelect(index);
57910         }
57911     },
57912
57913     /**
57914      * Select records.
57915      * @param {Array} records The records to select
57916      * @param {Boolean} keepExisting (optional) True to keep existing selections
57917      */
57918     selectRecords : function(records, keepExisting){
57919         if(!keepExisting){
57920             this.clearSelections();
57921         }
57922         var ds = this.grid.dataSource;
57923         for(var i = 0, len = records.length; i < len; i++){
57924             this.selectRow(ds.indexOf(records[i]), true);
57925         }
57926     },
57927
57928     /**
57929      * Gets the number of selected rows.
57930      * @return {Number}
57931      */
57932     getCount : function(){
57933         return this.selections.length;
57934     },
57935
57936     /**
57937      * Selects the first row in the grid.
57938      */
57939     selectFirstRow : function(){
57940         this.selectRow(0);
57941     },
57942
57943     /**
57944      * Select the last row.
57945      * @param {Boolean} keepExisting (optional) True to keep existing selections
57946      */
57947     selectLastRow : function(keepExisting){
57948         this.selectRow(this.grid.dataSource.getCount() - 1, keepExisting);
57949     },
57950
57951     /**
57952      * Selects the row immediately following the last selected row.
57953      * @param {Boolean} keepExisting (optional) True to keep existing selections
57954      */
57955     selectNext : function(keepExisting){
57956         if(this.last !== false && (this.last+1) < this.grid.dataSource.getCount()){
57957             this.selectRow(this.last+1, keepExisting);
57958             this.grid.getView().focusRow(this.last);
57959         }
57960     },
57961
57962     /**
57963      * Selects the row that precedes the last selected row.
57964      * @param {Boolean} keepExisting (optional) True to keep existing selections
57965      */
57966     selectPrevious : function(keepExisting){
57967         if(this.last){
57968             this.selectRow(this.last-1, keepExisting);
57969             this.grid.getView().focusRow(this.last);
57970         }
57971     },
57972
57973     /**
57974      * Returns the selected records
57975      * @return {Array} Array of selected records
57976      */
57977     getSelections : function(){
57978         return [].concat(this.selections.items);
57979     },
57980
57981     /**
57982      * Returns the first selected record.
57983      * @return {Record}
57984      */
57985     getSelected : function(){
57986         return this.selections.itemAt(0);
57987     },
57988
57989
57990     /**
57991      * Clears all selections.
57992      */
57993     clearSelections : function(fast){
57994         if(this.locked) {
57995             return;
57996         }
57997         if(fast !== true){
57998             var ds = this.grid.dataSource;
57999             var s = this.selections;
58000             s.each(function(r){
58001                 this.deselectRow(ds.indexOfId(r.id));
58002             }, this);
58003             s.clear();
58004         }else{
58005             this.selections.clear();
58006         }
58007         this.last = false;
58008     },
58009
58010
58011     /**
58012      * Selects all rows.
58013      */
58014     selectAll : function(){
58015         if(this.locked) {
58016             return;
58017         }
58018         this.selections.clear();
58019         for(var i = 0, len = this.grid.dataSource.getCount(); i < len; i++){
58020             this.selectRow(i, true);
58021         }
58022     },
58023
58024     /**
58025      * Returns True if there is a selection.
58026      * @return {Boolean}
58027      */
58028     hasSelection : function(){
58029         return this.selections.length > 0;
58030     },
58031
58032     /**
58033      * Returns True if the specified row is selected.
58034      * @param {Number/Record} record The record or index of the record to check
58035      * @return {Boolean}
58036      */
58037     isSelected : function(index){
58038         var r = typeof index == "number" ? this.grid.dataSource.getAt(index) : index;
58039         return (r && this.selections.key(r.id) ? true : false);
58040     },
58041
58042     /**
58043      * Returns True if the specified record id is selected.
58044      * @param {String} id The id of record to check
58045      * @return {Boolean}
58046      */
58047     isIdSelected : function(id){
58048         return (this.selections.key(id) ? true : false);
58049     },
58050
58051     // private
58052     handleMouseDown : function(e, t){
58053         var view = this.grid.getView(), rowIndex;
58054         if(this.isLocked() || (rowIndex = view.findRowIndex(t)) === false){
58055             return;
58056         };
58057         if(e.shiftKey && this.last !== false){
58058             var last = this.last;
58059             this.selectRange(last, rowIndex, e.ctrlKey);
58060             this.last = last; // reset the last
58061             view.focusRow(rowIndex);
58062         }else{
58063             var isSelected = this.isSelected(rowIndex);
58064             if(e.button !== 0 && isSelected){
58065                 view.focusRow(rowIndex);
58066             }else if(e.ctrlKey && isSelected){
58067                 this.deselectRow(rowIndex);
58068             }else if(!isSelected){
58069                 this.selectRow(rowIndex, e.button === 0 && (e.ctrlKey || e.shiftKey));
58070                 view.focusRow(rowIndex);
58071             }
58072         }
58073         this.fireEvent("afterselectionchange", this);
58074     },
58075     // private
58076     handleDragableRowClick :  function(grid, rowIndex, e) 
58077     {
58078         if(e.button === 0 && !e.shiftKey && !e.ctrlKey) {
58079             this.selectRow(rowIndex, false);
58080             grid.view.focusRow(rowIndex);
58081              this.fireEvent("afterselectionchange", this);
58082         }
58083     },
58084     
58085     /**
58086      * Selects multiple rows.
58087      * @param {Array} rows Array of the indexes of the row to select
58088      * @param {Boolean} keepExisting (optional) True to keep existing selections
58089      */
58090     selectRows : function(rows, keepExisting){
58091         if(!keepExisting){
58092             this.clearSelections();
58093         }
58094         for(var i = 0, len = rows.length; i < len; i++){
58095             this.selectRow(rows[i], true);
58096         }
58097     },
58098
58099     /**
58100      * Selects a range of rows. All rows in between startRow and endRow are also selected.
58101      * @param {Number} startRow The index of the first row in the range
58102      * @param {Number} endRow The index of the last row in the range
58103      * @param {Boolean} keepExisting (optional) True to retain existing selections
58104      */
58105     selectRange : function(startRow, endRow, keepExisting){
58106         if(this.locked) {
58107             return;
58108         }
58109         if(!keepExisting){
58110             this.clearSelections();
58111         }
58112         if(startRow <= endRow){
58113             for(var i = startRow; i <= endRow; i++){
58114                 this.selectRow(i, true);
58115             }
58116         }else{
58117             for(var i = startRow; i >= endRow; i--){
58118                 this.selectRow(i, true);
58119             }
58120         }
58121     },
58122
58123     /**
58124      * Deselects a range of rows. All rows in between startRow and endRow are also deselected.
58125      * @param {Number} startRow The index of the first row in the range
58126      * @param {Number} endRow The index of the last row in the range
58127      */
58128     deselectRange : function(startRow, endRow, preventViewNotify){
58129         if(this.locked) {
58130             return;
58131         }
58132         for(var i = startRow; i <= endRow; i++){
58133             this.deselectRow(i, preventViewNotify);
58134         }
58135     },
58136
58137     /**
58138      * Selects a row.
58139      * @param {Number} row The index of the row to select
58140      * @param {Boolean} keepExisting (optional) True to keep existing selections
58141      */
58142     selectRow : function(index, keepExisting, preventViewNotify){
58143         if(this.locked || (index < 0 || index >= this.grid.dataSource.getCount())) {
58144             return;
58145         }
58146         if(this.fireEvent("beforerowselect", this, index, keepExisting) !== false){
58147             if(!keepExisting || this.singleSelect){
58148                 this.clearSelections();
58149             }
58150             var r = this.grid.dataSource.getAt(index);
58151             this.selections.add(r);
58152             this.last = this.lastActive = index;
58153             if(!preventViewNotify){
58154                 this.grid.getView().onRowSelect(index);
58155             }
58156             this.fireEvent("rowselect", this, index, r);
58157             this.fireEvent("selectionchange", this);
58158         }
58159     },
58160
58161     /**
58162      * Deselects a row.
58163      * @param {Number} row The index of the row to deselect
58164      */
58165     deselectRow : function(index, preventViewNotify){
58166         if(this.locked) {
58167             return;
58168         }
58169         if(this.last == index){
58170             this.last = false;
58171         }
58172         if(this.lastActive == index){
58173             this.lastActive = false;
58174         }
58175         var r = this.grid.dataSource.getAt(index);
58176         this.selections.remove(r);
58177         if(!preventViewNotify){
58178             this.grid.getView().onRowDeselect(index);
58179         }
58180         this.fireEvent("rowdeselect", this, index);
58181         this.fireEvent("selectionchange", this);
58182     },
58183
58184     // private
58185     restoreLast : function(){
58186         if(this._last){
58187             this.last = this._last;
58188         }
58189     },
58190
58191     // private
58192     acceptsNav : function(row, col, cm){
58193         return !cm.isHidden(col) && cm.isCellEditable(col, row);
58194     },
58195
58196     // private
58197     onEditorKey : function(field, e){
58198         var k = e.getKey(), newCell, g = this.grid, ed = g.activeEditor;
58199         if(k == e.TAB){
58200             e.stopEvent();
58201             ed.completeEdit();
58202             if(e.shiftKey){
58203                 newCell = g.walkCells(ed.row, ed.col-1, -1, this.acceptsNav, this);
58204             }else{
58205                 newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
58206             }
58207         }else if(k == e.ENTER && !e.ctrlKey){
58208             e.stopEvent();
58209             ed.completeEdit();
58210             if(e.shiftKey){
58211                 newCell = g.walkCells(ed.row-1, ed.col, -1, this.acceptsNav, this);
58212             }else{
58213                 newCell = g.walkCells(ed.row+1, ed.col, 1, this.acceptsNav, this);
58214             }
58215         }else if(k == e.ESC){
58216             ed.cancelEdit();
58217         }
58218         if(newCell){
58219             g.startEditing(newCell[0], newCell[1]);
58220         }
58221     }
58222 });/*
58223  * Based on:
58224  * Ext JS Library 1.1.1
58225  * Copyright(c) 2006-2007, Ext JS, LLC.
58226  *
58227  * Originally Released Under LGPL - original licence link has changed is not relivant.
58228  *
58229  * Fork - LGPL
58230  * <script type="text/javascript">
58231  */
58232 /**
58233  * @class Roo.grid.CellSelectionModel
58234  * @extends Roo.grid.AbstractSelectionModel
58235  * This class provides the basic implementation for cell selection in a grid.
58236  * @constructor
58237  * @param {Object} config The object containing the configuration of this model.
58238  * @cfg {Boolean} enter_is_tab Enter behaves the same as tab. (eg. goes to next cell) default: false
58239  */
58240 Roo.grid.CellSelectionModel = function(config){
58241     Roo.apply(this, config);
58242
58243     this.selection = null;
58244
58245     this.addEvents({
58246         /**
58247              * @event beforerowselect
58248              * Fires before a cell is selected.
58249              * @param {SelectionModel} this
58250              * @param {Number} rowIndex The selected row index
58251              * @param {Number} colIndex The selected cell index
58252              */
58253             "beforecellselect" : true,
58254         /**
58255              * @event cellselect
58256              * Fires when a cell is selected.
58257              * @param {SelectionModel} this
58258              * @param {Number} rowIndex The selected row index
58259              * @param {Number} colIndex The selected cell index
58260              */
58261             "cellselect" : true,
58262         /**
58263              * @event selectionchange
58264              * Fires when the active selection changes.
58265              * @param {SelectionModel} this
58266              * @param {Object} selection null for no selection or an object (o) with two properties
58267                 <ul>
58268                 <li>o.record: the record object for the row the selection is in</li>
58269                 <li>o.cell: An array of [rowIndex, columnIndex]</li>
58270                 </ul>
58271              */
58272             "selectionchange" : true,
58273         /**
58274              * @event tabend
58275              * Fires when the tab (or enter) was pressed on the last editable cell
58276              * You can use this to trigger add new row.
58277              * @param {SelectionModel} this
58278              */
58279             "tabend" : true,
58280          /**
58281              * @event beforeeditnext
58282              * Fires before the next editable sell is made active
58283              * You can use this to skip to another cell or fire the tabend
58284              *    if you set cell to false
58285              * @param {Object} eventdata object : { cell : [ row, col ] } 
58286              */
58287             "beforeeditnext" : true
58288     });
58289     Roo.grid.CellSelectionModel.superclass.constructor.call(this);
58290 };
58291
58292 Roo.extend(Roo.grid.CellSelectionModel, Roo.grid.AbstractSelectionModel,  {
58293     
58294     enter_is_tab: false,
58295
58296     /** @ignore */
58297     initEvents : function(){
58298         this.grid.on("mousedown", this.handleMouseDown, this);
58299         this.grid.getGridEl().on(Roo.isIE ? "keydown" : "keypress", this.handleKeyDown, this);
58300         var view = this.grid.view;
58301         view.on("refresh", this.onViewChange, this);
58302         view.on("rowupdated", this.onRowUpdated, this);
58303         view.on("beforerowremoved", this.clearSelections, this);
58304         view.on("beforerowsinserted", this.clearSelections, this);
58305         if(this.grid.isEditor){
58306             this.grid.on("beforeedit", this.beforeEdit,  this);
58307         }
58308     },
58309
58310         //private
58311     beforeEdit : function(e){
58312         this.select(e.row, e.column, false, true, e.record);
58313     },
58314
58315         //private
58316     onRowUpdated : function(v, index, r){
58317         if(this.selection && this.selection.record == r){
58318             v.onCellSelect(index, this.selection.cell[1]);
58319         }
58320     },
58321
58322         //private
58323     onViewChange : function(){
58324         this.clearSelections(true);
58325     },
58326
58327         /**
58328          * Returns the currently selected cell,.
58329          * @return {Array} The selected cell (row, column) or null if none selected.
58330          */
58331     getSelectedCell : function(){
58332         return this.selection ? this.selection.cell : null;
58333     },
58334
58335     /**
58336      * Clears all selections.
58337      * @param {Boolean} true to prevent the gridview from being notified about the change.
58338      */
58339     clearSelections : function(preventNotify){
58340         var s = this.selection;
58341         if(s){
58342             if(preventNotify !== true){
58343                 this.grid.view.onCellDeselect(s.cell[0], s.cell[1]);
58344             }
58345             this.selection = null;
58346             this.fireEvent("selectionchange", this, null);
58347         }
58348     },
58349
58350     /**
58351      * Returns true if there is a selection.
58352      * @return {Boolean}
58353      */
58354     hasSelection : function(){
58355         return this.selection ? true : false;
58356     },
58357
58358     /** @ignore */
58359     handleMouseDown : function(e, t){
58360         var v = this.grid.getView();
58361         if(this.isLocked()){
58362             return;
58363         };
58364         var row = v.findRowIndex(t);
58365         var cell = v.findCellIndex(t);
58366         if(row !== false && cell !== false){
58367             this.select(row, cell);
58368         }
58369     },
58370
58371     /**
58372      * Selects a cell.
58373      * @param {Number} rowIndex
58374      * @param {Number} collIndex
58375      */
58376     select : function(rowIndex, colIndex, preventViewNotify, preventFocus, /*internal*/ r){
58377         if(this.fireEvent("beforecellselect", this, rowIndex, colIndex) !== false){
58378             this.clearSelections();
58379             r = r || this.grid.dataSource.getAt(rowIndex);
58380             this.selection = {
58381                 record : r,
58382                 cell : [rowIndex, colIndex]
58383             };
58384             if(!preventViewNotify){
58385                 var v = this.grid.getView();
58386                 v.onCellSelect(rowIndex, colIndex);
58387                 if(preventFocus !== true){
58388                     v.focusCell(rowIndex, colIndex);
58389                 }
58390             }
58391             this.fireEvent("cellselect", this, rowIndex, colIndex);
58392             this.fireEvent("selectionchange", this, this.selection);
58393         }
58394     },
58395
58396         //private
58397     isSelectable : function(rowIndex, colIndex, cm){
58398         return !cm.isHidden(colIndex);
58399     },
58400
58401     /** @ignore */
58402     handleKeyDown : function(e){
58403         //Roo.log('Cell Sel Model handleKeyDown');
58404         if(!e.isNavKeyPress()){
58405             return;
58406         }
58407         var g = this.grid, s = this.selection;
58408         if(!s){
58409             e.stopEvent();
58410             var cell = g.walkCells(0, 0, 1, this.isSelectable,  this);
58411             if(cell){
58412                 this.select(cell[0], cell[1]);
58413             }
58414             return;
58415         }
58416         var sm = this;
58417         var walk = function(row, col, step){
58418             return g.walkCells(row, col, step, sm.isSelectable,  sm);
58419         };
58420         var k = e.getKey(), r = s.cell[0], c = s.cell[1];
58421         var newCell;
58422
58423       
58424
58425         switch(k){
58426             case e.TAB:
58427                 // handled by onEditorKey
58428                 if (g.isEditor && g.editing) {
58429                     return;
58430                 }
58431                 if(e.shiftKey) {
58432                     newCell = walk(r, c-1, -1);
58433                 } else {
58434                     newCell = walk(r, c+1, 1);
58435                 }
58436                 break;
58437             
58438             case e.DOWN:
58439                newCell = walk(r+1, c, 1);
58440                 break;
58441             
58442             case e.UP:
58443                 newCell = walk(r-1, c, -1);
58444                 break;
58445             
58446             case e.RIGHT:
58447                 newCell = walk(r, c+1, 1);
58448                 break;
58449             
58450             case e.LEFT:
58451                 newCell = walk(r, c-1, -1);
58452                 break;
58453             
58454             case e.ENTER:
58455                 
58456                 if(g.isEditor && !g.editing){
58457                    g.startEditing(r, c);
58458                    e.stopEvent();
58459                    return;
58460                 }
58461                 
58462                 
58463              break;
58464         };
58465         if(newCell){
58466             this.select(newCell[0], newCell[1]);
58467             e.stopEvent();
58468             
58469         }
58470     },
58471
58472     acceptsNav : function(row, col, cm){
58473         return !cm.isHidden(col) && cm.isCellEditable(col, row);
58474     },
58475     /**
58476      * Selects a cell.
58477      * @param {Number} field (not used) - as it's normally used as a listener
58478      * @param {Number} e - event - fake it by using
58479      *
58480      * var e = Roo.EventObjectImpl.prototype;
58481      * e.keyCode = e.TAB
58482      *
58483      * 
58484      */
58485     onEditorKey : function(field, e){
58486         
58487         var k = e.getKey(),
58488             newCell,
58489             g = this.grid,
58490             ed = g.activeEditor,
58491             forward = false;
58492         ///Roo.log('onEditorKey' + k);
58493         
58494         
58495         if (this.enter_is_tab && k == e.ENTER) {
58496             k = e.TAB;
58497         }
58498         
58499         if(k == e.TAB){
58500             if(e.shiftKey){
58501                 newCell = g.walkCells(ed.row, ed.col-1, -1, this.acceptsNav, this);
58502             }else{
58503                 newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
58504                 forward = true;
58505             }
58506             
58507             e.stopEvent();
58508             
58509         } else if(k == e.ENTER &&  !e.ctrlKey){
58510             ed.completeEdit();
58511             e.stopEvent();
58512             newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
58513         
58514                 } else if(k == e.ESC){
58515             ed.cancelEdit();
58516         }
58517                 
58518         if (newCell) {
58519             var ecall = { cell : newCell, forward : forward };
58520             this.fireEvent('beforeeditnext', ecall );
58521             newCell = ecall.cell;
58522                         forward = ecall.forward;
58523         }
58524                 
58525         if(newCell){
58526             //Roo.log('next cell after edit');
58527             g.startEditing.defer(100, g, [newCell[0], newCell[1]]);
58528         } else if (forward) {
58529             // tabbed past last
58530             this.fireEvent.defer(100, this, ['tabend',this]);
58531         }
58532     }
58533 });/*
58534  * Based on:
58535  * Ext JS Library 1.1.1
58536  * Copyright(c) 2006-2007, Ext JS, LLC.
58537  *
58538  * Originally Released Under LGPL - original licence link has changed is not relivant.
58539  *
58540  * Fork - LGPL
58541  * <script type="text/javascript">
58542  */
58543  
58544 /**
58545  * @class Roo.grid.EditorGrid
58546  * @extends Roo.grid.Grid
58547  * Class for creating and editable grid.
58548  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered - 
58549  * The container MUST have some type of size defined for the grid to fill. The container will be 
58550  * automatically set to position relative if it isn't already.
58551  * @param {Object} dataSource The data model to bind to
58552  * @param {Object} colModel The column model with info about this grid's columns
58553  */
58554 Roo.grid.EditorGrid = function(container, config){
58555     Roo.grid.EditorGrid.superclass.constructor.call(this, container, config);
58556     this.getGridEl().addClass("xedit-grid");
58557
58558     if(!this.selModel){
58559         this.selModel = new Roo.grid.CellSelectionModel();
58560     }
58561
58562     this.activeEditor = null;
58563
58564         this.addEvents({
58565             /**
58566              * @event beforeedit
58567              * Fires before cell editing is triggered. The edit event object has the following properties <br />
58568              * <ul style="padding:5px;padding-left:16px;">
58569              * <li>grid - This grid</li>
58570              * <li>record - The record being edited</li>
58571              * <li>field - The field name being edited</li>
58572              * <li>value - The value for the field being edited.</li>
58573              * <li>row - The grid row index</li>
58574              * <li>column - The grid column index</li>
58575              * <li>cancel - Set this to true to cancel the edit or return false from your handler.</li>
58576              * </ul>
58577              * @param {Object} e An edit event (see above for description)
58578              */
58579             "beforeedit" : true,
58580             /**
58581              * @event afteredit
58582              * Fires after a cell is edited. <br />
58583              * <ul style="padding:5px;padding-left:16px;">
58584              * <li>grid - This grid</li>
58585              * <li>record - The record being edited</li>
58586              * <li>field - The field name being edited</li>
58587              * <li>value - The value being set</li>
58588              * <li>originalValue - The original value for the field, before the edit.</li>
58589              * <li>row - The grid row index</li>
58590              * <li>column - The grid column index</li>
58591              * </ul>
58592              * @param {Object} e An edit event (see above for description)
58593              */
58594             "afteredit" : true,
58595             /**
58596              * @event validateedit
58597              * Fires after a cell is edited, but before the value is set in the record. 
58598          * You can use this to modify the value being set in the field, Return false
58599              * to cancel the change. The edit event object has the following properties <br />
58600              * <ul style="padding:5px;padding-left:16px;">
58601          * <li>editor - This editor</li>
58602              * <li>grid - This grid</li>
58603              * <li>record - The record being edited</li>
58604              * <li>field - The field name being edited</li>
58605              * <li>value - The value being set</li>
58606              * <li>originalValue - The original value for the field, before the edit.</li>
58607              * <li>row - The grid row index</li>
58608              * <li>column - The grid column index</li>
58609              * <li>cancel - Set this to true to cancel the edit or return false from your handler.</li>
58610              * </ul>
58611              * @param {Object} e An edit event (see above for description)
58612              */
58613             "validateedit" : true
58614         });
58615     this.on("bodyscroll", this.stopEditing,  this);
58616     this.on(this.clicksToEdit == 1 ? "cellclick" : "celldblclick", this.onCellDblClick,  this);
58617 };
58618
58619 Roo.extend(Roo.grid.EditorGrid, Roo.grid.Grid, {
58620     /**
58621      * @cfg {Number} clicksToEdit
58622      * The number of clicks on a cell required to display the cell's editor (defaults to 2)
58623      */
58624     clicksToEdit: 2,
58625
58626     // private
58627     isEditor : true,
58628     // private
58629     trackMouseOver: false, // causes very odd FF errors
58630
58631     onCellDblClick : function(g, row, col){
58632         this.startEditing(row, col);
58633     },
58634
58635     onEditComplete : function(ed, value, startValue){
58636         this.editing = false;
58637         this.activeEditor = null;
58638         ed.un("specialkey", this.selModel.onEditorKey, this.selModel);
58639         var r = ed.record;
58640         var field = this.colModel.getDataIndex(ed.col);
58641         var e = {
58642             grid: this,
58643             record: r,
58644             field: field,
58645             originalValue: startValue,
58646             value: value,
58647             row: ed.row,
58648             column: ed.col,
58649             cancel:false,
58650             editor: ed
58651         };
58652         var cell = Roo.get(this.view.getCell(ed.row,ed.col));
58653         cell.show();
58654           
58655         if(String(value) !== String(startValue)){
58656             
58657             if(this.fireEvent("validateedit", e) !== false && !e.cancel){
58658                 r.set(field, e.value);
58659                 // if we are dealing with a combo box..
58660                 // then we also set the 'name' colum to be the displayField
58661                 if (ed.field.displayField && ed.field.name) {
58662                     r.set(ed.field.name, ed.field.el.dom.value);
58663                 }
58664                 
58665                 delete e.cancel; //?? why!!!
58666                 this.fireEvent("afteredit", e);
58667             }
58668         } else {
58669             this.fireEvent("afteredit", e); // always fire it!
58670         }
58671         this.view.focusCell(ed.row, ed.col);
58672     },
58673
58674     /**
58675      * Starts editing the specified for the specified row/column
58676      * @param {Number} rowIndex
58677      * @param {Number} colIndex
58678      */
58679     startEditing : function(row, col){
58680         this.stopEditing();
58681         if(this.colModel.isCellEditable(col, row)){
58682             this.view.ensureVisible(row, col, true);
58683           
58684             var r = this.dataSource.getAt(row);
58685             var field = this.colModel.getDataIndex(col);
58686             var cell = Roo.get(this.view.getCell(row,col));
58687             var e = {
58688                 grid: this,
58689                 record: r,
58690                 field: field,
58691                 value: r.data[field],
58692                 row: row,
58693                 column: col,
58694                 cancel:false 
58695             };
58696             if(this.fireEvent("beforeedit", e) !== false && !e.cancel){
58697                 this.editing = true;
58698                 var ed = this.colModel.getCellEditor(col, row);
58699                 
58700                 if (!ed) {
58701                     return;
58702                 }
58703                 if(!ed.rendered){
58704                     ed.render(ed.parentEl || document.body);
58705                 }
58706                 ed.field.reset();
58707                
58708                 cell.hide();
58709                 
58710                 (function(){ // complex but required for focus issues in safari, ie and opera
58711                     ed.row = row;
58712                     ed.col = col;
58713                     ed.record = r;
58714                     ed.on("complete",   this.onEditComplete,        this,       {single: true});
58715                     ed.on("specialkey", this.selModel.onEditorKey,  this.selModel);
58716                     this.activeEditor = ed;
58717                     var v = r.data[field];
58718                     ed.startEdit(this.view.getCell(row, col), v);
58719                     // combo's with 'displayField and name set
58720                     if (ed.field.displayField && ed.field.name) {
58721                         ed.field.el.dom.value = r.data[ed.field.name];
58722                     }
58723                     
58724                     
58725                 }).defer(50, this);
58726             }
58727         }
58728     },
58729         
58730     /**
58731      * Stops any active editing
58732      */
58733     stopEditing : function(){
58734         if(this.activeEditor){
58735             this.activeEditor.completeEdit();
58736         }
58737         this.activeEditor = null;
58738     },
58739         
58740          /**
58741      * Called to get grid's drag proxy text, by default returns this.ddText.
58742      * @return {String}
58743      */
58744     getDragDropText : function(){
58745         var count = this.selModel.getSelectedCell() ? 1 : 0;
58746         return String.format(this.ddText, count, count == 1 ? '' : 's');
58747     }
58748         
58749 });/*
58750  * Based on:
58751  * Ext JS Library 1.1.1
58752  * Copyright(c) 2006-2007, Ext JS, LLC.
58753  *
58754  * Originally Released Under LGPL - original licence link has changed is not relivant.
58755  *
58756  * Fork - LGPL
58757  * <script type="text/javascript">
58758  */
58759
58760 // private - not really -- you end up using it !
58761 // This is a support class used internally by the Grid components
58762
58763 /**
58764  * @class Roo.grid.GridEditor
58765  * @extends Roo.Editor
58766  * Class for creating and editable grid elements.
58767  * @param {Object} config any settings (must include field)
58768  */
58769 Roo.grid.GridEditor = function(field, config){
58770     if (!config && field.field) {
58771         config = field;
58772         field = Roo.factory(config.field, Roo.form);
58773     }
58774     Roo.grid.GridEditor.superclass.constructor.call(this, field, config);
58775     field.monitorTab = false;
58776 };
58777
58778 Roo.extend(Roo.grid.GridEditor, Roo.Editor, {
58779     
58780     /**
58781      * @cfg {Roo.form.Field} field Field to wrap (or xtyped)
58782      */
58783     
58784     alignment: "tl-tl",
58785     autoSize: "width",
58786     hideEl : false,
58787     cls: "x-small-editor x-grid-editor",
58788     shim:false,
58789     shadow:"frame"
58790 });/*
58791  * Based on:
58792  * Ext JS Library 1.1.1
58793  * Copyright(c) 2006-2007, Ext JS, LLC.
58794  *
58795  * Originally Released Under LGPL - original licence link has changed is not relivant.
58796  *
58797  * Fork - LGPL
58798  * <script type="text/javascript">
58799  */
58800   
58801
58802   
58803 Roo.grid.PropertyRecord = Roo.data.Record.create([
58804     {name:'name',type:'string'},  'value'
58805 ]);
58806
58807
58808 Roo.grid.PropertyStore = function(grid, source){
58809     this.grid = grid;
58810     this.store = new Roo.data.Store({
58811         recordType : Roo.grid.PropertyRecord
58812     });
58813     this.store.on('update', this.onUpdate,  this);
58814     if(source){
58815         this.setSource(source);
58816     }
58817     Roo.grid.PropertyStore.superclass.constructor.call(this);
58818 };
58819
58820
58821
58822 Roo.extend(Roo.grid.PropertyStore, Roo.util.Observable, {
58823     setSource : function(o){
58824         this.source = o;
58825         this.store.removeAll();
58826         var data = [];
58827         for(var k in o){
58828             if(this.isEditableValue(o[k])){
58829                 data.push(new Roo.grid.PropertyRecord({name: k, value: o[k]}, k));
58830             }
58831         }
58832         this.store.loadRecords({records: data}, {}, true);
58833     },
58834
58835     onUpdate : function(ds, record, type){
58836         if(type == Roo.data.Record.EDIT){
58837             var v = record.data['value'];
58838             var oldValue = record.modified['value'];
58839             if(this.grid.fireEvent('beforepropertychange', this.source, record.id, v, oldValue) !== false){
58840                 this.source[record.id] = v;
58841                 record.commit();
58842                 this.grid.fireEvent('propertychange', this.source, record.id, v, oldValue);
58843             }else{
58844                 record.reject();
58845             }
58846         }
58847     },
58848
58849     getProperty : function(row){
58850        return this.store.getAt(row);
58851     },
58852
58853     isEditableValue: function(val){
58854         if(val && val instanceof Date){
58855             return true;
58856         }else if(typeof val == 'object' || typeof val == 'function'){
58857             return false;
58858         }
58859         return true;
58860     },
58861
58862     setValue : function(prop, value){
58863         this.source[prop] = value;
58864         this.store.getById(prop).set('value', value);
58865     },
58866
58867     getSource : function(){
58868         return this.source;
58869     }
58870 });
58871
58872 Roo.grid.PropertyColumnModel = function(grid, store){
58873     this.grid = grid;
58874     var g = Roo.grid;
58875     g.PropertyColumnModel.superclass.constructor.call(this, [
58876         {header: this.nameText, sortable: true, dataIndex:'name', id: 'name'},
58877         {header: this.valueText, resizable:false, dataIndex: 'value', id: 'value'}
58878     ]);
58879     this.store = store;
58880     this.bselect = Roo.DomHelper.append(document.body, {
58881         tag: 'select', style:'display:none', cls: 'x-grid-editor', children: [
58882             {tag: 'option', value: 'true', html: 'true'},
58883             {tag: 'option', value: 'false', html: 'false'}
58884         ]
58885     });
58886     Roo.id(this.bselect);
58887     var f = Roo.form;
58888     this.editors = {
58889         'date' : new g.GridEditor(new f.DateField({selectOnFocus:true})),
58890         'string' : new g.GridEditor(new f.TextField({selectOnFocus:true})),
58891         'number' : new g.GridEditor(new f.NumberField({selectOnFocus:true, style:'text-align:left;'})),
58892         'int' : new g.GridEditor(new f.NumberField({selectOnFocus:true, allowDecimals:false, style:'text-align:left;'})),
58893         'boolean' : new g.GridEditor(new f.Field({el:this.bselect,selectOnFocus:true}))
58894     };
58895     this.renderCellDelegate = this.renderCell.createDelegate(this);
58896     this.renderPropDelegate = this.renderProp.createDelegate(this);
58897 };
58898
58899 Roo.extend(Roo.grid.PropertyColumnModel, Roo.grid.ColumnModel, {
58900     
58901     
58902     nameText : 'Name',
58903     valueText : 'Value',
58904     
58905     dateFormat : 'm/j/Y',
58906     
58907     
58908     renderDate : function(dateVal){
58909         return dateVal.dateFormat(this.dateFormat);
58910     },
58911
58912     renderBool : function(bVal){
58913         return bVal ? 'true' : 'false';
58914     },
58915
58916     isCellEditable : function(colIndex, rowIndex){
58917         return colIndex == 1;
58918     },
58919
58920     getRenderer : function(col){
58921         return col == 1 ?
58922             this.renderCellDelegate : this.renderPropDelegate;
58923     },
58924
58925     renderProp : function(v){
58926         return this.getPropertyName(v);
58927     },
58928
58929     renderCell : function(val){
58930         var rv = val;
58931         if(val instanceof Date){
58932             rv = this.renderDate(val);
58933         }else if(typeof val == 'boolean'){
58934             rv = this.renderBool(val);
58935         }
58936         return Roo.util.Format.htmlEncode(rv);
58937     },
58938
58939     getPropertyName : function(name){
58940         var pn = this.grid.propertyNames;
58941         return pn && pn[name] ? pn[name] : name;
58942     },
58943
58944     getCellEditor : function(colIndex, rowIndex){
58945         var p = this.store.getProperty(rowIndex);
58946         var n = p.data['name'], val = p.data['value'];
58947         
58948         if(typeof(this.grid.customEditors[n]) == 'string'){
58949             return this.editors[this.grid.customEditors[n]];
58950         }
58951         if(typeof(this.grid.customEditors[n]) != 'undefined'){
58952             return this.grid.customEditors[n];
58953         }
58954         if(val instanceof Date){
58955             return this.editors['date'];
58956         }else if(typeof val == 'number'){
58957             return this.editors['number'];
58958         }else if(typeof val == 'boolean'){
58959             return this.editors['boolean'];
58960         }else{
58961             return this.editors['string'];
58962         }
58963     }
58964 });
58965
58966 /**
58967  * @class Roo.grid.PropertyGrid
58968  * @extends Roo.grid.EditorGrid
58969  * This class represents the  interface of a component based property grid control.
58970  * <br><br>Usage:<pre><code>
58971  var grid = new Roo.grid.PropertyGrid("my-container-id", {
58972       
58973  });
58974  // set any options
58975  grid.render();
58976  * </code></pre>
58977   
58978  * @constructor
58979  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
58980  * The container MUST have some type of size defined for the grid to fill. The container will be
58981  * automatically set to position relative if it isn't already.
58982  * @param {Object} config A config object that sets properties on this grid.
58983  */
58984 Roo.grid.PropertyGrid = function(container, config){
58985     config = config || {};
58986     var store = new Roo.grid.PropertyStore(this);
58987     this.store = store;
58988     var cm = new Roo.grid.PropertyColumnModel(this, store);
58989     store.store.sort('name', 'ASC');
58990     Roo.grid.PropertyGrid.superclass.constructor.call(this, container, Roo.apply({
58991         ds: store.store,
58992         cm: cm,
58993         enableColLock:false,
58994         enableColumnMove:false,
58995         stripeRows:false,
58996         trackMouseOver: false,
58997         clicksToEdit:1
58998     }, config));
58999     this.getGridEl().addClass('x-props-grid');
59000     this.lastEditRow = null;
59001     this.on('columnresize', this.onColumnResize, this);
59002     this.addEvents({
59003          /**
59004              * @event beforepropertychange
59005              * Fires before a property changes (return false to stop?)
59006              * @param {Roo.grid.PropertyGrid} grid property grid? (check could be store)
59007              * @param {String} id Record Id
59008              * @param {String} newval New Value
59009          * @param {String} oldval Old Value
59010              */
59011         "beforepropertychange": true,
59012         /**
59013              * @event propertychange
59014              * Fires after a property changes
59015              * @param {Roo.grid.PropertyGrid} grid property grid? (check could be store)
59016              * @param {String} id Record Id
59017              * @param {String} newval New Value
59018          * @param {String} oldval Old Value
59019              */
59020         "propertychange": true
59021     });
59022     this.customEditors = this.customEditors || {};
59023 };
59024 Roo.extend(Roo.grid.PropertyGrid, Roo.grid.EditorGrid, {
59025     
59026      /**
59027      * @cfg {Object} customEditors map of colnames=> custom editors.
59028      * the custom editor can be one of the standard ones (date|string|number|int|boolean), or a
59029      * grid editor eg. Roo.grid.GridEditor(new Roo.form.TextArea({selectOnFocus:true})),
59030      * false disables editing of the field.
59031          */
59032     
59033       /**
59034      * @cfg {Object} propertyNames map of property Names to their displayed value
59035          */
59036     
59037     render : function(){
59038         Roo.grid.PropertyGrid.superclass.render.call(this);
59039         this.autoSize.defer(100, this);
59040     },
59041
59042     autoSize : function(){
59043         Roo.grid.PropertyGrid.superclass.autoSize.call(this);
59044         if(this.view){
59045             this.view.fitColumns();
59046         }
59047     },
59048
59049     onColumnResize : function(){
59050         this.colModel.setColumnWidth(1, this.container.getWidth(true)-this.colModel.getColumnWidth(0));
59051         this.autoSize();
59052     },
59053     /**
59054      * Sets the data for the Grid
59055      * accepts a Key => Value object of all the elements avaiable.
59056      * @param {Object} data  to appear in grid.
59057      */
59058     setSource : function(source){
59059         this.store.setSource(source);
59060         //this.autoSize();
59061     },
59062     /**
59063      * Gets all the data from the grid.
59064      * @return {Object} data  data stored in grid
59065      */
59066     getSource : function(){
59067         return this.store.getSource();
59068     }
59069 });/*
59070   
59071  * Licence LGPL
59072  
59073  */
59074  
59075 /**
59076  * @class Roo.grid.Calendar
59077  * @extends Roo.util.Grid
59078  * This class extends the Grid to provide a calendar widget
59079  * <br><br>Usage:<pre><code>
59080  var grid = new Roo.grid.Calendar("my-container-id", {
59081      ds: myDataStore,
59082      cm: myColModel,
59083      selModel: mySelectionModel,
59084      autoSizeColumns: true,
59085      monitorWindowResize: false,
59086      trackMouseOver: true
59087      eventstore : real data store..
59088  });
59089  // set any options
59090  grid.render();
59091   
59092   * @constructor
59093  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
59094  * The container MUST have some type of size defined for the grid to fill. The container will be
59095  * automatically set to position relative if it isn't already.
59096  * @param {Object} config A config object that sets properties on this grid.
59097  */
59098 Roo.grid.Calendar = function(container, config){
59099         // initialize the container
59100         this.container = Roo.get(container);
59101         this.container.update("");
59102         this.container.setStyle("overflow", "hidden");
59103     this.container.addClass('x-grid-container');
59104
59105     this.id = this.container.id;
59106
59107     Roo.apply(this, config);
59108     // check and correct shorthanded configs
59109     
59110     var rows = [];
59111     var d =1;
59112     for (var r = 0;r < 6;r++) {
59113         
59114         rows[r]=[];
59115         for (var c =0;c < 7;c++) {
59116             rows[r][c]= '';
59117         }
59118     }
59119     if (this.eventStore) {
59120         this.eventStore= Roo.factory(this.eventStore, Roo.data);
59121         this.eventStore.on('load',this.onLoad, this);
59122         this.eventStore.on('beforeload',this.clearEvents, this);
59123          
59124     }
59125     
59126     this.dataSource = new Roo.data.Store({
59127             proxy: new Roo.data.MemoryProxy(rows),
59128             reader: new Roo.data.ArrayReader({}, [
59129                    'weekday0', 'weekday1', 'weekday2', 'weekday3', 'weekday4', 'weekday5', 'weekday6' ])
59130     });
59131
59132     this.dataSource.load();
59133     this.ds = this.dataSource;
59134     this.ds.xmodule = this.xmodule || false;
59135     
59136     
59137     var cellRender = function(v,x,r)
59138     {
59139         return String.format(
59140             '<div class="fc-day  fc-widget-content"><div>' +
59141                 '<div class="fc-event-container"></div>' +
59142                 '<div class="fc-day-number">{0}</div>'+
59143                 
59144                 '<div class="fc-day-content"><div style="position:relative"></div></div>' +
59145             '</div></div>', v);
59146     
59147     }
59148     
59149     
59150     this.colModel = new Roo.grid.ColumnModel( [
59151         {
59152             xtype: 'ColumnModel',
59153             xns: Roo.grid,
59154             dataIndex : 'weekday0',
59155             header : 'Sunday',
59156             renderer : cellRender
59157         },
59158         {
59159             xtype: 'ColumnModel',
59160             xns: Roo.grid,
59161             dataIndex : 'weekday1',
59162             header : 'Monday',
59163             renderer : cellRender
59164         },
59165         {
59166             xtype: 'ColumnModel',
59167             xns: Roo.grid,
59168             dataIndex : 'weekday2',
59169             header : 'Tuesday',
59170             renderer : cellRender
59171         },
59172         {
59173             xtype: 'ColumnModel',
59174             xns: Roo.grid,
59175             dataIndex : 'weekday3',
59176             header : 'Wednesday',
59177             renderer : cellRender
59178         },
59179         {
59180             xtype: 'ColumnModel',
59181             xns: Roo.grid,
59182             dataIndex : 'weekday4',
59183             header : 'Thursday',
59184             renderer : cellRender
59185         },
59186         {
59187             xtype: 'ColumnModel',
59188             xns: Roo.grid,
59189             dataIndex : 'weekday5',
59190             header : 'Friday',
59191             renderer : cellRender
59192         },
59193         {
59194             xtype: 'ColumnModel',
59195             xns: Roo.grid,
59196             dataIndex : 'weekday6',
59197             header : 'Saturday',
59198             renderer : cellRender
59199         }
59200     ]);
59201     this.cm = this.colModel;
59202     this.cm.xmodule = this.xmodule || false;
59203  
59204         
59205           
59206     //this.selModel = new Roo.grid.CellSelectionModel();
59207     //this.sm = this.selModel;
59208     //this.selModel.init(this);
59209     
59210     
59211     if(this.width){
59212         this.container.setWidth(this.width);
59213     }
59214
59215     if(this.height){
59216         this.container.setHeight(this.height);
59217     }
59218     /** @private */
59219         this.addEvents({
59220         // raw events
59221         /**
59222          * @event click
59223          * The raw click event for the entire grid.
59224          * @param {Roo.EventObject} e
59225          */
59226         "click" : true,
59227         /**
59228          * @event dblclick
59229          * The raw dblclick event for the entire grid.
59230          * @param {Roo.EventObject} e
59231          */
59232         "dblclick" : true,
59233         /**
59234          * @event contextmenu
59235          * The raw contextmenu event for the entire grid.
59236          * @param {Roo.EventObject} e
59237          */
59238         "contextmenu" : true,
59239         /**
59240          * @event mousedown
59241          * The raw mousedown event for the entire grid.
59242          * @param {Roo.EventObject} e
59243          */
59244         "mousedown" : true,
59245         /**
59246          * @event mouseup
59247          * The raw mouseup event for the entire grid.
59248          * @param {Roo.EventObject} e
59249          */
59250         "mouseup" : true,
59251         /**
59252          * @event mouseover
59253          * The raw mouseover event for the entire grid.
59254          * @param {Roo.EventObject} e
59255          */
59256         "mouseover" : true,
59257         /**
59258          * @event mouseout
59259          * The raw mouseout event for the entire grid.
59260          * @param {Roo.EventObject} e
59261          */
59262         "mouseout" : true,
59263         /**
59264          * @event keypress
59265          * The raw keypress event for the entire grid.
59266          * @param {Roo.EventObject} e
59267          */
59268         "keypress" : true,
59269         /**
59270          * @event keydown
59271          * The raw keydown event for the entire grid.
59272          * @param {Roo.EventObject} e
59273          */
59274         "keydown" : true,
59275
59276         // custom events
59277
59278         /**
59279          * @event cellclick
59280          * Fires when a cell is clicked
59281          * @param {Grid} this
59282          * @param {Number} rowIndex
59283          * @param {Number} columnIndex
59284          * @param {Roo.EventObject} e
59285          */
59286         "cellclick" : true,
59287         /**
59288          * @event celldblclick
59289          * Fires when a cell is double clicked
59290          * @param {Grid} this
59291          * @param {Number} rowIndex
59292          * @param {Number} columnIndex
59293          * @param {Roo.EventObject} e
59294          */
59295         "celldblclick" : true,
59296         /**
59297          * @event rowclick
59298          * Fires when a row is clicked
59299          * @param {Grid} this
59300          * @param {Number} rowIndex
59301          * @param {Roo.EventObject} e
59302          */
59303         "rowclick" : true,
59304         /**
59305          * @event rowdblclick
59306          * Fires when a row is double clicked
59307          * @param {Grid} this
59308          * @param {Number} rowIndex
59309          * @param {Roo.EventObject} e
59310          */
59311         "rowdblclick" : true,
59312         /**
59313          * @event headerclick
59314          * Fires when a header is clicked
59315          * @param {Grid} this
59316          * @param {Number} columnIndex
59317          * @param {Roo.EventObject} e
59318          */
59319         "headerclick" : true,
59320         /**
59321          * @event headerdblclick
59322          * Fires when a header cell is double clicked
59323          * @param {Grid} this
59324          * @param {Number} columnIndex
59325          * @param {Roo.EventObject} e
59326          */
59327         "headerdblclick" : true,
59328         /**
59329          * @event rowcontextmenu
59330          * Fires when a row is right clicked
59331          * @param {Grid} this
59332          * @param {Number} rowIndex
59333          * @param {Roo.EventObject} e
59334          */
59335         "rowcontextmenu" : true,
59336         /**
59337          * @event cellcontextmenu
59338          * Fires when a cell is right clicked
59339          * @param {Grid} this
59340          * @param {Number} rowIndex
59341          * @param {Number} cellIndex
59342          * @param {Roo.EventObject} e
59343          */
59344          "cellcontextmenu" : true,
59345         /**
59346          * @event headercontextmenu
59347          * Fires when a header is right clicked
59348          * @param {Grid} this
59349          * @param {Number} columnIndex
59350          * @param {Roo.EventObject} e
59351          */
59352         "headercontextmenu" : true,
59353         /**
59354          * @event bodyscroll
59355          * Fires when the body element is scrolled
59356          * @param {Number} scrollLeft
59357          * @param {Number} scrollTop
59358          */
59359         "bodyscroll" : true,
59360         /**
59361          * @event columnresize
59362          * Fires when the user resizes a column
59363          * @param {Number} columnIndex
59364          * @param {Number} newSize
59365          */
59366         "columnresize" : true,
59367         /**
59368          * @event columnmove
59369          * Fires when the user moves a column
59370          * @param {Number} oldIndex
59371          * @param {Number} newIndex
59372          */
59373         "columnmove" : true,
59374         /**
59375          * @event startdrag
59376          * Fires when row(s) start being dragged
59377          * @param {Grid} this
59378          * @param {Roo.GridDD} dd The drag drop object
59379          * @param {event} e The raw browser event
59380          */
59381         "startdrag" : true,
59382         /**
59383          * @event enddrag
59384          * Fires when a drag operation is complete
59385          * @param {Grid} this
59386          * @param {Roo.GridDD} dd The drag drop object
59387          * @param {event} e The raw browser event
59388          */
59389         "enddrag" : true,
59390         /**
59391          * @event dragdrop
59392          * Fires when dragged row(s) are dropped on a valid DD target
59393          * @param {Grid} this
59394          * @param {Roo.GridDD} dd The drag drop object
59395          * @param {String} targetId The target drag drop object
59396          * @param {event} e The raw browser event
59397          */
59398         "dragdrop" : true,
59399         /**
59400          * @event dragover
59401          * Fires while row(s) are being dragged. "targetId" is the id of the Yahoo.util.DD object the selected rows are being dragged over.
59402          * @param {Grid} this
59403          * @param {Roo.GridDD} dd The drag drop object
59404          * @param {String} targetId The target drag drop object
59405          * @param {event} e The raw browser event
59406          */
59407         "dragover" : true,
59408         /**
59409          * @event dragenter
59410          *  Fires when the dragged row(s) first cross another DD target while being dragged
59411          * @param {Grid} this
59412          * @param {Roo.GridDD} dd The drag drop object
59413          * @param {String} targetId The target drag drop object
59414          * @param {event} e The raw browser event
59415          */
59416         "dragenter" : true,
59417         /**
59418          * @event dragout
59419          * Fires when the dragged row(s) leave another DD target while being dragged
59420          * @param {Grid} this
59421          * @param {Roo.GridDD} dd The drag drop object
59422          * @param {String} targetId The target drag drop object
59423          * @param {event} e The raw browser event
59424          */
59425         "dragout" : true,
59426         /**
59427          * @event rowclass
59428          * Fires when a row is rendered, so you can change add a style to it.
59429          * @param {GridView} gridview   The grid view
59430          * @param {Object} rowcfg   contains record  rowIndex and rowClass - set rowClass to add a style.
59431          */
59432         'rowclass' : true,
59433
59434         /**
59435          * @event render
59436          * Fires when the grid is rendered
59437          * @param {Grid} grid
59438          */
59439         'render' : true,
59440             /**
59441              * @event select
59442              * Fires when a date is selected
59443              * @param {DatePicker} this
59444              * @param {Date} date The selected date
59445              */
59446         'select': true,
59447         /**
59448              * @event monthchange
59449              * Fires when the displayed month changes 
59450              * @param {DatePicker} this
59451              * @param {Date} date The selected month
59452              */
59453         'monthchange': true,
59454         /**
59455              * @event evententer
59456              * Fires when mouse over an event
59457              * @param {Calendar} this
59458              * @param {event} Event
59459              */
59460         'evententer': true,
59461         /**
59462              * @event eventleave
59463              * Fires when the mouse leaves an
59464              * @param {Calendar} this
59465              * @param {event}
59466              */
59467         'eventleave': true,
59468         /**
59469              * @event eventclick
59470              * Fires when the mouse click an
59471              * @param {Calendar} this
59472              * @param {event}
59473              */
59474         'eventclick': true,
59475         /**
59476              * @event eventrender
59477              * Fires before each cell is rendered, so you can modify the contents, like cls / title / qtip
59478              * @param {Calendar} this
59479              * @param {data} data to be modified
59480              */
59481         'eventrender': true
59482         
59483     });
59484
59485     Roo.grid.Grid.superclass.constructor.call(this);
59486     this.on('render', function() {
59487         this.view.el.addClass('x-grid-cal'); 
59488         
59489         (function() { this.setDate(new Date()); }).defer(100,this); //default today..
59490
59491     },this);
59492     
59493     if (!Roo.grid.Calendar.style) {
59494         Roo.grid.Calendar.style = Roo.util.CSS.createStyleSheet({
59495             
59496             
59497             '.x-grid-cal .x-grid-col' :  {
59498                 height: 'auto !important',
59499                 'vertical-align': 'top'
59500             },
59501             '.x-grid-cal  .fc-event-hori' : {
59502                 height: '14px'
59503             }
59504              
59505             
59506         }, Roo.id());
59507     }
59508
59509     
59510     
59511 };
59512 Roo.extend(Roo.grid.Calendar, Roo.grid.Grid, {
59513     /**
59514      * @cfg {Store} eventStore The store that loads events.
59515      */
59516     eventStore : 25,
59517
59518      
59519     activeDate : false,
59520     startDay : 0,
59521     autoWidth : true,
59522     monitorWindowResize : false,
59523
59524     
59525     resizeColumns : function() {
59526         var col = (this.view.el.getWidth() / 7) - 3;
59527         // loop through cols, and setWidth
59528         for(var i =0 ; i < 7 ; i++){
59529             this.cm.setColumnWidth(i, col);
59530         }
59531     },
59532      setDate :function(date) {
59533         
59534         Roo.log('setDate?');
59535         
59536         this.resizeColumns();
59537         var vd = this.activeDate;
59538         this.activeDate = date;
59539 //        if(vd && this.el){
59540 //            var t = date.getTime();
59541 //            if(vd.getMonth() == date.getMonth() && vd.getFullYear() == date.getFullYear()){
59542 //                Roo.log('using add remove');
59543 //                
59544 //                this.fireEvent('monthchange', this, date);
59545 //                
59546 //                this.cells.removeClass("fc-state-highlight");
59547 //                this.cells.each(function(c){
59548 //                   if(c.dateValue == t){
59549 //                       c.addClass("fc-state-highlight");
59550 //                       setTimeout(function(){
59551 //                            try{c.dom.firstChild.focus();}catch(e){}
59552 //                       }, 50);
59553 //                       return false;
59554 //                   }
59555 //                   return true;
59556 //                });
59557 //                return;
59558 //            }
59559 //        }
59560         
59561         var days = date.getDaysInMonth();
59562         
59563         var firstOfMonth = date.getFirstDateOfMonth();
59564         var startingPos = firstOfMonth.getDay()-this.startDay;
59565         
59566         if(startingPos < this.startDay){
59567             startingPos += 7;
59568         }
59569         
59570         var pm = date.add(Date.MONTH, -1);
59571         var prevStart = pm.getDaysInMonth()-startingPos;
59572 //        
59573         
59574         
59575         this.cells = this.view.el.select('.x-grid-row .x-grid-col',true);
59576         
59577         this.textNodes = this.view.el.query('.x-grid-row .x-grid-col .x-grid-cell-text');
59578         //this.cells.addClassOnOver('fc-state-hover');
59579         
59580         var cells = this.cells.elements;
59581         var textEls = this.textNodes;
59582         
59583         //Roo.each(cells, function(cell){
59584         //    cell.removeClass([ 'fc-past', 'fc-other-month', 'fc-future', 'fc-state-highlight', 'fc-state-disabled']);
59585         //});
59586         
59587         days += startingPos;
59588
59589         // convert everything to numbers so it's fast
59590         var day = 86400000;
59591         var d = (new Date(pm.getFullYear(), pm.getMonth(), prevStart)).clearTime();
59592         //Roo.log(d);
59593         //Roo.log(pm);
59594         //Roo.log(prevStart);
59595         
59596         var today = new Date().clearTime().getTime();
59597         var sel = date.clearTime().getTime();
59598         var min = this.minDate ? this.minDate.clearTime() : Number.NEGATIVE_INFINITY;
59599         var max = this.maxDate ? this.maxDate.clearTime() : Number.POSITIVE_INFINITY;
59600         var ddMatch = this.disabledDatesRE;
59601         var ddText = this.disabledDatesText;
59602         var ddays = this.disabledDays ? this.disabledDays.join("") : false;
59603         var ddaysText = this.disabledDaysText;
59604         var format = this.format;
59605         
59606         var setCellClass = function(cal, cell){
59607             
59608             //Roo.log('set Cell Class');
59609             cell.title = "";
59610             var t = d.getTime();
59611             
59612             //Roo.log(d);
59613             
59614             
59615             cell.dateValue = t;
59616             if(t == today){
59617                 cell.className += " fc-today";
59618                 cell.className += " fc-state-highlight";
59619                 cell.title = cal.todayText;
59620             }
59621             if(t == sel){
59622                 // disable highlight in other month..
59623                 cell.className += " fc-state-highlight";
59624                 
59625             }
59626             // disabling
59627             if(t < min) {
59628                 //cell.className = " fc-state-disabled";
59629                 cell.title = cal.minText;
59630                 return;
59631             }
59632             if(t > max) {
59633                 //cell.className = " fc-state-disabled";
59634                 cell.title = cal.maxText;
59635                 return;
59636             }
59637             if(ddays){
59638                 if(ddays.indexOf(d.getDay()) != -1){
59639                     // cell.title = ddaysText;
59640                    // cell.className = " fc-state-disabled";
59641                 }
59642             }
59643             if(ddMatch && format){
59644                 var fvalue = d.dateFormat(format);
59645                 if(ddMatch.test(fvalue)){
59646                     cell.title = ddText.replace("%0", fvalue);
59647                    cell.className = " fc-state-disabled";
59648                 }
59649             }
59650             
59651             if (!cell.initialClassName) {
59652                 cell.initialClassName = cell.dom.className;
59653             }
59654             
59655             cell.dom.className = cell.initialClassName  + ' ' +  cell.className;
59656         };
59657
59658         var i = 0;
59659         
59660         for(; i < startingPos; i++) {
59661             cells[i].dayName =  (++prevStart);
59662             Roo.log(textEls[i]);
59663             d.setDate(d.getDate()+1);
59664             
59665             //cells[i].className = "fc-past fc-other-month";
59666             setCellClass(this, cells[i]);
59667         }
59668         
59669         var intDay = 0;
59670         
59671         for(; i < days; i++){
59672             intDay = i - startingPos + 1;
59673             cells[i].dayName =  (intDay);
59674             d.setDate(d.getDate()+1);
59675             
59676             cells[i].className = ''; // "x-date-active";
59677             setCellClass(this, cells[i]);
59678         }
59679         var extraDays = 0;
59680         
59681         for(; i < 42; i++) {
59682             //textEls[i].innerHTML = (++extraDays);
59683             
59684             d.setDate(d.getDate()+1);
59685             cells[i].dayName = (++extraDays);
59686             cells[i].className = "fc-future fc-other-month";
59687             setCellClass(this, cells[i]);
59688         }
59689         
59690         //this.el.select('.fc-header-title h2',true).update(Date.monthNames[date.getMonth()] + " " + date.getFullYear());
59691         
59692         var totalRows = Math.ceil((date.getDaysInMonth() + date.getFirstDateOfMonth().getDay()) / 7);
59693         
59694         // this will cause all the cells to mis
59695         var rows= [];
59696         var i =0;
59697         for (var r = 0;r < 6;r++) {
59698             for (var c =0;c < 7;c++) {
59699                 this.ds.getAt(r).set('weekday' + c ,cells[i++].dayName );
59700             }    
59701         }
59702         
59703         this.cells = this.view.el.select('.x-grid-row .x-grid-col',true);
59704         for(i=0;i<cells.length;i++) {
59705             
59706             this.cells.elements[i].dayName = cells[i].dayName ;
59707             this.cells.elements[i].className = cells[i].className;
59708             this.cells.elements[i].initialClassName = cells[i].initialClassName ;
59709             this.cells.elements[i].title = cells[i].title ;
59710             this.cells.elements[i].dateValue = cells[i].dateValue ;
59711         }
59712         
59713         
59714         
59715         
59716         //this.el.select('tr.fc-week.fc-prev-last',true).removeClass('fc-last');
59717         //this.el.select('tr.fc-week.fc-next-last',true).addClass('fc-last').show();
59718         
59719         ////if(totalRows != 6){
59720             //this.el.select('tr.fc-week.fc-last',true).removeClass('fc-last').addClass('fc-next-last').hide();
59721            // this.el.select('tr.fc-week.fc-prev-last',true).addClass('fc-last');
59722        // }
59723         
59724         this.fireEvent('monthchange', this, date);
59725         
59726         
59727     },
59728  /**
59729      * Returns the grid's SelectionModel.
59730      * @return {SelectionModel}
59731      */
59732     getSelectionModel : function(){
59733         if(!this.selModel){
59734             this.selModel = new Roo.grid.CellSelectionModel();
59735         }
59736         return this.selModel;
59737     },
59738
59739     load: function() {
59740         this.eventStore.load()
59741         
59742         
59743         
59744     },
59745     
59746     findCell : function(dt) {
59747         dt = dt.clearTime().getTime();
59748         var ret = false;
59749         this.cells.each(function(c){
59750             //Roo.log("check " +c.dateValue + '?=' + dt);
59751             if(c.dateValue == dt){
59752                 ret = c;
59753                 return false;
59754             }
59755             return true;
59756         });
59757         
59758         return ret;
59759     },
59760     
59761     findCells : function(rec) {
59762         var s = rec.data.start_dt.clone().clearTime().getTime();
59763        // Roo.log(s);
59764         var e= rec.data.end_dt.clone().clearTime().getTime();
59765        // Roo.log(e);
59766         var ret = [];
59767         this.cells.each(function(c){
59768              ////Roo.log("check " +c.dateValue + '<' + e + ' > ' + s);
59769             
59770             if(c.dateValue > e){
59771                 return ;
59772             }
59773             if(c.dateValue < s){
59774                 return ;
59775             }
59776             ret.push(c);
59777         });
59778         
59779         return ret;    
59780     },
59781     
59782     findBestRow: function(cells)
59783     {
59784         var ret = 0;
59785         
59786         for (var i =0 ; i < cells.length;i++) {
59787             ret  = Math.max(cells[i].rows || 0,ret);
59788         }
59789         return ret;
59790         
59791     },
59792     
59793     
59794     addItem : function(rec)
59795     {
59796         // look for vertical location slot in
59797         var cells = this.findCells(rec);
59798         
59799         rec.row = this.findBestRow(cells);
59800         
59801         // work out the location.
59802         
59803         var crow = false;
59804         var rows = [];
59805         for(var i =0; i < cells.length; i++) {
59806             if (!crow) {
59807                 crow = {
59808                     start : cells[i],
59809                     end :  cells[i]
59810                 };
59811                 continue;
59812             }
59813             if (crow.start.getY() == cells[i].getY()) {
59814                 // on same row.
59815                 crow.end = cells[i];
59816                 continue;
59817             }
59818             // different row.
59819             rows.push(crow);
59820             crow = {
59821                 start: cells[i],
59822                 end : cells[i]
59823             };
59824             
59825         }
59826         
59827         rows.push(crow);
59828         rec.els = [];
59829         rec.rows = rows;
59830         rec.cells = cells;
59831         for (var i = 0; i < cells.length;i++) {
59832             cells[i].rows = Math.max(cells[i].rows || 0 , rec.row + 1 );
59833             
59834         }
59835         
59836         
59837     },
59838     
59839     clearEvents: function() {
59840         
59841         if (!this.eventStore.getCount()) {
59842             return;
59843         }
59844         // reset number of rows in cells.
59845         Roo.each(this.cells.elements, function(c){
59846             c.rows = 0;
59847         });
59848         
59849         this.eventStore.each(function(e) {
59850             this.clearEvent(e);
59851         },this);
59852         
59853     },
59854     
59855     clearEvent : function(ev)
59856     {
59857         if (ev.els) {
59858             Roo.each(ev.els, function(el) {
59859                 el.un('mouseenter' ,this.onEventEnter, this);
59860                 el.un('mouseleave' ,this.onEventLeave, this);
59861                 el.remove();
59862             },this);
59863             ev.els = [];
59864         }
59865     },
59866     
59867     
59868     renderEvent : function(ev,ctr) {
59869         if (!ctr) {
59870              ctr = this.view.el.select('.fc-event-container',true).first();
59871         }
59872         
59873          
59874         this.clearEvent(ev);
59875             //code
59876        
59877         
59878         
59879         ev.els = [];
59880         var cells = ev.cells;
59881         var rows = ev.rows;
59882         this.fireEvent('eventrender', this, ev);
59883         
59884         for(var i =0; i < rows.length; i++) {
59885             
59886             cls = '';
59887             if (i == 0) {
59888                 cls += ' fc-event-start';
59889             }
59890             if ((i+1) == rows.length) {
59891                 cls += ' fc-event-end';
59892             }
59893             
59894             //Roo.log(ev.data);
59895             // how many rows should it span..
59896             var cg = this.eventTmpl.append(ctr,Roo.apply({
59897                 fccls : cls
59898                 
59899             }, ev.data) , true);
59900             
59901             
59902             cg.on('mouseenter' ,this.onEventEnter, this, ev);
59903             cg.on('mouseleave' ,this.onEventLeave, this, ev);
59904             cg.on('click', this.onEventClick, this, ev);
59905             
59906             ev.els.push(cg);
59907             
59908             var sbox = rows[i].start.select('.fc-day-content',true).first().getBox();
59909             var ebox = rows[i].end.select('.fc-day-content',true).first().getBox();
59910             //Roo.log(cg);
59911              
59912             cg.setXY([sbox.x +2, sbox.y +(ev.row * 20)]);    
59913             cg.setWidth(ebox.right - sbox.x -2);
59914         }
59915     },
59916     
59917     renderEvents: function()
59918     {   
59919         // first make sure there is enough space..
59920         
59921         if (!this.eventTmpl) {
59922             this.eventTmpl = new Roo.Template(
59923                 '<div class="roo-dynamic fc-event fc-event-hori fc-event-draggable ui-draggable {fccls} {cls}"  style="position: absolute" unselectable="on">' +
59924                     '<div class="fc-event-inner">' +
59925                         '<span class="fc-event-time">{time}</span>' +
59926                         '<span class="fc-event-title" qtip="{qtip}">{title}</span>' +
59927                     '</div>' +
59928                     '<div class="ui-resizable-heandle ui-resizable-e">&nbsp;&nbsp;&nbsp;</div>' +
59929                 '</div>'
59930             );
59931                 
59932         }
59933                
59934         
59935         
59936         this.cells.each(function(c) {
59937             //Roo.log(c.select('.fc-day-content div',true).first());
59938             c.select('.fc-day-content div',true).first().setHeight(Math.max(34, (c.rows || 1) * 20));
59939         });
59940         
59941         var ctr = this.view.el.select('.fc-event-container',true).first();
59942         
59943         var cls;
59944         this.eventStore.each(function(ev){
59945             
59946             this.renderEvent(ev);
59947              
59948              
59949         }, this);
59950         this.view.layout();
59951         
59952     },
59953     
59954     onEventEnter: function (e, el,event,d) {
59955         this.fireEvent('evententer', this, el, event);
59956     },
59957     
59958     onEventLeave: function (e, el,event,d) {
59959         this.fireEvent('eventleave', this, el, event);
59960     },
59961     
59962     onEventClick: function (e, el,event,d) {
59963         this.fireEvent('eventclick', this, el, event);
59964     },
59965     
59966     onMonthChange: function () {
59967         this.store.load();
59968     },
59969     
59970     onLoad: function () {
59971         
59972         //Roo.log('calendar onload');
59973 //         
59974         if(this.eventStore.getCount() > 0){
59975             
59976            
59977             
59978             this.eventStore.each(function(d){
59979                 
59980                 
59981                 // FIXME..
59982                 var add =   d.data;
59983                 if (typeof(add.end_dt) == 'undefined')  {
59984                     Roo.log("Missing End time in calendar data: ");
59985                     Roo.log(d);
59986                     return;
59987                 }
59988                 if (typeof(add.start_dt) == 'undefined')  {
59989                     Roo.log("Missing Start time in calendar data: ");
59990                     Roo.log(d);
59991                     return;
59992                 }
59993                 add.start_dt = typeof(add.start_dt) == 'string' ? Date.parseDate(add.start_dt,'Y-m-d H:i:s') : add.start_dt,
59994                 add.end_dt = typeof(add.end_dt) == 'string' ? Date.parseDate(add.end_dt,'Y-m-d H:i:s') : add.end_dt,
59995                 add.id = add.id || d.id;
59996                 add.title = add.title || '??';
59997                 
59998                 this.addItem(d);
59999                 
60000              
60001             },this);
60002         }
60003         
60004         this.renderEvents();
60005     }
60006     
60007
60008 });
60009 /*
60010  grid : {
60011                 xtype: 'Grid',
60012                 xns: Roo.grid,
60013                 listeners : {
60014                     render : function ()
60015                     {
60016                         _this.grid = this;
60017                         
60018                         if (!this.view.el.hasClass('course-timesheet')) {
60019                             this.view.el.addClass('course-timesheet');
60020                         }
60021                         if (this.tsStyle) {
60022                             this.ds.load({});
60023                             return; 
60024                         }
60025                         Roo.log('width');
60026                         Roo.log(_this.grid.view.el.getWidth());
60027                         
60028                         
60029                         this.tsStyle =  Roo.util.CSS.createStyleSheet({
60030                             '.course-timesheet .x-grid-row' : {
60031                                 height: '80px'
60032                             },
60033                             '.x-grid-row td' : {
60034                                 'vertical-align' : 0
60035                             },
60036                             '.course-edit-link' : {
60037                                 'color' : 'blue',
60038                                 'text-overflow' : 'ellipsis',
60039                                 'overflow' : 'hidden',
60040                                 'white-space' : 'nowrap',
60041                                 'cursor' : 'pointer'
60042                             },
60043                             '.sub-link' : {
60044                                 'color' : 'green'
60045                             },
60046                             '.de-act-sup-link' : {
60047                                 'color' : 'purple',
60048                                 'text-decoration' : 'line-through'
60049                             },
60050                             '.de-act-link' : {
60051                                 'color' : 'red',
60052                                 'text-decoration' : 'line-through'
60053                             },
60054                             '.course-timesheet .course-highlight' : {
60055                                 'border-top-style': 'dashed !important',
60056                                 'border-bottom-bottom': 'dashed !important'
60057                             },
60058                             '.course-timesheet .course-item' : {
60059                                 'font-family'   : 'tahoma, arial, helvetica',
60060                                 'font-size'     : '11px',
60061                                 'overflow'      : 'hidden',
60062                                 'padding-left'  : '10px',
60063                                 'padding-right' : '10px',
60064                                 'padding-top' : '10px' 
60065                             }
60066                             
60067                         }, Roo.id());
60068                                 this.ds.load({});
60069                     }
60070                 },
60071                 autoWidth : true,
60072                 monitorWindowResize : false,
60073                 cellrenderer : function(v,x,r)
60074                 {
60075                     return v;
60076                 },
60077                 sm : {
60078                     xtype: 'CellSelectionModel',
60079                     xns: Roo.grid
60080                 },
60081                 dataSource : {
60082                     xtype: 'Store',
60083                     xns: Roo.data,
60084                     listeners : {
60085                         beforeload : function (_self, options)
60086                         {
60087                             options.params = options.params || {};
60088                             options.params._month = _this.monthField.getValue();
60089                             options.params.limit = 9999;
60090                             options.params['sort'] = 'when_dt';    
60091                             options.params['dir'] = 'ASC';    
60092                             this.proxy.loadResponse = this.loadResponse;
60093                             Roo.log("load?");
60094                             //this.addColumns();
60095                         },
60096                         load : function (_self, records, options)
60097                         {
60098                             _this.grid.view.el.select('.course-edit-link', true).on('click', function() {
60099                                 // if you click on the translation.. you can edit it...
60100                                 var el = Roo.get(this);
60101                                 var id = el.dom.getAttribute('data-id');
60102                                 var d = el.dom.getAttribute('data-date');
60103                                 var t = el.dom.getAttribute('data-time');
60104                                 //var id = this.child('span').dom.textContent;
60105                                 
60106                                 //Roo.log(this);
60107                                 Pman.Dialog.CourseCalendar.show({
60108                                     id : id,
60109                                     when_d : d,
60110                                     when_t : t,
60111                                     productitem_active : id ? 1 : 0
60112                                 }, function() {
60113                                     _this.grid.ds.load({});
60114                                 });
60115                            
60116                            });
60117                            
60118                            _this.panel.fireEvent('resize', [ '', '' ]);
60119                         }
60120                     },
60121                     loadResponse : function(o, success, response){
60122                             // this is overridden on before load..
60123                             
60124                             Roo.log("our code?");       
60125                             //Roo.log(success);
60126                             //Roo.log(response)
60127                             delete this.activeRequest;
60128                             if(!success){
60129                                 this.fireEvent("loadexception", this, o, response);
60130                                 o.request.callback.call(o.request.scope, null, o.request.arg, false);
60131                                 return;
60132                             }
60133                             var result;
60134                             try {
60135                                 result = o.reader.read(response);
60136                             }catch(e){
60137                                 Roo.log("load exception?");
60138                                 this.fireEvent("loadexception", this, o, response, e);
60139                                 o.request.callback.call(o.request.scope, null, o.request.arg, false);
60140                                 return;
60141                             }
60142                             Roo.log("ready...");        
60143                             // loop through result.records;
60144                             // and set this.tdate[date] = [] << array of records..
60145                             _this.tdata  = {};
60146                             Roo.each(result.records, function(r){
60147                                 //Roo.log(r.data);
60148                                 if(typeof(_this.tdata[r.data.when_dt.format('j')]) == 'undefined'){
60149                                     _this.tdata[r.data.when_dt.format('j')] = [];
60150                                 }
60151                                 _this.tdata[r.data.when_dt.format('j')].push(r.data);
60152                             });
60153                             
60154                             //Roo.log(_this.tdata);
60155                             
60156                             result.records = [];
60157                             result.totalRecords = 6;
60158                     
60159                             // let's generate some duumy records for the rows.
60160                             //var st = _this.dateField.getValue();
60161                             
60162                             // work out monday..
60163                             //st = st.add(Date.DAY, -1 * st.format('w'));
60164                             
60165                             var date = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60166                             
60167                             var firstOfMonth = date.getFirstDayOfMonth();
60168                             var days = date.getDaysInMonth();
60169                             var d = 1;
60170                             var firstAdded = false;
60171                             for (var i = 0; i < result.totalRecords ; i++) {
60172                                 //var d= st.add(Date.DAY, i);
60173                                 var row = {};
60174                                 var added = 0;
60175                                 for(var w = 0 ; w < 7 ; w++){
60176                                     if(!firstAdded && firstOfMonth != w){
60177                                         continue;
60178                                     }
60179                                     if(d > days){
60180                                         continue;
60181                                     }
60182                                     firstAdded = true;
60183                                     var dd = (d > 0 && d < 10) ? "0"+d : d;
60184                                     row['weekday'+w] = String.format(
60185                                                     '<span style="font-size: 16px;"><b>{0}</b></span>'+
60186                                                     '<span class="course-edit-link" style="color:blue;" data-id="0" data-date="{1}"> Add New</span>',
60187                                                     d,
60188                                                     date.format('Y-m-')+dd
60189                                                 );
60190                                     added++;
60191                                     if(typeof(_this.tdata[d]) != 'undefined'){
60192                                         Roo.each(_this.tdata[d], function(r){
60193                                             var is_sub = '';
60194                                             var deactive = '';
60195                                             var id = r.id;
60196                                             var desc = (r.productitem_id_descrip) ? r.productitem_id_descrip : '';
60197                                             if(r.parent_id*1>0){
60198                                                 is_sub = (r.productitem_id_visible*1 < 1) ? 'de-act-sup-link' :'sub-link';
60199                                                 id = r.parent_id;
60200                                             }
60201                                             if(r.productitem_id_visible*1 < 1 && r.parent_id*1 < 1){
60202                                                 deactive = 'de-act-link';
60203                                             }
60204                                             
60205                                             row['weekday'+w] += String.format(
60206                                                     '<br /><span class="course-edit-link {3} {4}" qtip="{5}" data-id="{0}">{2} - {1}</span>',
60207                                                     id, //0
60208                                                     r.product_id_name, //1
60209                                                     r.when_dt.format('h:ia'), //2
60210                                                     is_sub, //3
60211                                                     deactive, //4
60212                                                     desc // 5
60213                                             );
60214                                         });
60215                                     }
60216                                     d++;
60217                                 }
60218                                 
60219                                 // only do this if something added..
60220                                 if(added > 0){ 
60221                                     result.records.push(_this.grid.dataSource.reader.newRow(row));
60222                                 }
60223                                 
60224                                 
60225                                 // push it twice. (second one with an hour..
60226                                 
60227                             }
60228                             //Roo.log(result);
60229                             this.fireEvent("load", this, o, o.request.arg);
60230                             o.request.callback.call(o.request.scope, result, o.request.arg, true);
60231                         },
60232                     sortInfo : {field: 'when_dt', direction : 'ASC' },
60233                     proxy : {
60234                         xtype: 'HttpProxy',
60235                         xns: Roo.data,
60236                         method : 'GET',
60237                         url : baseURL + '/Roo/Shop_course.php'
60238                     },
60239                     reader : {
60240                         xtype: 'JsonReader',
60241                         xns: Roo.data,
60242                         id : 'id',
60243                         fields : [
60244                             {
60245                                 'name': 'id',
60246                                 'type': 'int'
60247                             },
60248                             {
60249                                 'name': 'when_dt',
60250                                 'type': 'string'
60251                             },
60252                             {
60253                                 'name': 'end_dt',
60254                                 'type': 'string'
60255                             },
60256                             {
60257                                 'name': 'parent_id',
60258                                 'type': 'int'
60259                             },
60260                             {
60261                                 'name': 'product_id',
60262                                 'type': 'int'
60263                             },
60264                             {
60265                                 'name': 'productitem_id',
60266                                 'type': 'int'
60267                             },
60268                             {
60269                                 'name': 'guid',
60270                                 'type': 'int'
60271                             }
60272                         ]
60273                     }
60274                 },
60275                 toolbar : {
60276                     xtype: 'Toolbar',
60277                     xns: Roo,
60278                     items : [
60279                         {
60280                             xtype: 'Button',
60281                             xns: Roo.Toolbar,
60282                             listeners : {
60283                                 click : function (_self, e)
60284                                 {
60285                                     var sd = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60286                                     sd.setMonth(sd.getMonth()-1);
60287                                     _this.monthField.setValue(sd.format('Y-m-d'));
60288                                     _this.grid.ds.load({});
60289                                 }
60290                             },
60291                             text : "Back"
60292                         },
60293                         {
60294                             xtype: 'Separator',
60295                             xns: Roo.Toolbar
60296                         },
60297                         {
60298                             xtype: 'MonthField',
60299                             xns: Roo.form,
60300                             listeners : {
60301                                 render : function (_self)
60302                                 {
60303                                     _this.monthField = _self;
60304                                    // _this.monthField.set  today
60305                                 },
60306                                 select : function (combo, date)
60307                                 {
60308                                     _this.grid.ds.load({});
60309                                 }
60310                             },
60311                             value : (function() { return new Date(); })()
60312                         },
60313                         {
60314                             xtype: 'Separator',
60315                             xns: Roo.Toolbar
60316                         },
60317                         {
60318                             xtype: 'TextItem',
60319                             xns: Roo.Toolbar,
60320                             text : "Blue: in-active, green: in-active sup-event, red: de-active, purple: de-active sup-event"
60321                         },
60322                         {
60323                             xtype: 'Fill',
60324                             xns: Roo.Toolbar
60325                         },
60326                         {
60327                             xtype: 'Button',
60328                             xns: Roo.Toolbar,
60329                             listeners : {
60330                                 click : function (_self, e)
60331                                 {
60332                                     var sd = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60333                                     sd.setMonth(sd.getMonth()+1);
60334                                     _this.monthField.setValue(sd.format('Y-m-d'));
60335                                     _this.grid.ds.load({});
60336                                 }
60337                             },
60338                             text : "Next"
60339                         }
60340                     ]
60341                 },
60342                  
60343             }
60344         };
60345         
60346         *//*
60347  * Based on:
60348  * Ext JS Library 1.1.1
60349  * Copyright(c) 2006-2007, Ext JS, LLC.
60350  *
60351  * Originally Released Under LGPL - original licence link has changed is not relivant.
60352  *
60353  * Fork - LGPL
60354  * <script type="text/javascript">
60355  */
60356  
60357 /**
60358  * @class Roo.LoadMask
60359  * A simple utility class for generically masking elements while loading data.  If the element being masked has
60360  * an underlying {@link Roo.data.Store}, the masking will be automatically synchronized with the store's loading
60361  * process and the mask element will be cached for reuse.  For all other elements, this mask will replace the
60362  * element's UpdateManager load indicator and will be destroyed after the initial load.
60363  * @constructor
60364  * Create a new LoadMask
60365  * @param {String/HTMLElement/Roo.Element} el The element or DOM node, or its id
60366  * @param {Object} config The config object
60367  */
60368 Roo.LoadMask = function(el, config){
60369     this.el = Roo.get(el);
60370     Roo.apply(this, config);
60371     if(this.store){
60372         this.store.on('beforeload', this.onBeforeLoad, this);
60373         this.store.on('load', this.onLoad, this);
60374         this.store.on('loadexception', this.onLoadException, this);
60375         this.removeMask = false;
60376     }else{
60377         var um = this.el.getUpdateManager();
60378         um.showLoadIndicator = false; // disable the default indicator
60379         um.on('beforeupdate', this.onBeforeLoad, this);
60380         um.on('update', this.onLoad, this);
60381         um.on('failure', this.onLoad, this);
60382         this.removeMask = true;
60383     }
60384 };
60385
60386 Roo.LoadMask.prototype = {
60387     /**
60388      * @cfg {Boolean} removeMask
60389      * True to create a single-use mask that is automatically destroyed after loading (useful for page loads),
60390      * False to persist the mask element reference for multiple uses (e.g., for paged data widgets).  Defaults to false.
60391      */
60392     /**
60393      * @cfg {String} msg
60394      * The text to display in a centered loading message box (defaults to 'Loading...')
60395      */
60396     msg : 'Loading...',
60397     /**
60398      * @cfg {String} msgCls
60399      * The CSS class to apply to the loading message element (defaults to "x-mask-loading")
60400      */
60401     msgCls : 'x-mask-loading',
60402
60403     /**
60404      * Read-only. True if the mask is currently disabled so that it will not be displayed (defaults to false)
60405      * @type Boolean
60406      */
60407     disabled: false,
60408
60409     /**
60410      * Disables the mask to prevent it from being displayed
60411      */
60412     disable : function(){
60413        this.disabled = true;
60414     },
60415
60416     /**
60417      * Enables the mask so that it can be displayed
60418      */
60419     enable : function(){
60420         this.disabled = false;
60421     },
60422     
60423     onLoadException : function()
60424     {
60425         Roo.log(arguments);
60426         
60427         if (typeof(arguments[3]) != 'undefined') {
60428             Roo.MessageBox.alert("Error loading",arguments[3]);
60429         } 
60430         /*
60431         try {
60432             if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
60433                 Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
60434             }   
60435         } catch(e) {
60436             
60437         }
60438         */
60439     
60440         (function() { this.el.unmask(this.removeMask); }).defer(50, this);
60441     },
60442     // private
60443     onLoad : function()
60444     {
60445         (function() { this.el.unmask(this.removeMask); }).defer(50, this);
60446     },
60447
60448     // private
60449     onBeforeLoad : function(){
60450         if(!this.disabled){
60451             (function() { this.el.mask(this.msg, this.msgCls); }).defer(50, this);
60452         }
60453     },
60454
60455     // private
60456     destroy : function(){
60457         if(this.store){
60458             this.store.un('beforeload', this.onBeforeLoad, this);
60459             this.store.un('load', this.onLoad, this);
60460             this.store.un('loadexception', this.onLoadException, this);
60461         }else{
60462             var um = this.el.getUpdateManager();
60463             um.un('beforeupdate', this.onBeforeLoad, this);
60464             um.un('update', this.onLoad, this);
60465             um.un('failure', this.onLoad, this);
60466         }
60467     }
60468 };/*
60469  * Based on:
60470  * Ext JS Library 1.1.1
60471  * Copyright(c) 2006-2007, Ext JS, LLC.
60472  *
60473  * Originally Released Under LGPL - original licence link has changed is not relivant.
60474  *
60475  * Fork - LGPL
60476  * <script type="text/javascript">
60477  */
60478
60479
60480 /**
60481  * @class Roo.XTemplate
60482  * @extends Roo.Template
60483  * Provides a template that can have nested templates for loops or conditionals. The syntax is:
60484 <pre><code>
60485 var t = new Roo.XTemplate(
60486         '&lt;select name="{name}"&gt;',
60487                 '&lt;tpl for="options"&gt;&lt;option value="{value:trim}"&gt;{text:ellipsis(10)}&lt;/option&gt;&lt;/tpl&gt;',
60488         '&lt;/select&gt;'
60489 );
60490  
60491 // then append, applying the master template values
60492  </code></pre>
60493  *
60494  * Supported features:
60495  *
60496  *  Tags:
60497
60498 <pre><code>
60499       {a_variable} - output encoded.
60500       {a_variable.format:("Y-m-d")} - call a method on the variable
60501       {a_variable:raw} - unencoded output
60502       {a_variable:toFixed(1,2)} - Roo.util.Format."toFixed"
60503       {a_variable:this.method_on_template(...)} - call a method on the template object.
60504  
60505 </code></pre>
60506  *  The tpl tag:
60507 <pre><code>
60508         &lt;tpl for="a_variable or condition.."&gt;&lt;/tpl&gt;
60509         &lt;tpl if="a_variable or condition"&gt;&lt;/tpl&gt;
60510         &lt;tpl exec="some javascript"&gt;&lt;/tpl&gt;
60511         &lt;tpl name="named_template"&gt;&lt;/tpl&gt; (experimental)
60512   
60513         &lt;tpl for="."&gt;&lt;/tpl&gt; - just iterate the property..
60514         &lt;tpl for=".."&gt;&lt;/tpl&gt; - iterates with the parent (probably the template) 
60515 </code></pre>
60516  *      
60517  */
60518 Roo.XTemplate = function()
60519 {
60520     Roo.XTemplate.superclass.constructor.apply(this, arguments);
60521     if (this.html) {
60522         this.compile();
60523     }
60524 };
60525
60526
60527 Roo.extend(Roo.XTemplate, Roo.Template, {
60528
60529     /**
60530      * The various sub templates
60531      */
60532     tpls : false,
60533     /**
60534      *
60535      * basic tag replacing syntax
60536      * WORD:WORD()
60537      *
60538      * // you can fake an object call by doing this
60539      *  x.t:(test,tesT) 
60540      * 
60541      */
60542     re : /\{([\w-\.]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
60543
60544     /**
60545      * compile the template
60546      *
60547      * This is not recursive, so I'm not sure how nested templates are really going to be handled..
60548      *
60549      */
60550     compile: function()
60551     {
60552         var s = this.html;
60553      
60554         s = ['<tpl>', s, '</tpl>'].join('');
60555     
60556         var re     = /<tpl\b[^>]*>((?:(?=([^<]+))\2|<(?!tpl\b[^>]*>))*?)<\/tpl>/,
60557             nameRe = /^<tpl\b[^>]*?for="(.*?)"/,
60558             ifRe   = /^<tpl\b[^>]*?if="(.*?)"/,
60559             execRe = /^<tpl\b[^>]*?exec="(.*?)"/,
60560             namedRe = /^<tpl\b[^>]*?name="(\w+)"/,  // named templates..
60561             m,
60562             id     = 0,
60563             tpls   = [];
60564     
60565         while(true == !!(m = s.match(re))){
60566             var forMatch   = m[0].match(nameRe),
60567                 ifMatch   = m[0].match(ifRe),
60568                 execMatch   = m[0].match(execRe),
60569                 namedMatch   = m[0].match(namedRe),
60570                 
60571                 exp  = null, 
60572                 fn   = null,
60573                 exec = null,
60574                 name = forMatch && forMatch[1] ? forMatch[1] : '';
60575                 
60576             if (ifMatch) {
60577                 // if - puts fn into test..
60578                 exp = ifMatch && ifMatch[1] ? ifMatch[1] : null;
60579                 if(exp){
60580                    fn = new Function('values', 'parent', 'with(values){ return '+(Roo.util.Format.htmlDecode(exp))+'; }');
60581                 }
60582             }
60583             
60584             if (execMatch) {
60585                 // exec - calls a function... returns empty if true is  returned.
60586                 exp = execMatch && execMatch[1] ? execMatch[1] : null;
60587                 if(exp){
60588                    exec = new Function('values', 'parent', 'with(values){ '+(Roo.util.Format.htmlDecode(exp))+'; }');
60589                 }
60590             }
60591             
60592             
60593             if (name) {
60594                 // for = 
60595                 switch(name){
60596                     case '.':  name = new Function('values', 'parent', 'with(values){ return values; }'); break;
60597                     case '..': name = new Function('values', 'parent', 'with(values){ return parent; }'); break;
60598                     default:   name = new Function('values', 'parent', 'with(values){ return '+name+'; }');
60599                 }
60600             }
60601             var uid = namedMatch ? namedMatch[1] : id;
60602             
60603             
60604             tpls.push({
60605                 id:     namedMatch ? namedMatch[1] : id,
60606                 target: name,
60607                 exec:   exec,
60608                 test:   fn,
60609                 body:   m[1] || ''
60610             });
60611             if (namedMatch) {
60612                 s = s.replace(m[0], '');
60613             } else { 
60614                 s = s.replace(m[0], '{xtpl'+ id + '}');
60615             }
60616             ++id;
60617         }
60618         this.tpls = [];
60619         for(var i = tpls.length-1; i >= 0; --i){
60620             this.compileTpl(tpls[i]);
60621             this.tpls[tpls[i].id] = tpls[i];
60622         }
60623         this.master = tpls[tpls.length-1];
60624         return this;
60625     },
60626     /**
60627      * same as applyTemplate, except it's done to one of the subTemplates
60628      * when using named templates, you can do:
60629      *
60630      * var str = pl.applySubTemplate('your-name', values);
60631      *
60632      * 
60633      * @param {Number} id of the template
60634      * @param {Object} values to apply to template
60635      * @param {Object} parent (normaly the instance of this object)
60636      */
60637     applySubTemplate : function(id, values, parent)
60638     {
60639         
60640         
60641         var t = this.tpls[id];
60642         
60643         
60644         try { 
60645             if(t.test && !t.test.call(this, values, parent)){
60646                 return '';
60647             }
60648         } catch(e) {
60649             Roo.log("Xtemplate.applySubTemplate 'test': Exception thrown");
60650             Roo.log(e.toString());
60651             Roo.log(t.test);
60652             return ''
60653         }
60654         try { 
60655             
60656             if(t.exec && t.exec.call(this, values, parent)){
60657                 return '';
60658             }
60659         } catch(e) {
60660             Roo.log("Xtemplate.applySubTemplate 'exec': Exception thrown");
60661             Roo.log(e.toString());
60662             Roo.log(t.exec);
60663             return ''
60664         }
60665         try {
60666             var vs = t.target ? t.target.call(this, values, parent) : values;
60667             parent = t.target ? values : parent;
60668             if(t.target && vs instanceof Array){
60669                 var buf = [];
60670                 for(var i = 0, len = vs.length; i < len; i++){
60671                     buf[buf.length] = t.compiled.call(this, vs[i], parent);
60672                 }
60673                 return buf.join('');
60674             }
60675             return t.compiled.call(this, vs, parent);
60676         } catch (e) {
60677             Roo.log("Xtemplate.applySubTemplate : Exception thrown");
60678             Roo.log(e.toString());
60679             Roo.log(t.compiled);
60680             return '';
60681         }
60682     },
60683
60684     compileTpl : function(tpl)
60685     {
60686         var fm = Roo.util.Format;
60687         var useF = this.disableFormats !== true;
60688         var sep = Roo.isGecko ? "+" : ",";
60689         var undef = function(str) {
60690             Roo.log("Property not found :"  + str);
60691             return '';
60692         };
60693         
60694         var fn = function(m, name, format, args)
60695         {
60696             //Roo.log(arguments);
60697             args = args ? args.replace(/\\'/g,"'") : args;
60698             //["{TEST:(a,b,c)}", "TEST", "", "a,b,c", 0, "{TEST:(a,b,c)}"]
60699             if (typeof(format) == 'undefined') {
60700                 format= 'htmlEncode';
60701             }
60702             if (format == 'raw' ) {
60703                 format = false;
60704             }
60705             
60706             if(name.substr(0, 4) == 'xtpl'){
60707                 return "'"+ sep +'this.applySubTemplate('+name.substr(4)+', values, parent)'+sep+"'";
60708             }
60709             
60710             // build an array of options to determine if value is undefined..
60711             
60712             // basically get 'xxxx.yyyy' then do
60713             // (typeof(xxxx) == 'undefined' || typeof(xxx.yyyy) == 'undefined') ?
60714             //    (function () { Roo.log("Property not found"); return ''; })() :
60715             //    ......
60716             
60717             var udef_ar = [];
60718             var lookfor = '';
60719             Roo.each(name.split('.'), function(st) {
60720                 lookfor += (lookfor.length ? '.': '') + st;
60721                 udef_ar.push(  "(typeof(" + lookfor + ") == 'undefined')"  );
60722             });
60723             
60724             var udef_st = '((' + udef_ar.join(" || ") +") ? undef('" + name + "') : "; // .. needs )
60725             
60726             
60727             if(format && useF){
60728                 
60729                 args = args ? ',' + args : "";
60730                  
60731                 if(format.substr(0, 5) != "this."){
60732                     format = "fm." + format + '(';
60733                 }else{
60734                     format = 'this.call("'+ format.substr(5) + '", ';
60735                     args = ", values";
60736                 }
60737                 
60738                 return "'"+ sep +   udef_st   +    format + name + args + "))"+sep+"'";
60739             }
60740              
60741             if (args.length) {
60742                 // called with xxyx.yuu:(test,test)
60743                 // change to ()
60744                 return "'"+ sep + udef_st  + name + '(' +  args + "))"+sep+"'";
60745             }
60746             // raw.. - :raw modifier..
60747             return "'"+ sep + udef_st  + name + ")"+sep+"'";
60748             
60749         };
60750         var body;
60751         // branched to use + in gecko and [].join() in others
60752         if(Roo.isGecko){
60753             body = "tpl.compiled = function(values, parent){  with(values) { return '" +
60754                    tpl.body.replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
60755                     "';};};";
60756         }else{
60757             body = ["tpl.compiled = function(values, parent){  with (values) { return ['"];
60758             body.push(tpl.body.replace(/(\r\n|\n)/g,
60759                             '\\n').replace(/'/g, "\\'").replace(this.re, fn));
60760             body.push("'].join('');};};");
60761             body = body.join('');
60762         }
60763         
60764         Roo.debug && Roo.log(body.replace(/\\n/,'\n'));
60765        
60766         /** eval:var:tpl eval:var:fm eval:var:useF eval:var:undef  */
60767         eval(body);
60768         
60769         return this;
60770     },
60771
60772     applyTemplate : function(values){
60773         return this.master.compiled.call(this, values, {});
60774         //var s = this.subs;
60775     },
60776
60777     apply : function(){
60778         return this.applyTemplate.apply(this, arguments);
60779     }
60780
60781  });
60782
60783 Roo.XTemplate.from = function(el){
60784     el = Roo.getDom(el);
60785     return new Roo.XTemplate(el.value || el.innerHTML);
60786 };