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         Roo.log(["applyTemplate", values]);
4671         try {
4672            
4673             if(this.compiled){
4674                 return this.compiled(values);
4675             }
4676             var useF = this.disableFormats !== true;
4677             var fm = Roo.util.Format, tpl = this;
4678             var fn = function(m, name, format, args){
4679                 if(format && useF){
4680                     if(format.substr(0, 5) == "this."){
4681                         return tpl.call(format.substr(5), values[name], values);
4682                     }else{
4683                         if(args){
4684                             // quoted values are required for strings in compiled templates, 
4685                             // but for non compiled we need to strip them
4686                             // quoted reversed for jsmin
4687                             var re = /^\s*['"](.*)["']\s*$/;
4688                             args = args.split(',');
4689                             for(var i = 0, len = args.length; i < len; i++){
4690                                 args[i] = args[i].replace(re, "$1");
4691                             }
4692                             args = [values[name]].concat(args);
4693                         }else{
4694                             args = [values[name]];
4695                         }
4696                         return fm[format].apply(fm, args);
4697                     }
4698                 }else{
4699                     return values[name] !== undefined ? values[name] : "";
4700                 }
4701             };
4702             return this.html.replace(this.re, fn);
4703         } catch (e) {
4704             Roo.log(e);
4705             throw e;
4706         }
4707          
4708     },
4709     
4710     loading : false,
4711       
4712     load : function ()
4713     {
4714          
4715         if (this.loading) {
4716             return;
4717         }
4718         var _t = this;
4719         
4720         this.loading = true;
4721         this.compiled = false;
4722         
4723         var cx = new Roo.data.Connection();
4724         cx.request({
4725             url : this.url,
4726             method : 'GET',
4727             success : function (response) {
4728                 _t.loading = false;
4729                 _t.html = response.responseText;
4730                 _t.url = false;
4731                 _t.compile();
4732              },
4733             failure : function(response) {
4734                 Roo.log("Template failed to load from " + _t.url);
4735                 _t.loading = false;
4736             }
4737         });
4738     },
4739
4740     /**
4741      * Sets the HTML used as the template and optionally compiles it.
4742      * @param {String} html
4743      * @param {Boolean} compile (optional) True to compile the template (defaults to undefined)
4744      * @return {Roo.Template} this
4745      */
4746     set : function(html, compile){
4747         this.html = html;
4748         this.compiled = null;
4749         if(compile){
4750             this.compile();
4751         }
4752         return this;
4753     },
4754     
4755     /**
4756      * True to disable format functions (defaults to false)
4757      * @type Boolean
4758      */
4759     disableFormats : false,
4760     
4761     /**
4762     * The regular expression used to match template variables 
4763     * @type RegExp
4764     * @property 
4765     */
4766     re : /\{([\w-]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
4767     
4768     /**
4769      * Compiles the template into an internal function, eliminating the RegEx overhead.
4770      * @return {Roo.Template} this
4771      */
4772     compile : function(){
4773         var fm = Roo.util.Format;
4774         var useF = this.disableFormats !== true;
4775         var sep = Roo.isGecko ? "+" : ",";
4776         var fn = function(m, name, format, args){
4777             if(format && useF){
4778                 args = args ? ',' + args : "";
4779                 if(format.substr(0, 5) != "this."){
4780                     format = "fm." + format + '(';
4781                 }else{
4782                     format = 'this.call("'+ format.substr(5) + '", ';
4783                     args = ", values";
4784                 }
4785             }else{
4786                 args= ''; format = "(values['" + name + "'] == undefined ? '' : ";
4787             }
4788             return "'"+ sep + format + "values['" + name + "']" + args + ")"+sep+"'";
4789         };
4790         var body;
4791         // branched to use + in gecko and [].join() in others
4792         if(Roo.isGecko){
4793             body = "this.compiled = function(values){ return '" +
4794                    this.html.replace(/\\/g, '\\\\').replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
4795                     "';};";
4796         }else{
4797             body = ["this.compiled = function(values){ return ['"];
4798             body.push(this.html.replace(/\\/g, '\\\\').replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn));
4799             body.push("'].join('');};");
4800             body = body.join('');
4801         }
4802         /**
4803          * eval:var:values
4804          * eval:var:fm
4805          */
4806         eval(body);
4807         return this;
4808     },
4809     
4810     // private function used to call members
4811     call : function(fnName, value, allValues){
4812         return this[fnName](value, allValues);
4813     },
4814     
4815     /**
4816      * Applies the supplied values to the template and inserts the new node(s) as the first child of el.
4817      * @param {String/HTMLElement/Roo.Element} el The context element
4818      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4819      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4820      * @return {HTMLElement/Roo.Element} The new node or Element
4821      */
4822     insertFirst: function(el, values, returnElement){
4823         return this.doInsert('afterBegin', el, values, returnElement);
4824     },
4825
4826     /**
4827      * Applies the supplied values to the template and inserts the new node(s) before el.
4828      * @param {String/HTMLElement/Roo.Element} el The context element
4829      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4830      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4831      * @return {HTMLElement/Roo.Element} The new node or Element
4832      */
4833     insertBefore: function(el, values, returnElement){
4834         return this.doInsert('beforeBegin', el, values, returnElement);
4835     },
4836
4837     /**
4838      * Applies the supplied values to the template and inserts the new node(s) after el.
4839      * @param {String/HTMLElement/Roo.Element} el The context element
4840      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4841      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4842      * @return {HTMLElement/Roo.Element} The new node or Element
4843      */
4844     insertAfter : function(el, values, returnElement){
4845         return this.doInsert('afterEnd', el, values, returnElement);
4846     },
4847     
4848     /**
4849      * Applies the supplied values to the template and appends the new node(s) to el.
4850      * @param {String/HTMLElement/Roo.Element} el The context element
4851      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4852      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4853      * @return {HTMLElement/Roo.Element} The new node or Element
4854      */
4855     append : function(el, values, returnElement){
4856         return this.doInsert('beforeEnd', el, values, returnElement);
4857     },
4858
4859     doInsert : function(where, el, values, returnEl){
4860         el = Roo.getDom(el);
4861         var newNode = Roo.DomHelper.insertHtml(where, el, this.applyTemplate(values));
4862         return returnEl ? Roo.get(newNode, true) : newNode;
4863     },
4864
4865     /**
4866      * Applies the supplied values to the template and overwrites the content of el with the new node(s).
4867      * @param {String/HTMLElement/Roo.Element} el The context element
4868      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4869      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4870      * @return {HTMLElement/Roo.Element} The new node or Element
4871      */
4872     overwrite : function(el, values, returnElement){
4873         el = Roo.getDom(el);
4874         el.innerHTML = this.applyTemplate(values);
4875         return returnElement ? Roo.get(el.firstChild, true) : el.firstChild;
4876     }
4877 };
4878 /**
4879  * Alias for {@link #applyTemplate}
4880  * @method
4881  */
4882 Roo.Template.prototype.apply = Roo.Template.prototype.applyTemplate;
4883
4884 // backwards compat
4885 Roo.DomHelper.Template = Roo.Template;
4886
4887 /**
4888  * Creates a template from the passed element's value (<i>display:none</i> textarea, preferred) or innerHTML.
4889  * @param {String/HTMLElement} el A DOM element or its id
4890  * @returns {Roo.Template} The created template
4891  * @static
4892  */
4893 Roo.Template.from = function(el){
4894     el = Roo.getDom(el);
4895     return new Roo.Template(el.value || el.innerHTML);
4896 };/*
4897  * Based on:
4898  * Ext JS Library 1.1.1
4899  * Copyright(c) 2006-2007, Ext JS, LLC.
4900  *
4901  * Originally Released Under LGPL - original licence link has changed is not relivant.
4902  *
4903  * Fork - LGPL
4904  * <script type="text/javascript">
4905  */
4906  
4907
4908 /*
4909  * This is code is also distributed under MIT license for use
4910  * with jQuery and prototype JavaScript libraries.
4911  */
4912 /**
4913  * @class Roo.DomQuery
4914 Provides high performance selector/xpath processing by compiling queries into reusable functions. New pseudo classes and matchers can be plugged. It works on HTML and XML documents (if a content node is passed in).
4915 <p>
4916 DomQuery supports most of the <a href="http://www.w3.org/TR/2005/WD-css3-selectors-20051215/">CSS3 selectors spec</a>, along with some custom selectors and basic XPath.</p>
4917
4918 <p>
4919 All selectors, attribute filters and pseudos below can be combined infinitely in any order. For example "div.foo:nth-child(odd)[@foo=bar].bar:first" would be a perfectly valid selector. Node filters are processed in the order in which they appear, which allows you to optimize your queries for your document structure.
4920 </p>
4921 <h4>Element Selectors:</h4>
4922 <ul class="list">
4923     <li> <b>*</b> any element</li>
4924     <li> <b>E</b> an element with the tag E</li>
4925     <li> <b>E F</b> All descendent elements of E that have the tag F</li>
4926     <li> <b>E > F</b> or <b>E/F</b> all direct children elements of E that have the tag F</li>
4927     <li> <b>E + F</b> all elements with the tag F that are immediately preceded by an element with the tag E</li>
4928     <li> <b>E ~ F</b> all elements with the tag F that are preceded by a sibling element with the tag E</li>
4929 </ul>
4930 <h4>Attribute Selectors:</h4>
4931 <p>The use of @ and quotes are optional. For example, div[@foo='bar'] is also a valid attribute selector.</p>
4932 <ul class="list">
4933     <li> <b>E[foo]</b> has an attribute "foo"</li>
4934     <li> <b>E[foo=bar]</b> has an attribute "foo" that equals "bar"</li>
4935     <li> <b>E[foo^=bar]</b> has an attribute "foo" that starts with "bar"</li>
4936     <li> <b>E[foo$=bar]</b> has an attribute "foo" that ends with "bar"</li>
4937     <li> <b>E[foo*=bar]</b> has an attribute "foo" that contains the substring "bar"</li>
4938     <li> <b>E[foo%=2]</b> has an attribute "foo" that is evenly divisible by 2</li>
4939     <li> <b>E[foo!=bar]</b> has an attribute "foo" that does not equal "bar"</li>
4940 </ul>
4941 <h4>Pseudo Classes:</h4>
4942 <ul class="list">
4943     <li> <b>E:first-child</b> E is the first child of its parent</li>
4944     <li> <b>E:last-child</b> E is the last child of its parent</li>
4945     <li> <b>E:nth-child(<i>n</i>)</b> E is the <i>n</i>th child of its parent (1 based as per the spec)</li>
4946     <li> <b>E:nth-child(odd)</b> E is an odd child of its parent</li>
4947     <li> <b>E:nth-child(even)</b> E is an even child of its parent</li>
4948     <li> <b>E:only-child</b> E is the only child of its parent</li>
4949     <li> <b>E:checked</b> E is an element that is has a checked attribute that is true (e.g. a radio or checkbox) </li>
4950     <li> <b>E:first</b> the first E in the resultset</li>
4951     <li> <b>E:last</b> the last E in the resultset</li>
4952     <li> <b>E:nth(<i>n</i>)</b> the <i>n</i>th E in the resultset (1 based)</li>
4953     <li> <b>E:odd</b> shortcut for :nth-child(odd)</li>
4954     <li> <b>E:even</b> shortcut for :nth-child(even)</li>
4955     <li> <b>E:contains(foo)</b> E's innerHTML contains the substring "foo"</li>
4956     <li> <b>E:nodeValue(foo)</b> E contains a textNode with a nodeValue that equals "foo"</li>
4957     <li> <b>E:not(S)</b> an E element that does not match simple selector S</li>
4958     <li> <b>E:has(S)</b> an E element that has a descendent that matches simple selector S</li>
4959     <li> <b>E:next(S)</b> an E element whose next sibling matches simple selector S</li>
4960     <li> <b>E:prev(S)</b> an E element whose previous sibling matches simple selector S</li>
4961 </ul>
4962 <h4>CSS Value Selectors:</h4>
4963 <ul class="list">
4964     <li> <b>E{display=none}</b> css value "display" that equals "none"</li>
4965     <li> <b>E{display^=none}</b> css value "display" that starts with "none"</li>
4966     <li> <b>E{display$=none}</b> css value "display" that ends with "none"</li>
4967     <li> <b>E{display*=none}</b> css value "display" that contains the substring "none"</li>
4968     <li> <b>E{display%=2}</b> css value "display" that is evenly divisible by 2</li>
4969     <li> <b>E{display!=none}</b> css value "display" that does not equal "none"</li>
4970 </ul>
4971  * @singleton
4972  */
4973 Roo.DomQuery = function(){
4974     var cache = {}, simpleCache = {}, valueCache = {};
4975     var nonSpace = /\S/;
4976     var trimRe = /^\s+|\s+$/g;
4977     var tplRe = /\{(\d+)\}/g;
4978     var modeRe = /^(\s?[\/>+~]\s?|\s|$)/;
4979     var tagTokenRe = /^(#)?([\w-\*]+)/;
4980     var nthRe = /(\d*)n\+?(\d*)/, nthRe2 = /\D/;
4981
4982     function child(p, index){
4983         var i = 0;
4984         var n = p.firstChild;
4985         while(n){
4986             if(n.nodeType == 1){
4987                if(++i == index){
4988                    return n;
4989                }
4990             }
4991             n = n.nextSibling;
4992         }
4993         return null;
4994     };
4995
4996     function next(n){
4997         while((n = n.nextSibling) && n.nodeType != 1);
4998         return n;
4999     };
5000
5001     function prev(n){
5002         while((n = n.previousSibling) && n.nodeType != 1);
5003         return n;
5004     };
5005
5006     function children(d){
5007         var n = d.firstChild, ni = -1;
5008             while(n){
5009                 var nx = n.nextSibling;
5010                 if(n.nodeType == 3 && !nonSpace.test(n.nodeValue)){
5011                     d.removeChild(n);
5012                 }else{
5013                     n.nodeIndex = ++ni;
5014                 }
5015                 n = nx;
5016             }
5017             return this;
5018         };
5019
5020     function byClassName(c, a, v){
5021         if(!v){
5022             return c;
5023         }
5024         var r = [], ri = -1, cn;
5025         for(var i = 0, ci; ci = c[i]; i++){
5026             if((' '+ci.className+' ').indexOf(v) != -1){
5027                 r[++ri] = ci;
5028             }
5029         }
5030         return r;
5031     };
5032
5033     function attrValue(n, attr){
5034         if(!n.tagName && typeof n.length != "undefined"){
5035             n = n[0];
5036         }
5037         if(!n){
5038             return null;
5039         }
5040         if(attr == "for"){
5041             return n.htmlFor;
5042         }
5043         if(attr == "class" || attr == "className"){
5044             return n.className;
5045         }
5046         return n.getAttribute(attr) || n[attr];
5047
5048     };
5049
5050     function getNodes(ns, mode, tagName){
5051         var result = [], ri = -1, cs;
5052         if(!ns){
5053             return result;
5054         }
5055         tagName = tagName || "*";
5056         if(typeof ns.getElementsByTagName != "undefined"){
5057             ns = [ns];
5058         }
5059         if(!mode){
5060             for(var i = 0, ni; ni = ns[i]; i++){
5061                 cs = ni.getElementsByTagName(tagName);
5062                 for(var j = 0, ci; ci = cs[j]; j++){
5063                     result[++ri] = ci;
5064                 }
5065             }
5066         }else if(mode == "/" || mode == ">"){
5067             var utag = tagName.toUpperCase();
5068             for(var i = 0, ni, cn; ni = ns[i]; i++){
5069                 cn = ni.children || ni.childNodes;
5070                 for(var j = 0, cj; cj = cn[j]; j++){
5071                     if(cj.nodeName == utag || cj.nodeName == tagName  || tagName == '*'){
5072                         result[++ri] = cj;
5073                     }
5074                 }
5075             }
5076         }else if(mode == "+"){
5077             var utag = tagName.toUpperCase();
5078             for(var i = 0, n; n = ns[i]; i++){
5079                 while((n = n.nextSibling) && n.nodeType != 1);
5080                 if(n && (n.nodeName == utag || n.nodeName == tagName || tagName == '*')){
5081                     result[++ri] = n;
5082                 }
5083             }
5084         }else if(mode == "~"){
5085             for(var i = 0, n; n = ns[i]; i++){
5086                 while((n = n.nextSibling) && (n.nodeType != 1 || (tagName == '*' || n.tagName.toLowerCase()!=tagName)));
5087                 if(n){
5088                     result[++ri] = n;
5089                 }
5090             }
5091         }
5092         return result;
5093     };
5094
5095     function concat(a, b){
5096         if(b.slice){
5097             return a.concat(b);
5098         }
5099         for(var i = 0, l = b.length; i < l; i++){
5100             a[a.length] = b[i];
5101         }
5102         return a;
5103     }
5104
5105     function byTag(cs, tagName){
5106         if(cs.tagName || cs == document){
5107             cs = [cs];
5108         }
5109         if(!tagName){
5110             return cs;
5111         }
5112         var r = [], ri = -1;
5113         tagName = tagName.toLowerCase();
5114         for(var i = 0, ci; ci = cs[i]; i++){
5115             if(ci.nodeType == 1 && ci.tagName.toLowerCase()==tagName){
5116                 r[++ri] = ci;
5117             }
5118         }
5119         return r;
5120     };
5121
5122     function byId(cs, attr, id){
5123         if(cs.tagName || cs == document){
5124             cs = [cs];
5125         }
5126         if(!id){
5127             return cs;
5128         }
5129         var r = [], ri = -1;
5130         for(var i = 0,ci; ci = cs[i]; i++){
5131             if(ci && ci.id == id){
5132                 r[++ri] = ci;
5133                 return r;
5134             }
5135         }
5136         return r;
5137     };
5138
5139     function byAttribute(cs, attr, value, op, custom){
5140         var r = [], ri = -1, st = custom=="{";
5141         var f = Roo.DomQuery.operators[op];
5142         for(var i = 0, ci; ci = cs[i]; i++){
5143             var a;
5144             if(st){
5145                 a = Roo.DomQuery.getStyle(ci, attr);
5146             }
5147             else if(attr == "class" || attr == "className"){
5148                 a = ci.className;
5149             }else if(attr == "for"){
5150                 a = ci.htmlFor;
5151             }else if(attr == "href"){
5152                 a = ci.getAttribute("href", 2);
5153             }else{
5154                 a = ci.getAttribute(attr);
5155             }
5156             if((f && f(a, value)) || (!f && a)){
5157                 r[++ri] = ci;
5158             }
5159         }
5160         return r;
5161     };
5162
5163     function byPseudo(cs, name, value){
5164         return Roo.DomQuery.pseudos[name](cs, value);
5165     };
5166
5167     // This is for IE MSXML which does not support expandos.
5168     // IE runs the same speed using setAttribute, however FF slows way down
5169     // and Safari completely fails so they need to continue to use expandos.
5170     var isIE = window.ActiveXObject ? true : false;
5171
5172     // this eval is stop the compressor from
5173     // renaming the variable to something shorter
5174     
5175     /** eval:var:batch */
5176     var batch = 30803; 
5177
5178     var key = 30803;
5179
5180     function nodupIEXml(cs){
5181         var d = ++key;
5182         cs[0].setAttribute("_nodup", d);
5183         var r = [cs[0]];
5184         for(var i = 1, len = cs.length; i < len; i++){
5185             var c = cs[i];
5186             if(!c.getAttribute("_nodup") != d){
5187                 c.setAttribute("_nodup", d);
5188                 r[r.length] = c;
5189             }
5190         }
5191         for(var i = 0, len = cs.length; i < len; i++){
5192             cs[i].removeAttribute("_nodup");
5193         }
5194         return r;
5195     }
5196
5197     function nodup(cs){
5198         if(!cs){
5199             return [];
5200         }
5201         var len = cs.length, c, i, r = cs, cj, ri = -1;
5202         if(!len || typeof cs.nodeType != "undefined" || len == 1){
5203             return cs;
5204         }
5205         if(isIE && typeof cs[0].selectSingleNode != "undefined"){
5206             return nodupIEXml(cs);
5207         }
5208         var d = ++key;
5209         cs[0]._nodup = d;
5210         for(i = 1; c = cs[i]; i++){
5211             if(c._nodup != d){
5212                 c._nodup = d;
5213             }else{
5214                 r = [];
5215                 for(var j = 0; j < i; j++){
5216                     r[++ri] = cs[j];
5217                 }
5218                 for(j = i+1; cj = cs[j]; j++){
5219                     if(cj._nodup != d){
5220                         cj._nodup = d;
5221                         r[++ri] = cj;
5222                     }
5223                 }
5224                 return r;
5225             }
5226         }
5227         return r;
5228     }
5229
5230     function quickDiffIEXml(c1, c2){
5231         var d = ++key;
5232         for(var i = 0, len = c1.length; i < len; i++){
5233             c1[i].setAttribute("_qdiff", d);
5234         }
5235         var r = [];
5236         for(var i = 0, len = c2.length; i < len; i++){
5237             if(c2[i].getAttribute("_qdiff") != d){
5238                 r[r.length] = c2[i];
5239             }
5240         }
5241         for(var i = 0, len = c1.length; i < len; i++){
5242            c1[i].removeAttribute("_qdiff");
5243         }
5244         return r;
5245     }
5246
5247     function quickDiff(c1, c2){
5248         var len1 = c1.length;
5249         if(!len1){
5250             return c2;
5251         }
5252         if(isIE && c1[0].selectSingleNode){
5253             return quickDiffIEXml(c1, c2);
5254         }
5255         var d = ++key;
5256         for(var i = 0; i < len1; i++){
5257             c1[i]._qdiff = d;
5258         }
5259         var r = [];
5260         for(var i = 0, len = c2.length; i < len; i++){
5261             if(c2[i]._qdiff != d){
5262                 r[r.length] = c2[i];
5263             }
5264         }
5265         return r;
5266     }
5267
5268     function quickId(ns, mode, root, id){
5269         if(ns == root){
5270            var d = root.ownerDocument || root;
5271            return d.getElementById(id);
5272         }
5273         ns = getNodes(ns, mode, "*");
5274         return byId(ns, null, id);
5275     }
5276
5277     return {
5278         getStyle : function(el, name){
5279             return Roo.fly(el).getStyle(name);
5280         },
5281         /**
5282          * Compiles a selector/xpath query into a reusable function. The returned function
5283          * takes one parameter "root" (optional), which is the context node from where the query should start.
5284          * @param {String} selector The selector/xpath query
5285          * @param {String} type (optional) Either "select" (the default) or "simple" for a simple selector match
5286          * @return {Function}
5287          */
5288         compile : function(path, type){
5289             type = type || "select";
5290             
5291             var fn = ["var f = function(root){\n var mode; ++batch; var n = root || document;\n"];
5292             var q = path, mode, lq;
5293             var tk = Roo.DomQuery.matchers;
5294             var tklen = tk.length;
5295             var mm;
5296
5297             // accept leading mode switch
5298             var lmode = q.match(modeRe);
5299             if(lmode && lmode[1]){
5300                 fn[fn.length] = 'mode="'+lmode[1].replace(trimRe, "")+'";';
5301                 q = q.replace(lmode[1], "");
5302             }
5303             // strip leading slashes
5304             while(path.substr(0, 1)=="/"){
5305                 path = path.substr(1);
5306             }
5307
5308             while(q && lq != q){
5309                 lq = q;
5310                 var tm = q.match(tagTokenRe);
5311                 if(type == "select"){
5312                     if(tm){
5313                         if(tm[1] == "#"){
5314                             fn[fn.length] = 'n = quickId(n, mode, root, "'+tm[2]+'");';
5315                         }else{
5316                             fn[fn.length] = 'n = getNodes(n, mode, "'+tm[2]+'");';
5317                         }
5318                         q = q.replace(tm[0], "");
5319                     }else if(q.substr(0, 1) != '@'){
5320                         fn[fn.length] = 'n = getNodes(n, mode, "*");';
5321                     }
5322                 }else{
5323                     if(tm){
5324                         if(tm[1] == "#"){
5325                             fn[fn.length] = 'n = byId(n, null, "'+tm[2]+'");';
5326                         }else{
5327                             fn[fn.length] = 'n = byTag(n, "'+tm[2]+'");';
5328                         }
5329                         q = q.replace(tm[0], "");
5330                     }
5331                 }
5332                 while(!(mm = q.match(modeRe))){
5333                     var matched = false;
5334                     for(var j = 0; j < tklen; j++){
5335                         var t = tk[j];
5336                         var m = q.match(t.re);
5337                         if(m){
5338                             fn[fn.length] = t.select.replace(tplRe, function(x, i){
5339                                                     return m[i];
5340                                                 });
5341                             q = q.replace(m[0], "");
5342                             matched = true;
5343                             break;
5344                         }
5345                     }
5346                     // prevent infinite loop on bad selector
5347                     if(!matched){
5348                         throw 'Error parsing selector, parsing failed at "' + q + '"';
5349                     }
5350                 }
5351                 if(mm[1]){
5352                     fn[fn.length] = 'mode="'+mm[1].replace(trimRe, "")+'";';
5353                     q = q.replace(mm[1], "");
5354                 }
5355             }
5356             fn[fn.length] = "return nodup(n);\n}";
5357             
5358              /** 
5359               * list of variables that need from compression as they are used by eval.
5360              *  eval:var:batch 
5361              *  eval:var:nodup
5362              *  eval:var:byTag
5363              *  eval:var:ById
5364              *  eval:var:getNodes
5365              *  eval:var:quickId
5366              *  eval:var:mode
5367              *  eval:var:root
5368              *  eval:var:n
5369              *  eval:var:byClassName
5370              *  eval:var:byPseudo
5371              *  eval:var:byAttribute
5372              *  eval:var:attrValue
5373              * 
5374              **/ 
5375             eval(fn.join(""));
5376             return f;
5377         },
5378
5379         /**
5380          * Selects a group of elements.
5381          * @param {String} selector The selector/xpath query (can be a comma separated list of selectors)
5382          * @param {Node} root (optional) The start of the query (defaults to document).
5383          * @return {Array}
5384          */
5385         select : function(path, root, type){
5386             if(!root || root == document){
5387                 root = document;
5388             }
5389             if(typeof root == "string"){
5390                 root = document.getElementById(root);
5391             }
5392             var paths = path.split(",");
5393             var results = [];
5394             for(var i = 0, len = paths.length; i < len; i++){
5395                 var p = paths[i].replace(trimRe, "");
5396                 if(!cache[p]){
5397                     cache[p] = Roo.DomQuery.compile(p);
5398                     if(!cache[p]){
5399                         throw p + " is not a valid selector";
5400                     }
5401                 }
5402                 var result = cache[p](root);
5403                 if(result && result != document){
5404                     results = results.concat(result);
5405                 }
5406             }
5407             if(paths.length > 1){
5408                 return nodup(results);
5409             }
5410             return results;
5411         },
5412
5413         /**
5414          * Selects a single element.
5415          * @param {String} selector The selector/xpath query
5416          * @param {Node} root (optional) The start of the query (defaults to document).
5417          * @return {Element}
5418          */
5419         selectNode : function(path, root){
5420             return Roo.DomQuery.select(path, root)[0];
5421         },
5422
5423         /**
5424          * Selects the value of a node, optionally replacing null with the defaultValue.
5425          * @param {String} selector The selector/xpath query
5426          * @param {Node} root (optional) The start of the query (defaults to document).
5427          * @param {String} defaultValue
5428          */
5429         selectValue : function(path, root, defaultValue){
5430             path = path.replace(trimRe, "");
5431             if(!valueCache[path]){
5432                 valueCache[path] = Roo.DomQuery.compile(path, "select");
5433             }
5434             var n = valueCache[path](root);
5435             n = n[0] ? n[0] : n;
5436             var v = (n && n.firstChild ? n.firstChild.nodeValue : null);
5437             return ((v === null||v === undefined||v==='') ? defaultValue : v);
5438         },
5439
5440         /**
5441          * Selects the value of a node, parsing integers and floats.
5442          * @param {String} selector The selector/xpath query
5443          * @param {Node} root (optional) The start of the query (defaults to document).
5444          * @param {Number} defaultValue
5445          * @return {Number}
5446          */
5447         selectNumber : function(path, root, defaultValue){
5448             var v = Roo.DomQuery.selectValue(path, root, defaultValue || 0);
5449             return parseFloat(v);
5450         },
5451
5452         /**
5453          * Returns true if the passed element(s) match the passed simple selector (e.g. div.some-class or span:first-child)
5454          * @param {String/HTMLElement/Array} el An element id, element or array of elements
5455          * @param {String} selector The simple selector to test
5456          * @return {Boolean}
5457          */
5458         is : function(el, ss){
5459             if(typeof el == "string"){
5460                 el = document.getElementById(el);
5461             }
5462             var isArray = (el instanceof Array);
5463             var result = Roo.DomQuery.filter(isArray ? el : [el], ss);
5464             return isArray ? (result.length == el.length) : (result.length > 0);
5465         },
5466
5467         /**
5468          * Filters an array of elements to only include matches of a simple selector (e.g. div.some-class or span:first-child)
5469          * @param {Array} el An array of elements to filter
5470          * @param {String} selector The simple selector to test
5471          * @param {Boolean} nonMatches If true, it returns the elements that DON'T match
5472          * the selector instead of the ones that match
5473          * @return {Array}
5474          */
5475         filter : function(els, ss, nonMatches){
5476             ss = ss.replace(trimRe, "");
5477             if(!simpleCache[ss]){
5478                 simpleCache[ss] = Roo.DomQuery.compile(ss, "simple");
5479             }
5480             var result = simpleCache[ss](els);
5481             return nonMatches ? quickDiff(result, els) : result;
5482         },
5483
5484         /**
5485          * Collection of matching regular expressions and code snippets.
5486          */
5487         matchers : [{
5488                 re: /^\.([\w-]+)/,
5489                 select: 'n = byClassName(n, null, " {1} ");'
5490             }, {
5491                 re: /^\:([\w-]+)(?:\(((?:[^\s>\/]*|.*?))\))?/,
5492                 select: 'n = byPseudo(n, "{1}", "{2}");'
5493             },{
5494                 re: /^(?:([\[\{])(?:@)?([\w-]+)\s?(?:(=|.=)\s?['"]?(.*?)["']?)?[\]\}])/,
5495                 select: 'n = byAttribute(n, "{2}", "{4}", "{3}", "{1}");'
5496             }, {
5497                 re: /^#([\w-]+)/,
5498                 select: 'n = byId(n, null, "{1}");'
5499             },{
5500                 re: /^@([\w-]+)/,
5501                 select: 'return {firstChild:{nodeValue:attrValue(n, "{1}")}};'
5502             }
5503         ],
5504
5505         /**
5506          * Collection of operator comparison functions. The default operators are =, !=, ^=, $=, *=, %=, |= and ~=.
5507          * New operators can be added as long as the match the format <i>c</i>= where <i>c</i> is any character other than space, &gt; &lt;.
5508          */
5509         operators : {
5510             "=" : function(a, v){
5511                 return a == v;
5512             },
5513             "!=" : function(a, v){
5514                 return a != v;
5515             },
5516             "^=" : function(a, v){
5517                 return a && a.substr(0, v.length) == v;
5518             },
5519             "$=" : function(a, v){
5520                 return a && a.substr(a.length-v.length) == v;
5521             },
5522             "*=" : function(a, v){
5523                 return a && a.indexOf(v) !== -1;
5524             },
5525             "%=" : function(a, v){
5526                 return (a % v) == 0;
5527             },
5528             "|=" : function(a, v){
5529                 return a && (a == v || a.substr(0, v.length+1) == v+'-');
5530             },
5531             "~=" : function(a, v){
5532                 return a && (' '+a+' ').indexOf(' '+v+' ') != -1;
5533             }
5534         },
5535
5536         /**
5537          * Collection of "pseudo class" processors. Each processor is passed the current nodeset (array)
5538          * and the argument (if any) supplied in the selector.
5539          */
5540         pseudos : {
5541             "first-child" : function(c){
5542                 var r = [], ri = -1, n;
5543                 for(var i = 0, ci; ci = n = c[i]; i++){
5544                     while((n = n.previousSibling) && n.nodeType != 1);
5545                     if(!n){
5546                         r[++ri] = ci;
5547                     }
5548                 }
5549                 return r;
5550             },
5551
5552             "last-child" : function(c){
5553                 var r = [], ri = -1, n;
5554                 for(var i = 0, ci; ci = n = c[i]; i++){
5555                     while((n = n.nextSibling) && n.nodeType != 1);
5556                     if(!n){
5557                         r[++ri] = ci;
5558                     }
5559                 }
5560                 return r;
5561             },
5562
5563             "nth-child" : function(c, a) {
5564                 var r = [], ri = -1;
5565                 var m = nthRe.exec(a == "even" && "2n" || a == "odd" && "2n+1" || !nthRe2.test(a) && "n+" + a || a);
5566                 var f = (m[1] || 1) - 0, l = m[2] - 0;
5567                 for(var i = 0, n; n = c[i]; i++){
5568                     var pn = n.parentNode;
5569                     if (batch != pn._batch) {
5570                         var j = 0;
5571                         for(var cn = pn.firstChild; cn; cn = cn.nextSibling){
5572                             if(cn.nodeType == 1){
5573                                cn.nodeIndex = ++j;
5574                             }
5575                         }
5576                         pn._batch = batch;
5577                     }
5578                     if (f == 1) {
5579                         if (l == 0 || n.nodeIndex == l){
5580                             r[++ri] = n;
5581                         }
5582                     } else if ((n.nodeIndex + l) % f == 0){
5583                         r[++ri] = n;
5584                     }
5585                 }
5586
5587                 return r;
5588             },
5589
5590             "only-child" : function(c){
5591                 var r = [], ri = -1;;
5592                 for(var i = 0, ci; ci = c[i]; i++){
5593                     if(!prev(ci) && !next(ci)){
5594                         r[++ri] = ci;
5595                     }
5596                 }
5597                 return r;
5598             },
5599
5600             "empty" : function(c){
5601                 var r = [], ri = -1;
5602                 for(var i = 0, ci; ci = c[i]; i++){
5603                     var cns = ci.childNodes, j = 0, cn, empty = true;
5604                     while(cn = cns[j]){
5605                         ++j;
5606                         if(cn.nodeType == 1 || cn.nodeType == 3){
5607                             empty = false;
5608                             break;
5609                         }
5610                     }
5611                     if(empty){
5612                         r[++ri] = ci;
5613                     }
5614                 }
5615                 return r;
5616             },
5617
5618             "contains" : function(c, v){
5619                 var r = [], ri = -1;
5620                 for(var i = 0, ci; ci = c[i]; i++){
5621                     if((ci.textContent||ci.innerText||'').indexOf(v) != -1){
5622                         r[++ri] = ci;
5623                     }
5624                 }
5625                 return r;
5626             },
5627
5628             "nodeValue" : function(c, v){
5629                 var r = [], ri = -1;
5630                 for(var i = 0, ci; ci = c[i]; i++){
5631                     if(ci.firstChild && ci.firstChild.nodeValue == v){
5632                         r[++ri] = ci;
5633                     }
5634                 }
5635                 return r;
5636             },
5637
5638             "checked" : function(c){
5639                 var r = [], ri = -1;
5640                 for(var i = 0, ci; ci = c[i]; i++){
5641                     if(ci.checked == true){
5642                         r[++ri] = ci;
5643                     }
5644                 }
5645                 return r;
5646             },
5647
5648             "not" : function(c, ss){
5649                 return Roo.DomQuery.filter(c, ss, true);
5650             },
5651
5652             "odd" : function(c){
5653                 return this["nth-child"](c, "odd");
5654             },
5655
5656             "even" : function(c){
5657                 return this["nth-child"](c, "even");
5658             },
5659
5660             "nth" : function(c, a){
5661                 return c[a-1] || [];
5662             },
5663
5664             "first" : function(c){
5665                 return c[0] || [];
5666             },
5667
5668             "last" : function(c){
5669                 return c[c.length-1] || [];
5670             },
5671
5672             "has" : function(c, ss){
5673                 var s = Roo.DomQuery.select;
5674                 var r = [], ri = -1;
5675                 for(var i = 0, ci; ci = c[i]; i++){
5676                     if(s(ss, ci).length > 0){
5677                         r[++ri] = ci;
5678                     }
5679                 }
5680                 return r;
5681             },
5682
5683             "next" : function(c, ss){
5684                 var is = Roo.DomQuery.is;
5685                 var r = [], ri = -1;
5686                 for(var i = 0, ci; ci = c[i]; i++){
5687                     var n = next(ci);
5688                     if(n && is(n, ss)){
5689                         r[++ri] = ci;
5690                     }
5691                 }
5692                 return r;
5693             },
5694
5695             "prev" : function(c, ss){
5696                 var is = Roo.DomQuery.is;
5697                 var r = [], ri = -1;
5698                 for(var i = 0, ci; ci = c[i]; i++){
5699                     var n = prev(ci);
5700                     if(n && is(n, ss)){
5701                         r[++ri] = ci;
5702                     }
5703                 }
5704                 return r;
5705             }
5706         }
5707     };
5708 }();
5709
5710 /**
5711  * Selects an array of DOM nodes by CSS/XPath selector. Shorthand of {@link Roo.DomQuery#select}
5712  * @param {String} path The selector/xpath query
5713  * @param {Node} root (optional) The start of the query (defaults to document).
5714  * @return {Array}
5715  * @member Roo
5716  * @method query
5717  */
5718 Roo.query = Roo.DomQuery.select;
5719 /*
5720  * Based on:
5721  * Ext JS Library 1.1.1
5722  * Copyright(c) 2006-2007, Ext JS, LLC.
5723  *
5724  * Originally Released Under LGPL - original licence link has changed is not relivant.
5725  *
5726  * Fork - LGPL
5727  * <script type="text/javascript">
5728  */
5729
5730 /**
5731  * @class Roo.util.Observable
5732  * Base class that provides a common interface for publishing events. Subclasses are expected to
5733  * to have a property "events" with all the events defined.<br>
5734  * For example:
5735  * <pre><code>
5736  Employee = function(name){
5737     this.name = name;
5738     this.addEvents({
5739         "fired" : true,
5740         "quit" : true
5741     });
5742  }
5743  Roo.extend(Employee, Roo.util.Observable);
5744 </code></pre>
5745  * @param {Object} config properties to use (incuding events / listeners)
5746  */
5747
5748 Roo.util.Observable = function(cfg){
5749     
5750     cfg = cfg|| {};
5751     this.addEvents(cfg.events || {});
5752     if (cfg.events) {
5753         delete cfg.events; // make sure
5754     }
5755      
5756     Roo.apply(this, cfg);
5757     
5758     if(this.listeners){
5759         this.on(this.listeners);
5760         delete this.listeners;
5761     }
5762 };
5763 Roo.util.Observable.prototype = {
5764     /** 
5765  * @cfg {Object} listeners  list of events and functions to call for this object, 
5766  * For example :
5767  * <pre><code>
5768     listeners :  { 
5769        'click' : function(e) {
5770            ..... 
5771         } ,
5772         .... 
5773     } 
5774   </code></pre>
5775  */
5776     
5777     
5778     /**
5779      * Fires the specified event with the passed parameters (minus the event name).
5780      * @param {String} eventName
5781      * @param {Object...} args Variable number of parameters are passed to handlers
5782      * @return {Boolean} returns false if any of the handlers return false otherwise it returns true
5783      */
5784     fireEvent : function(){
5785         var ce = this.events[arguments[0].toLowerCase()];
5786         if(typeof ce == "object"){
5787             return ce.fire.apply(ce, Array.prototype.slice.call(arguments, 1));
5788         }else{
5789             return true;
5790         }
5791     },
5792
5793     // private
5794     filterOptRe : /^(?:scope|delay|buffer|single)$/,
5795
5796     /**
5797      * Appends an event handler to this component
5798      * @param {String}   eventName The type of event to listen for
5799      * @param {Function} handler The method the event invokes
5800      * @param {Object}   scope (optional) The scope in which to execute the handler
5801      * function. The handler function's "this" context.
5802      * @param {Object}   options (optional) An object containing handler configuration
5803      * properties. This may contain any of the following properties:<ul>
5804      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
5805      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
5806      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
5807      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
5808      * by the specified number of milliseconds. If the event fires again within that time, the original
5809      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
5810      * </ul><br>
5811      * <p>
5812      * <b>Combining Options</b><br>
5813      * Using the options argument, it is possible to combine different types of listeners:<br>
5814      * <br>
5815      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)
5816                 <pre><code>
5817                 el.on('click', this.onClick, this, {
5818                         single: true,
5819                 delay: 100,
5820                 forumId: 4
5821                 });
5822                 </code></pre>
5823      * <p>
5824      * <b>Attaching multiple handlers in 1 call</b><br>
5825      * The method also allows for a single argument to be passed which is a config object containing properties
5826      * which specify multiple handlers.
5827      * <pre><code>
5828                 el.on({
5829                         'click': {
5830                         fn: this.onClick,
5831                         scope: this,
5832                         delay: 100
5833                 }, 
5834                 'mouseover': {
5835                         fn: this.onMouseOver,
5836                         scope: this
5837                 },
5838                 'mouseout': {
5839                         fn: this.onMouseOut,
5840                         scope: this
5841                 }
5842                 });
5843                 </code></pre>
5844      * <p>
5845      * Or a shorthand syntax which passes the same scope object to all handlers:
5846         <pre><code>
5847                 el.on({
5848                         'click': this.onClick,
5849                 'mouseover': this.onMouseOver,
5850                 'mouseout': this.onMouseOut,
5851                 scope: this
5852                 });
5853                 </code></pre>
5854      */
5855     addListener : function(eventName, fn, scope, o){
5856         if(typeof eventName == "object"){
5857             o = eventName;
5858             for(var e in o){
5859                 if(this.filterOptRe.test(e)){
5860                     continue;
5861                 }
5862                 if(typeof o[e] == "function"){
5863                     // shared options
5864                     this.addListener(e, o[e], o.scope,  o);
5865                 }else{
5866                     // individual options
5867                     this.addListener(e, o[e].fn, o[e].scope, o[e]);
5868                 }
5869             }
5870             return;
5871         }
5872         o = (!o || typeof o == "boolean") ? {} : o;
5873         eventName = eventName.toLowerCase();
5874         var ce = this.events[eventName] || true;
5875         if(typeof ce == "boolean"){
5876             ce = new Roo.util.Event(this, eventName);
5877             this.events[eventName] = ce;
5878         }
5879         ce.addListener(fn, scope, o);
5880     },
5881
5882     /**
5883      * Removes a listener
5884      * @param {String}   eventName     The type of event to listen for
5885      * @param {Function} handler        The handler to remove
5886      * @param {Object}   scope  (optional) The scope (this object) for the handler
5887      */
5888     removeListener : function(eventName, fn, scope){
5889         var ce = this.events[eventName.toLowerCase()];
5890         if(typeof ce == "object"){
5891             ce.removeListener(fn, scope);
5892         }
5893     },
5894
5895     /**
5896      * Removes all listeners for this object
5897      */
5898     purgeListeners : function(){
5899         for(var evt in this.events){
5900             if(typeof this.events[evt] == "object"){
5901                  this.events[evt].clearListeners();
5902             }
5903         }
5904     },
5905
5906     relayEvents : function(o, events){
5907         var createHandler = function(ename){
5908             return function(){
5909                 return this.fireEvent.apply(this, Roo.combine(ename, Array.prototype.slice.call(arguments, 0)));
5910             };
5911         };
5912         for(var i = 0, len = events.length; i < len; i++){
5913             var ename = events[i];
5914             if(!this.events[ename]){ this.events[ename] = true; };
5915             o.on(ename, createHandler(ename), this);
5916         }
5917     },
5918
5919     /**
5920      * Used to define events on this Observable
5921      * @param {Object} object The object with the events defined
5922      */
5923     addEvents : function(o){
5924         if(!this.events){
5925             this.events = {};
5926         }
5927         Roo.applyIf(this.events, o);
5928     },
5929
5930     /**
5931      * Checks to see if this object has any listeners for a specified event
5932      * @param {String} eventName The name of the event to check for
5933      * @return {Boolean} True if the event is being listened for, else false
5934      */
5935     hasListener : function(eventName){
5936         var e = this.events[eventName];
5937         return typeof e == "object" && e.listeners.length > 0;
5938     }
5939 };
5940 /**
5941  * Appends an event handler to this element (shorthand for addListener)
5942  * @param {String}   eventName     The type of event to listen for
5943  * @param {Function} handler        The method the event invokes
5944  * @param {Object}   scope (optional) The scope in which to execute the handler
5945  * function. The handler function's "this" context.
5946  * @param {Object}   options  (optional)
5947  * @method
5948  */
5949 Roo.util.Observable.prototype.on = Roo.util.Observable.prototype.addListener;
5950 /**
5951  * Removes a listener (shorthand for removeListener)
5952  * @param {String}   eventName     The type of event to listen for
5953  * @param {Function} handler        The handler to remove
5954  * @param {Object}   scope  (optional) The scope (this object) for the handler
5955  * @method
5956  */
5957 Roo.util.Observable.prototype.un = Roo.util.Observable.prototype.removeListener;
5958
5959 /**
5960  * Starts capture on the specified Observable. All events will be passed
5961  * to the supplied function with the event name + standard signature of the event
5962  * <b>before</b> the event is fired. If the supplied function returns false,
5963  * the event will not fire.
5964  * @param {Observable} o The Observable to capture
5965  * @param {Function} fn The function to call
5966  * @param {Object} scope (optional) The scope (this object) for the fn
5967  * @static
5968  */
5969 Roo.util.Observable.capture = function(o, fn, scope){
5970     o.fireEvent = o.fireEvent.createInterceptor(fn, scope);
5971 };
5972
5973 /**
5974  * Removes <b>all</b> added captures from the Observable.
5975  * @param {Observable} o The Observable to release
5976  * @static
5977  */
5978 Roo.util.Observable.releaseCapture = function(o){
5979     o.fireEvent = Roo.util.Observable.prototype.fireEvent;
5980 };
5981
5982 (function(){
5983
5984     var createBuffered = function(h, o, scope){
5985         var task = new Roo.util.DelayedTask();
5986         return function(){
5987             task.delay(o.buffer, h, scope, Array.prototype.slice.call(arguments, 0));
5988         };
5989     };
5990
5991     var createSingle = function(h, e, fn, scope){
5992         return function(){
5993             e.removeListener(fn, scope);
5994             return h.apply(scope, arguments);
5995         };
5996     };
5997
5998     var createDelayed = function(h, o, scope){
5999         return function(){
6000             var args = Array.prototype.slice.call(arguments, 0);
6001             setTimeout(function(){
6002                 h.apply(scope, args);
6003             }, o.delay || 10);
6004         };
6005     };
6006
6007     Roo.util.Event = function(obj, name){
6008         this.name = name;
6009         this.obj = obj;
6010         this.listeners = [];
6011     };
6012
6013     Roo.util.Event.prototype = {
6014         addListener : function(fn, scope, options){
6015             var o = options || {};
6016             scope = scope || this.obj;
6017             if(!this.isListening(fn, scope)){
6018                 var l = {fn: fn, scope: scope, options: o};
6019                 var h = fn;
6020                 if(o.delay){
6021                     h = createDelayed(h, o, scope);
6022                 }
6023                 if(o.single){
6024                     h = createSingle(h, this, fn, scope);
6025                 }
6026                 if(o.buffer){
6027                     h = createBuffered(h, o, scope);
6028                 }
6029                 l.fireFn = h;
6030                 if(!this.firing){ // if we are currently firing this event, don't disturb the listener loop
6031                     this.listeners.push(l);
6032                 }else{
6033                     this.listeners = this.listeners.slice(0);
6034                     this.listeners.push(l);
6035                 }
6036             }
6037         },
6038
6039         findListener : function(fn, scope){
6040             scope = scope || this.obj;
6041             var ls = this.listeners;
6042             for(var i = 0, len = ls.length; i < len; i++){
6043                 var l = ls[i];
6044                 if(l.fn == fn && l.scope == scope){
6045                     return i;
6046                 }
6047             }
6048             return -1;
6049         },
6050
6051         isListening : function(fn, scope){
6052             return this.findListener(fn, scope) != -1;
6053         },
6054
6055         removeListener : function(fn, scope){
6056             var index;
6057             if((index = this.findListener(fn, scope)) != -1){
6058                 if(!this.firing){
6059                     this.listeners.splice(index, 1);
6060                 }else{
6061                     this.listeners = this.listeners.slice(0);
6062                     this.listeners.splice(index, 1);
6063                 }
6064                 return true;
6065             }
6066             return false;
6067         },
6068
6069         clearListeners : function(){
6070             this.listeners = [];
6071         },
6072
6073         fire : function(){
6074             var ls = this.listeners, scope, len = ls.length;
6075             if(len > 0){
6076                 this.firing = true;
6077                 
6078                 for(var i = 0; i < len; i++){
6079                     var args = Array.prototype.slice.call(arguments, 0);
6080                     var l = ls[i];
6081                     args.push(l.options);
6082                     if(l.fireFn.apply(l.scope||this.obj||window, args) === false){
6083                         this.firing = false;
6084                         return false;
6085                     }
6086                 }
6087                 this.firing = false;
6088             }
6089             return true;
6090         }
6091     };
6092 })();/*
6093  * RooJS Library 
6094  * Copyright(c) 2007-2017, Roo J Solutions Ltd
6095  *
6096  * Licence LGPL 
6097  *
6098  */
6099  
6100 /**
6101  * @class Roo.Document
6102  * @extends Roo.util.Observable
6103  * This is a convience class to wrap up the main document loading code.. , rather than adding Roo.onReady(......)
6104  * 
6105  * @param {Object} config the methods and properties of the 'base' class for the application.
6106  * 
6107  *  Generic Page handler - implement this to start your app..
6108  * 
6109  * eg.
6110  *  MyProject = new Roo.Document({
6111         events : {
6112             'load' : true // your events..
6113         },
6114         listeners : {
6115             'ready' : function() {
6116                 // fired on Roo.onReady()
6117             }
6118         }
6119  * 
6120  */
6121 Roo.Document = function(cfg) {
6122      
6123     this.addEvents({ 
6124         'ready' : true
6125     });
6126     Roo.util.Observable.call(this,cfg);
6127     
6128     var _this = this;
6129     
6130     Roo.onReady(function() {
6131         _this.fireEvent('ready');
6132     },null,false);
6133     
6134     
6135 }
6136
6137 Roo.extend(Roo.Document, Roo.util.Observable, {});/*
6138  * Based on:
6139  * Ext JS Library 1.1.1
6140  * Copyright(c) 2006-2007, Ext JS, LLC.
6141  *
6142  * Originally Released Under LGPL - original licence link has changed is not relivant.
6143  *
6144  * Fork - LGPL
6145  * <script type="text/javascript">
6146  */
6147
6148 /**
6149  * @class Roo.EventManager
6150  * Registers event handlers that want to receive a normalized EventObject instead of the standard browser event and provides 
6151  * several useful events directly.
6152  * See {@link Roo.EventObject} for more details on normalized event objects.
6153  * @singleton
6154  */
6155 Roo.EventManager = function(){
6156     var docReadyEvent, docReadyProcId, docReadyState = false;
6157     var resizeEvent, resizeTask, textEvent, textSize;
6158     var E = Roo.lib.Event;
6159     var D = Roo.lib.Dom;
6160
6161     
6162     
6163
6164     var fireDocReady = function(){
6165         if(!docReadyState){
6166             docReadyState = true;
6167             Roo.isReady = true;
6168             if(docReadyProcId){
6169                 clearInterval(docReadyProcId);
6170             }
6171             if(Roo.isGecko || Roo.isOpera) {
6172                 document.removeEventListener("DOMContentLoaded", fireDocReady, false);
6173             }
6174             if(Roo.isIE){
6175                 var defer = document.getElementById("ie-deferred-loader");
6176                 if(defer){
6177                     defer.onreadystatechange = null;
6178                     defer.parentNode.removeChild(defer);
6179                 }
6180             }
6181             if(docReadyEvent){
6182                 docReadyEvent.fire();
6183                 docReadyEvent.clearListeners();
6184             }
6185         }
6186     };
6187     
6188     var initDocReady = function(){
6189         docReadyEvent = new Roo.util.Event();
6190         if(Roo.isGecko || Roo.isOpera) {
6191             document.addEventListener("DOMContentLoaded", fireDocReady, false);
6192         }else if(Roo.isIE){
6193             document.write("<s"+'cript id="ie-deferred-loader" defer="defer" src="/'+'/:"></s'+"cript>");
6194             var defer = document.getElementById("ie-deferred-loader");
6195             defer.onreadystatechange = function(){
6196                 if(this.readyState == "complete"){
6197                     fireDocReady();
6198                 }
6199             };
6200         }else if(Roo.isSafari){ 
6201             docReadyProcId = setInterval(function(){
6202                 var rs = document.readyState;
6203                 if(rs == "complete") {
6204                     fireDocReady();     
6205                  }
6206             }, 10);
6207         }
6208         // no matter what, make sure it fires on load
6209         E.on(window, "load", fireDocReady);
6210     };
6211
6212     var createBuffered = function(h, o){
6213         var task = new Roo.util.DelayedTask(h);
6214         return function(e){
6215             // create new event object impl so new events don't wipe out properties
6216             e = new Roo.EventObjectImpl(e);
6217             task.delay(o.buffer, h, null, [e]);
6218         };
6219     };
6220
6221     var createSingle = function(h, el, ename, fn){
6222         return function(e){
6223             Roo.EventManager.removeListener(el, ename, fn);
6224             h(e);
6225         };
6226     };
6227
6228     var createDelayed = function(h, o){
6229         return function(e){
6230             // create new event object impl so new events don't wipe out properties
6231             e = new Roo.EventObjectImpl(e);
6232             setTimeout(function(){
6233                 h(e);
6234             }, o.delay || 10);
6235         };
6236     };
6237     var transitionEndVal = false;
6238     
6239     var transitionEnd = function()
6240     {
6241         if (transitionEndVal) {
6242             return transitionEndVal;
6243         }
6244         var el = document.createElement('div');
6245
6246         var transEndEventNames = {
6247             WebkitTransition : 'webkitTransitionEnd',
6248             MozTransition    : 'transitionend',
6249             OTransition      : 'oTransitionEnd otransitionend',
6250             transition       : 'transitionend'
6251         };
6252     
6253         for (var name in transEndEventNames) {
6254             if (el.style[name] !== undefined) {
6255                 transitionEndVal = transEndEventNames[name];
6256                 return  transitionEndVal ;
6257             }
6258         }
6259     }
6260     
6261
6262     var listen = function(element, ename, opt, fn, scope){
6263         var o = (!opt || typeof opt == "boolean") ? {} : opt;
6264         fn = fn || o.fn; scope = scope || o.scope;
6265         var el = Roo.getDom(element);
6266         
6267         
6268         if(!el){
6269             throw "Error listening for \"" + ename + '\". Element "' + element + '" doesn\'t exist.';
6270         }
6271         
6272         if (ename == 'transitionend') {
6273             ename = transitionEnd();
6274         }
6275         var h = function(e){
6276             e = Roo.EventObject.setEvent(e);
6277             var t;
6278             if(o.delegate){
6279                 t = e.getTarget(o.delegate, el);
6280                 if(!t){
6281                     return;
6282                 }
6283             }else{
6284                 t = e.target;
6285             }
6286             if(o.stopEvent === true){
6287                 e.stopEvent();
6288             }
6289             if(o.preventDefault === true){
6290                e.preventDefault();
6291             }
6292             if(o.stopPropagation === true){
6293                 e.stopPropagation();
6294             }
6295
6296             if(o.normalized === false){
6297                 e = e.browserEvent;
6298             }
6299
6300             fn.call(scope || el, e, t, o);
6301         };
6302         if(o.delay){
6303             h = createDelayed(h, o);
6304         }
6305         if(o.single){
6306             h = createSingle(h, el, ename, fn);
6307         }
6308         if(o.buffer){
6309             h = createBuffered(h, o);
6310         }
6311         
6312         fn._handlers = fn._handlers || [];
6313         
6314         
6315         fn._handlers.push([Roo.id(el), ename, h]);
6316         
6317         
6318          
6319         E.on(el, ename, h);
6320         if(ename == "mousewheel" && el.addEventListener){ // workaround for jQuery
6321             el.addEventListener("DOMMouseScroll", h, false);
6322             E.on(window, 'unload', function(){
6323                 el.removeEventListener("DOMMouseScroll", h, false);
6324             });
6325         }
6326         if(ename == "mousedown" && el == document){ // fix stopped mousedowns on the document
6327             Roo.EventManager.stoppedMouseDownEvent.addListener(h);
6328         }
6329         return h;
6330     };
6331
6332     var stopListening = function(el, ename, fn){
6333         var id = Roo.id(el), hds = fn._handlers, hd = fn;
6334         if(hds){
6335             for(var i = 0, len = hds.length; i < len; i++){
6336                 var h = hds[i];
6337                 if(h[0] == id && h[1] == ename){
6338                     hd = h[2];
6339                     hds.splice(i, 1);
6340                     break;
6341                 }
6342             }
6343         }
6344         E.un(el, ename, hd);
6345         el = Roo.getDom(el);
6346         if(ename == "mousewheel" && el.addEventListener){
6347             el.removeEventListener("DOMMouseScroll", hd, false);
6348         }
6349         if(ename == "mousedown" && el == document){ // fix stopped mousedowns on the document
6350             Roo.EventManager.stoppedMouseDownEvent.removeListener(hd);
6351         }
6352     };
6353
6354     var propRe = /^(?:scope|delay|buffer|single|stopEvent|preventDefault|stopPropagation|normalized|args|delegate)$/;
6355     
6356     var pub = {
6357         
6358         
6359         /** 
6360          * Fix for doc tools
6361          * @scope Roo.EventManager
6362          */
6363         
6364         
6365         /** 
6366          * This is no longer needed and is deprecated. Places a simple wrapper around an event handler to override the browser event
6367          * object with a Roo.EventObject
6368          * @param {Function} fn        The method the event invokes
6369          * @param {Object}   scope    An object that becomes the scope of the handler
6370          * @param {boolean}  override If true, the obj passed in becomes
6371          *                             the execution scope of the listener
6372          * @return {Function} The wrapped function
6373          * @deprecated
6374          */
6375         wrap : function(fn, scope, override){
6376             return function(e){
6377                 Roo.EventObject.setEvent(e);
6378                 fn.call(override ? scope || window : window, Roo.EventObject, scope);
6379             };
6380         },
6381         
6382         /**
6383      * Appends an event handler to an element (shorthand for addListener)
6384      * @param {String/HTMLElement}   element        The html element or id to assign the
6385      * @param {String}   eventName The type of event to listen for
6386      * @param {Function} handler The method the event invokes
6387      * @param {Object}   scope (optional) The scope in which to execute the handler
6388      * function. The handler function's "this" context.
6389      * @param {Object}   options (optional) An object containing handler configuration
6390      * properties. This may contain any of the following properties:<ul>
6391      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
6392      * <li>delegate {String} A simple selector to filter the target or look for a descendant of the target</li>
6393      * <li>stopEvent {Boolean} True to stop the event. That is stop propagation, and prevent the default action.</li>
6394      * <li>preventDefault {Boolean} True to prevent the default action</li>
6395      * <li>stopPropagation {Boolean} True to prevent event propagation</li>
6396      * <li>normalized {Boolean} False to pass a browser event to the handler function instead of an Roo.EventObject</li>
6397      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
6398      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
6399      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
6400      * by the specified number of milliseconds. If the event fires again within that time, the original
6401      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
6402      * </ul><br>
6403      * <p>
6404      * <b>Combining Options</b><br>
6405      * Using the options argument, it is possible to combine different types of listeners:<br>
6406      * <br>
6407      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)<div style="margin: 5px 20px 20px;">
6408      * Code:<pre><code>
6409 el.on('click', this.onClick, this, {
6410     single: true,
6411     delay: 100,
6412     stopEvent : true,
6413     forumId: 4
6414 });</code></pre>
6415      * <p>
6416      * <b>Attaching multiple handlers in 1 call</b><br>
6417       * The method also allows for a single argument to be passed which is a config object containing properties
6418      * which specify multiple handlers.
6419      * <p>
6420      * Code:<pre><code>
6421 el.on({
6422     'click' : {
6423         fn: this.onClick
6424         scope: this,
6425         delay: 100
6426     },
6427     'mouseover' : {
6428         fn: this.onMouseOver
6429         scope: this
6430     },
6431     'mouseout' : {
6432         fn: this.onMouseOut
6433         scope: this
6434     }
6435 });</code></pre>
6436      * <p>
6437      * Or a shorthand syntax:<br>
6438      * Code:<pre><code>
6439 el.on({
6440     'click' : this.onClick,
6441     'mouseover' : this.onMouseOver,
6442     'mouseout' : this.onMouseOut
6443     scope: this
6444 });</code></pre>
6445      */
6446         addListener : function(element, eventName, fn, scope, options){
6447             if(typeof eventName == "object"){
6448                 var o = eventName;
6449                 for(var e in o){
6450                     if(propRe.test(e)){
6451                         continue;
6452                     }
6453                     if(typeof o[e] == "function"){
6454                         // shared options
6455                         listen(element, e, o, o[e], o.scope);
6456                     }else{
6457                         // individual options
6458                         listen(element, e, o[e]);
6459                     }
6460                 }
6461                 return;
6462             }
6463             return listen(element, eventName, options, fn, scope);
6464         },
6465         
6466         /**
6467          * Removes an event handler
6468          *
6469          * @param {String/HTMLElement}   element        The id or html element to remove the 
6470          *                             event from
6471          * @param {String}   eventName     The type of event
6472          * @param {Function} fn
6473          * @return {Boolean} True if a listener was actually removed
6474          */
6475         removeListener : function(element, eventName, fn){
6476             return stopListening(element, eventName, fn);
6477         },
6478         
6479         /**
6480          * Fires when the document is ready (before onload and before images are loaded). Can be 
6481          * accessed shorthanded Roo.onReady().
6482          * @param {Function} fn        The method the event invokes
6483          * @param {Object}   scope    An  object that becomes the scope of the handler
6484          * @param {boolean}  options
6485          */
6486         onDocumentReady : function(fn, scope, options){
6487             if(docReadyState){ // if it already fired
6488                 docReadyEvent.addListener(fn, scope, options);
6489                 docReadyEvent.fire();
6490                 docReadyEvent.clearListeners();
6491                 return;
6492             }
6493             if(!docReadyEvent){
6494                 initDocReady();
6495             }
6496             docReadyEvent.addListener(fn, scope, options);
6497         },
6498         
6499         /**
6500          * Fires when the window is resized and provides resize event buffering (50 milliseconds), passes new viewport width and height to handlers.
6501          * @param {Function} fn        The method the event invokes
6502          * @param {Object}   scope    An object that becomes the scope of the handler
6503          * @param {boolean}  options
6504          */
6505         onWindowResize : function(fn, scope, options){
6506             if(!resizeEvent){
6507                 resizeEvent = new Roo.util.Event();
6508                 resizeTask = new Roo.util.DelayedTask(function(){
6509                     resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6510                 });
6511                 E.on(window, "resize", function(){
6512                     if(Roo.isIE){
6513                         resizeTask.delay(50);
6514                     }else{
6515                         resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6516                     }
6517                 });
6518             }
6519             resizeEvent.addListener(fn, scope, options);
6520         },
6521
6522         /**
6523          * Fires when the user changes the active text size. Handler gets called with 2 params, the old size and the new size.
6524          * @param {Function} fn        The method the event invokes
6525          * @param {Object}   scope    An object that becomes the scope of the handler
6526          * @param {boolean}  options
6527          */
6528         onTextResize : function(fn, scope, options){
6529             if(!textEvent){
6530                 textEvent = new Roo.util.Event();
6531                 var textEl = new Roo.Element(document.createElement('div'));
6532                 textEl.dom.className = 'x-text-resize';
6533                 textEl.dom.innerHTML = 'X';
6534                 textEl.appendTo(document.body);
6535                 textSize = textEl.dom.offsetHeight;
6536                 setInterval(function(){
6537                     if(textEl.dom.offsetHeight != textSize){
6538                         textEvent.fire(textSize, textSize = textEl.dom.offsetHeight);
6539                     }
6540                 }, this.textResizeInterval);
6541             }
6542             textEvent.addListener(fn, scope, options);
6543         },
6544
6545         /**
6546          * Removes the passed window resize listener.
6547          * @param {Function} fn        The method the event invokes
6548          * @param {Object}   scope    The scope of handler
6549          */
6550         removeResizeListener : function(fn, scope){
6551             if(resizeEvent){
6552                 resizeEvent.removeListener(fn, scope);
6553             }
6554         },
6555
6556         // private
6557         fireResize : function(){
6558             if(resizeEvent){
6559                 resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6560             }   
6561         },
6562         /**
6563          * Url used for onDocumentReady with using SSL (defaults to Roo.SSL_SECURE_URL)
6564          */
6565         ieDeferSrc : false,
6566         /**
6567          * The frequency, in milliseconds, to check for text resize events (defaults to 50)
6568          */
6569         textResizeInterval : 50
6570     };
6571     
6572     /**
6573      * Fix for doc tools
6574      * @scopeAlias pub=Roo.EventManager
6575      */
6576     
6577      /**
6578      * Appends an event handler to an element (shorthand for addListener)
6579      * @param {String/HTMLElement}   element        The html element or id to assign the
6580      * @param {String}   eventName The type of event to listen for
6581      * @param {Function} handler The method the event invokes
6582      * @param {Object}   scope (optional) The scope in which to execute the handler
6583      * function. The handler function's "this" context.
6584      * @param {Object}   options (optional) An object containing handler configuration
6585      * properties. This may contain any of the following properties:<ul>
6586      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
6587      * <li>delegate {String} A simple selector to filter the target or look for a descendant of the target</li>
6588      * <li>stopEvent {Boolean} True to stop the event. That is stop propagation, and prevent the default action.</li>
6589      * <li>preventDefault {Boolean} True to prevent the default action</li>
6590      * <li>stopPropagation {Boolean} True to prevent event propagation</li>
6591      * <li>normalized {Boolean} False to pass a browser event to the handler function instead of an Roo.EventObject</li>
6592      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
6593      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
6594      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
6595      * by the specified number of milliseconds. If the event fires again within that time, the original
6596      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
6597      * </ul><br>
6598      * <p>
6599      * <b>Combining Options</b><br>
6600      * Using the options argument, it is possible to combine different types of listeners:<br>
6601      * <br>
6602      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)<div style="margin: 5px 20px 20px;">
6603      * Code:<pre><code>
6604 el.on('click', this.onClick, this, {
6605     single: true,
6606     delay: 100,
6607     stopEvent : true,
6608     forumId: 4
6609 });</code></pre>
6610      * <p>
6611      * <b>Attaching multiple handlers in 1 call</b><br>
6612       * The method also allows for a single argument to be passed which is a config object containing properties
6613      * which specify multiple handlers.
6614      * <p>
6615      * Code:<pre><code>
6616 el.on({
6617     'click' : {
6618         fn: this.onClick
6619         scope: this,
6620         delay: 100
6621     },
6622     'mouseover' : {
6623         fn: this.onMouseOver
6624         scope: this
6625     },
6626     'mouseout' : {
6627         fn: this.onMouseOut
6628         scope: this
6629     }
6630 });</code></pre>
6631      * <p>
6632      * Or a shorthand syntax:<br>
6633      * Code:<pre><code>
6634 el.on({
6635     'click' : this.onClick,
6636     'mouseover' : this.onMouseOver,
6637     'mouseout' : this.onMouseOut
6638     scope: this
6639 });</code></pre>
6640      */
6641     pub.on = pub.addListener;
6642     pub.un = pub.removeListener;
6643
6644     pub.stoppedMouseDownEvent = new Roo.util.Event();
6645     return pub;
6646 }();
6647 /**
6648   * Fires when the document is ready (before onload and before images are loaded).  Shorthand of {@link Roo.EventManager#onDocumentReady}.
6649   * @param {Function} fn        The method the event invokes
6650   * @param {Object}   scope    An  object that becomes the scope of the handler
6651   * @param {boolean}  override If true, the obj passed in becomes
6652   *                             the execution scope of the listener
6653   * @member Roo
6654   * @method onReady
6655  */
6656 Roo.onReady = Roo.EventManager.onDocumentReady;
6657
6658 Roo.onReady(function(){
6659     var bd = Roo.get(document.body);
6660     if(!bd){ return; }
6661
6662     var cls = [
6663             Roo.isIE ? "roo-ie"
6664             : Roo.isIE11 ? "roo-ie11"
6665             : Roo.isEdge ? "roo-edge"
6666             : Roo.isGecko ? "roo-gecko"
6667             : Roo.isOpera ? "roo-opera"
6668             : Roo.isSafari ? "roo-safari" : ""];
6669
6670     if(Roo.isMac){
6671         cls.push("roo-mac");
6672     }
6673     if(Roo.isLinux){
6674         cls.push("roo-linux");
6675     }
6676     if(Roo.isIOS){
6677         cls.push("roo-ios");
6678     }
6679     if(Roo.isTouch){
6680         cls.push("roo-touch");
6681     }
6682     if(Roo.isBorderBox){
6683         cls.push('roo-border-box');
6684     }
6685     if(Roo.isStrict){ // add to the parent to allow for selectors like ".ext-strict .ext-ie"
6686         var p = bd.dom.parentNode;
6687         if(p){
6688             p.className += ' roo-strict';
6689         }
6690     }
6691     bd.addClass(cls.join(' '));
6692 });
6693
6694 /**
6695  * @class Roo.EventObject
6696  * EventObject exposes the Yahoo! UI Event functionality directly on the object
6697  * passed to your event handler. It exists mostly for convenience. It also fixes the annoying null checks automatically to cleanup your code 
6698  * Example:
6699  * <pre><code>
6700  function handleClick(e){ // e is not a standard event object, it is a Roo.EventObject
6701     e.preventDefault();
6702     var target = e.getTarget();
6703     ...
6704  }
6705  var myDiv = Roo.get("myDiv");
6706  myDiv.on("click", handleClick);
6707  //or
6708  Roo.EventManager.on("myDiv", 'click', handleClick);
6709  Roo.EventManager.addListener("myDiv", 'click', handleClick);
6710  </code></pre>
6711  * @singleton
6712  */
6713 Roo.EventObject = function(){
6714     
6715     var E = Roo.lib.Event;
6716     
6717     // safari keypress events for special keys return bad keycodes
6718     var safariKeys = {
6719         63234 : 37, // left
6720         63235 : 39, // right
6721         63232 : 38, // up
6722         63233 : 40, // down
6723         63276 : 33, // page up
6724         63277 : 34, // page down
6725         63272 : 46, // delete
6726         63273 : 36, // home
6727         63275 : 35  // end
6728     };
6729
6730     // normalize button clicks
6731     var btnMap = Roo.isIE ? {1:0,4:1,2:2} :
6732                 (Roo.isSafari ? {1:0,2:1,3:2} : {0:0,1:1,2:2});
6733
6734     Roo.EventObjectImpl = function(e){
6735         if(e){
6736             this.setEvent(e.browserEvent || e);
6737         }
6738     };
6739     Roo.EventObjectImpl.prototype = {
6740         /**
6741          * Used to fix doc tools.
6742          * @scope Roo.EventObject.prototype
6743          */
6744             
6745
6746         
6747         
6748         /** The normal browser event */
6749         browserEvent : null,
6750         /** The button pressed in a mouse event */
6751         button : -1,
6752         /** True if the shift key was down during the event */
6753         shiftKey : false,
6754         /** True if the control key was down during the event */
6755         ctrlKey : false,
6756         /** True if the alt key was down during the event */
6757         altKey : false,
6758
6759         /** Key constant 
6760         * @type Number */
6761         BACKSPACE : 8,
6762         /** Key constant 
6763         * @type Number */
6764         TAB : 9,
6765         /** Key constant 
6766         * @type Number */
6767         RETURN : 13,
6768         /** Key constant 
6769         * @type Number */
6770         ENTER : 13,
6771         /** Key constant 
6772         * @type Number */
6773         SHIFT : 16,
6774         /** Key constant 
6775         * @type Number */
6776         CONTROL : 17,
6777         /** Key constant 
6778         * @type Number */
6779         ESC : 27,
6780         /** Key constant 
6781         * @type Number */
6782         SPACE : 32,
6783         /** Key constant 
6784         * @type Number */
6785         PAGEUP : 33,
6786         /** Key constant 
6787         * @type Number */
6788         PAGEDOWN : 34,
6789         /** Key constant 
6790         * @type Number */
6791         END : 35,
6792         /** Key constant 
6793         * @type Number */
6794         HOME : 36,
6795         /** Key constant 
6796         * @type Number */
6797         LEFT : 37,
6798         /** Key constant 
6799         * @type Number */
6800         UP : 38,
6801         /** Key constant 
6802         * @type Number */
6803         RIGHT : 39,
6804         /** Key constant 
6805         * @type Number */
6806         DOWN : 40,
6807         /** Key constant 
6808         * @type Number */
6809         DELETE : 46,
6810         /** Key constant 
6811         * @type Number */
6812         F5 : 116,
6813
6814            /** @private */
6815         setEvent : function(e){
6816             if(e == this || (e && e.browserEvent)){ // already wrapped
6817                 return e;
6818             }
6819             this.browserEvent = e;
6820             if(e){
6821                 // normalize buttons
6822                 this.button = e.button ? btnMap[e.button] : (e.which ? e.which-1 : -1);
6823                 if(e.type == 'click' && this.button == -1){
6824                     this.button = 0;
6825                 }
6826                 this.type = e.type;
6827                 this.shiftKey = e.shiftKey;
6828                 // mac metaKey behaves like ctrlKey
6829                 this.ctrlKey = e.ctrlKey || e.metaKey;
6830                 this.altKey = e.altKey;
6831                 // in getKey these will be normalized for the mac
6832                 this.keyCode = e.keyCode;
6833                 // keyup warnings on firefox.
6834                 this.charCode = (e.type == 'keyup' || e.type == 'keydown') ? 0 : e.charCode;
6835                 // cache the target for the delayed and or buffered events
6836                 this.target = E.getTarget(e);
6837                 // same for XY
6838                 this.xy = E.getXY(e);
6839             }else{
6840                 this.button = -1;
6841                 this.shiftKey = false;
6842                 this.ctrlKey = false;
6843                 this.altKey = false;
6844                 this.keyCode = 0;
6845                 this.charCode =0;
6846                 this.target = null;
6847                 this.xy = [0, 0];
6848             }
6849             return this;
6850         },
6851
6852         /**
6853          * Stop the event (preventDefault and stopPropagation)
6854          */
6855         stopEvent : function(){
6856             if(this.browserEvent){
6857                 if(this.browserEvent.type == 'mousedown'){
6858                     Roo.EventManager.stoppedMouseDownEvent.fire(this);
6859                 }
6860                 E.stopEvent(this.browserEvent);
6861             }
6862         },
6863
6864         /**
6865          * Prevents the browsers default handling of the event.
6866          */
6867         preventDefault : function(){
6868             if(this.browserEvent){
6869                 E.preventDefault(this.browserEvent);
6870             }
6871         },
6872
6873         /** @private */
6874         isNavKeyPress : function(){
6875             var k = this.keyCode;
6876             k = Roo.isSafari ? (safariKeys[k] || k) : k;
6877             return (k >= 33 && k <= 40) || k == this.RETURN || k == this.TAB || k == this.ESC;
6878         },
6879
6880         isSpecialKey : function(){
6881             var k = this.keyCode;
6882             return (this.type == 'keypress' && this.ctrlKey) || k == 9 || k == 13  || k == 40 || k == 27 ||
6883             (k == 16) || (k == 17) ||
6884             (k >= 18 && k <= 20) ||
6885             (k >= 33 && k <= 35) ||
6886             (k >= 36 && k <= 39) ||
6887             (k >= 44 && k <= 45);
6888         },
6889         /**
6890          * Cancels bubbling of the event.
6891          */
6892         stopPropagation : function(){
6893             if(this.browserEvent){
6894                 if(this.type == 'mousedown'){
6895                     Roo.EventManager.stoppedMouseDownEvent.fire(this);
6896                 }
6897                 E.stopPropagation(this.browserEvent);
6898             }
6899         },
6900
6901         /**
6902          * Gets the key code for the event.
6903          * @return {Number}
6904          */
6905         getCharCode : function(){
6906             return this.charCode || this.keyCode;
6907         },
6908
6909         /**
6910          * Returns a normalized keyCode for the event.
6911          * @return {Number} The key code
6912          */
6913         getKey : function(){
6914             var k = this.keyCode || this.charCode;
6915             return Roo.isSafari ? (safariKeys[k] || k) : k;
6916         },
6917
6918         /**
6919          * Gets the x coordinate of the event.
6920          * @return {Number}
6921          */
6922         getPageX : function(){
6923             return this.xy[0];
6924         },
6925
6926         /**
6927          * Gets the y coordinate of the event.
6928          * @return {Number}
6929          */
6930         getPageY : function(){
6931             return this.xy[1];
6932         },
6933
6934         /**
6935          * Gets the time of the event.
6936          * @return {Number}
6937          */
6938         getTime : function(){
6939             if(this.browserEvent){
6940                 return E.getTime(this.browserEvent);
6941             }
6942             return null;
6943         },
6944
6945         /**
6946          * Gets the page coordinates of the event.
6947          * @return {Array} The xy values like [x, y]
6948          */
6949         getXY : function(){
6950             return this.xy;
6951         },
6952
6953         /**
6954          * Gets the target for the event.
6955          * @param {String} selector (optional) A simple selector to filter the target or look for an ancestor of the target
6956          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
6957                 search as a number or element (defaults to 10 || document.body)
6958          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
6959          * @return {HTMLelement}
6960          */
6961         getTarget : function(selector, maxDepth, returnEl){
6962             return selector ? Roo.fly(this.target).findParent(selector, maxDepth, returnEl) : this.target;
6963         },
6964         /**
6965          * Gets the related target.
6966          * @return {HTMLElement}
6967          */
6968         getRelatedTarget : function(){
6969             if(this.browserEvent){
6970                 return E.getRelatedTarget(this.browserEvent);
6971             }
6972             return null;
6973         },
6974
6975         /**
6976          * Normalizes mouse wheel delta across browsers
6977          * @return {Number} The delta
6978          */
6979         getWheelDelta : function(){
6980             var e = this.browserEvent;
6981             var delta = 0;
6982             if(e.wheelDelta){ /* IE/Opera. */
6983                 delta = e.wheelDelta/120;
6984             }else if(e.detail){ /* Mozilla case. */
6985                 delta = -e.detail/3;
6986             }
6987             return delta;
6988         },
6989
6990         /**
6991          * Returns true if the control, meta, shift or alt key was pressed during this event.
6992          * @return {Boolean}
6993          */
6994         hasModifier : function(){
6995             return !!((this.ctrlKey || this.altKey) || this.shiftKey);
6996         },
6997
6998         /**
6999          * Returns true if the target of this event equals el or is a child of el
7000          * @param {String/HTMLElement/Element} el
7001          * @param {Boolean} related (optional) true to test if the related target is within el instead of the target
7002          * @return {Boolean}
7003          */
7004         within : function(el, related){
7005             var t = this[related ? "getRelatedTarget" : "getTarget"]();
7006             return t && Roo.fly(el).contains(t);
7007         },
7008
7009         getPoint : function(){
7010             return new Roo.lib.Point(this.xy[0], this.xy[1]);
7011         }
7012     };
7013
7014     return new Roo.EventObjectImpl();
7015 }();
7016             
7017     /*
7018  * Based on:
7019  * Ext JS Library 1.1.1
7020  * Copyright(c) 2006-2007, Ext JS, LLC.
7021  *
7022  * Originally Released Under LGPL - original licence link has changed is not relivant.
7023  *
7024  * Fork - LGPL
7025  * <script type="text/javascript">
7026  */
7027
7028  
7029 // was in Composite Element!??!?!
7030  
7031 (function(){
7032     var D = Roo.lib.Dom;
7033     var E = Roo.lib.Event;
7034     var A = Roo.lib.Anim;
7035
7036     // local style camelizing for speed
7037     var propCache = {};
7038     var camelRe = /(-[a-z])/gi;
7039     var camelFn = function(m, a){ return a.charAt(1).toUpperCase(); };
7040     var view = document.defaultView;
7041
7042 /**
7043  * @class Roo.Element
7044  * Represents an Element in the DOM.<br><br>
7045  * Usage:<br>
7046 <pre><code>
7047 var el = Roo.get("my-div");
7048
7049 // or with getEl
7050 var el = getEl("my-div");
7051
7052 // or with a DOM element
7053 var el = Roo.get(myDivElement);
7054 </code></pre>
7055  * Using Roo.get() or getEl() instead of calling the constructor directly ensures you get the same object
7056  * each call instead of constructing a new one.<br><br>
7057  * <b>Animations</b><br />
7058  * Many of the functions for manipulating an element have an optional "animate" parameter. The animate parameter
7059  * should either be a boolean (true) or an object literal with animation options. The animation options are:
7060 <pre>
7061 Option    Default   Description
7062 --------- --------  ---------------------------------------------
7063 duration  .35       The duration of the animation in seconds
7064 easing    easeOut   The YUI easing method
7065 callback  none      A function to execute when the anim completes
7066 scope     this      The scope (this) of the callback function
7067 </pre>
7068 * Also, the Anim object being used for the animation will be set on your options object as "anim", which allows you to stop or
7069 * manipulate the animation. Here's an example:
7070 <pre><code>
7071 var el = Roo.get("my-div");
7072
7073 // no animation
7074 el.setWidth(100);
7075
7076 // default animation
7077 el.setWidth(100, true);
7078
7079 // animation with some options set
7080 el.setWidth(100, {
7081     duration: 1,
7082     callback: this.foo,
7083     scope: this
7084 });
7085
7086 // using the "anim" property to get the Anim object
7087 var opt = {
7088     duration: 1,
7089     callback: this.foo,
7090     scope: this
7091 };
7092 el.setWidth(100, opt);
7093 ...
7094 if(opt.anim.isAnimated()){
7095     opt.anim.stop();
7096 }
7097 </code></pre>
7098 * <b> Composite (Collections of) Elements</b><br />
7099  * For working with collections of Elements, see <a href="Roo.CompositeElement.html">Roo.CompositeElement</a>
7100  * @constructor Create a new Element directly.
7101  * @param {String/HTMLElement} element
7102  * @param {Boolean} forceNew (optional) By default the constructor checks to see if there is already an instance of this element in the cache and if there is it returns the same instance. This will skip that check (useful for extending this class).
7103  */
7104     Roo.Element = function(element, forceNew){
7105         var dom = typeof element == "string" ?
7106                 document.getElementById(element) : element;
7107         if(!dom){ // invalid id/element
7108             return null;
7109         }
7110         var id = dom.id;
7111         if(forceNew !== true && id && Roo.Element.cache[id]){ // element object already exists
7112             return Roo.Element.cache[id];
7113         }
7114
7115         /**
7116          * The DOM element
7117          * @type HTMLElement
7118          */
7119         this.dom = dom;
7120
7121         /**
7122          * The DOM element ID
7123          * @type String
7124          */
7125         this.id = id || Roo.id(dom);
7126     };
7127
7128     var El = Roo.Element;
7129
7130     El.prototype = {
7131         /**
7132          * The element's default display mode  (defaults to "")
7133          * @type String
7134          */
7135         originalDisplay : "",
7136
7137         visibilityMode : 1,
7138         /**
7139          * The default unit to append to CSS values where a unit isn't provided (defaults to px).
7140          * @type String
7141          */
7142         defaultUnit : "px",
7143         
7144         /**
7145          * Sets the element's visibility mode. When setVisible() is called it
7146          * will use this to determine whether to set the visibility or the display property.
7147          * @param visMode Element.VISIBILITY or Element.DISPLAY
7148          * @return {Roo.Element} this
7149          */
7150         setVisibilityMode : function(visMode){
7151             this.visibilityMode = visMode;
7152             return this;
7153         },
7154         /**
7155          * Convenience method for setVisibilityMode(Element.DISPLAY)
7156          * @param {String} display (optional) What to set display to when visible
7157          * @return {Roo.Element} this
7158          */
7159         enableDisplayMode : function(display){
7160             this.setVisibilityMode(El.DISPLAY);
7161             if(typeof display != "undefined") { this.originalDisplay = display; }
7162             return this;
7163         },
7164
7165         /**
7166          * Looks at this node and then at parent nodes for a match of the passed simple selector (e.g. div.some-class or span:first-child)
7167          * @param {String} selector The simple selector to test
7168          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7169                 search as a number or element (defaults to 10 || document.body)
7170          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
7171          * @return {HTMLElement} The matching DOM node (or null if no match was found)
7172          */
7173         findParent : function(simpleSelector, maxDepth, returnEl){
7174             var p = this.dom, b = document.body, depth = 0, dq = Roo.DomQuery, stopEl;
7175             maxDepth = maxDepth || 50;
7176             if(typeof maxDepth != "number"){
7177                 stopEl = Roo.getDom(maxDepth);
7178                 maxDepth = 10;
7179             }
7180             while(p && p.nodeType == 1 && depth < maxDepth && p != b && p != stopEl){
7181                 if(dq.is(p, simpleSelector)){
7182                     return returnEl ? Roo.get(p) : p;
7183                 }
7184                 depth++;
7185                 p = p.parentNode;
7186             }
7187             return null;
7188         },
7189
7190
7191         /**
7192          * Looks at parent nodes for a match of the passed simple selector (e.g. div.some-class or span:first-child)
7193          * @param {String} selector The simple selector to test
7194          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7195                 search as a number or element (defaults to 10 || document.body)
7196          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
7197          * @return {HTMLElement} The matching DOM node (or null if no match was found)
7198          */
7199         findParentNode : function(simpleSelector, maxDepth, returnEl){
7200             var p = Roo.fly(this.dom.parentNode, '_internal');
7201             return p ? p.findParent(simpleSelector, maxDepth, returnEl) : null;
7202         },
7203         
7204         /**
7205          * Looks at  the scrollable parent element
7206          */
7207         findScrollableParent : function()
7208         {
7209             var overflowRegex = /(auto|scroll)/;
7210             
7211             if(this.getStyle('position') === 'fixed'){
7212                 return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7213             }
7214             
7215             var excludeStaticParent = this.getStyle('position') === "absolute";
7216             
7217             for (var parent = this; (parent = Roo.get(parent.dom.parentNode));){
7218                 
7219                 if (excludeStaticParent && parent.getStyle('position') === "static") {
7220                     continue;
7221                 }
7222                 
7223                 if (overflowRegex.test(parent.getStyle('overflow') + parent.getStyle('overflow-x') + parent.getStyle('overflow-y'))){
7224                     return parent;
7225                 }
7226                 
7227                 if(parent.dom.nodeName.toLowerCase() == 'body'){
7228                     return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7229                 }
7230             }
7231             
7232             return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7233         },
7234
7235         /**
7236          * Walks up the dom looking for a parent node that matches the passed simple selector (e.g. div.some-class or span:first-child).
7237          * This is a shortcut for findParentNode() that always returns an Roo.Element.
7238          * @param {String} selector The simple selector to test
7239          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7240                 search as a number or element (defaults to 10 || document.body)
7241          * @return {Roo.Element} The matching DOM node (or null if no match was found)
7242          */
7243         up : function(simpleSelector, maxDepth){
7244             return this.findParentNode(simpleSelector, maxDepth, true);
7245         },
7246
7247
7248
7249         /**
7250          * Returns true if this element matches the passed simple selector (e.g. div.some-class or span:first-child)
7251          * @param {String} selector The simple selector to test
7252          * @return {Boolean} True if this element matches the selector, else false
7253          */
7254         is : function(simpleSelector){
7255             return Roo.DomQuery.is(this.dom, simpleSelector);
7256         },
7257
7258         /**
7259          * Perform animation on this element.
7260          * @param {Object} args The YUI animation control args
7261          * @param {Float} duration (optional) How long the animation lasts in seconds (defaults to .35)
7262          * @param {Function} onComplete (optional) Function to call when animation completes
7263          * @param {String} easing (optional) Easing method to use (defaults to 'easeOut')
7264          * @param {String} animType (optional) 'run' is the default. Can also be 'color', 'motion', or 'scroll'
7265          * @return {Roo.Element} this
7266          */
7267         animate : function(args, duration, onComplete, easing, animType){
7268             this.anim(args, {duration: duration, callback: onComplete, easing: easing}, animType);
7269             return this;
7270         },
7271
7272         /*
7273          * @private Internal animation call
7274          */
7275         anim : function(args, opt, animType, defaultDur, defaultEase, cb){
7276             animType = animType || 'run';
7277             opt = opt || {};
7278             var anim = Roo.lib.Anim[animType](
7279                 this.dom, args,
7280                 (opt.duration || defaultDur) || .35,
7281                 (opt.easing || defaultEase) || 'easeOut',
7282                 function(){
7283                     Roo.callback(cb, this);
7284                     Roo.callback(opt.callback, opt.scope || this, [this, opt]);
7285                 },
7286                 this
7287             );
7288             opt.anim = anim;
7289             return anim;
7290         },
7291
7292         // private legacy anim prep
7293         preanim : function(a, i){
7294             return !a[i] ? false : (typeof a[i] == "object" ? a[i]: {duration: a[i+1], callback: a[i+2], easing: a[i+3]});
7295         },
7296
7297         /**
7298          * Removes worthless text nodes
7299          * @param {Boolean} forceReclean (optional) By default the element
7300          * keeps track if it has been cleaned already so
7301          * you can call this over and over. However, if you update the element and
7302          * need to force a reclean, you can pass true.
7303          */
7304         clean : function(forceReclean){
7305             if(this.isCleaned && forceReclean !== true){
7306                 return this;
7307             }
7308             var ns = /\S/;
7309             var d = this.dom, n = d.firstChild, ni = -1;
7310             while(n){
7311                 var nx = n.nextSibling;
7312                 if(n.nodeType == 3 && !ns.test(n.nodeValue)){
7313                     d.removeChild(n);
7314                 }else{
7315                     n.nodeIndex = ++ni;
7316                 }
7317                 n = nx;
7318             }
7319             this.isCleaned = true;
7320             return this;
7321         },
7322
7323         // private
7324         calcOffsetsTo : function(el){
7325             el = Roo.get(el);
7326             var d = el.dom;
7327             var restorePos = false;
7328             if(el.getStyle('position') == 'static'){
7329                 el.position('relative');
7330                 restorePos = true;
7331             }
7332             var x = 0, y =0;
7333             var op = this.dom;
7334             while(op && op != d && op.tagName != 'HTML'){
7335                 x+= op.offsetLeft;
7336                 y+= op.offsetTop;
7337                 op = op.offsetParent;
7338             }
7339             if(restorePos){
7340                 el.position('static');
7341             }
7342             return [x, y];
7343         },
7344
7345         /**
7346          * Scrolls this element into view within the passed container.
7347          * @param {String/HTMLElement/Element} container (optional) The container element to scroll (defaults to document.body)
7348          * @param {Boolean} hscroll (optional) False to disable horizontal scroll (defaults to true)
7349          * @return {Roo.Element} this
7350          */
7351         scrollIntoView : function(container, hscroll){
7352             var c = Roo.getDom(container) || document.body;
7353             var el = this.dom;
7354
7355             var o = this.calcOffsetsTo(c),
7356                 l = o[0],
7357                 t = o[1],
7358                 b = t+el.offsetHeight,
7359                 r = l+el.offsetWidth;
7360
7361             var ch = c.clientHeight;
7362             var ct = parseInt(c.scrollTop, 10);
7363             var cl = parseInt(c.scrollLeft, 10);
7364             var cb = ct + ch;
7365             var cr = cl + c.clientWidth;
7366
7367             if(t < ct){
7368                 c.scrollTop = t;
7369             }else if(b > cb){
7370                 c.scrollTop = b-ch;
7371             }
7372
7373             if(hscroll !== false){
7374                 if(l < cl){
7375                     c.scrollLeft = l;
7376                 }else if(r > cr){
7377                     c.scrollLeft = r-c.clientWidth;
7378                 }
7379             }
7380             return this;
7381         },
7382
7383         // private
7384         scrollChildIntoView : function(child, hscroll){
7385             Roo.fly(child, '_scrollChildIntoView').scrollIntoView(this, hscroll);
7386         },
7387
7388         /**
7389          * Measures the element's content height and updates height to match. Note: this function uses setTimeout so
7390          * the new height may not be available immediately.
7391          * @param {Boolean} animate (optional) Animate the transition (defaults to false)
7392          * @param {Float} duration (optional) Length of the animation in seconds (defaults to .35)
7393          * @param {Function} onComplete (optional) Function to call when animation completes
7394          * @param {String} easing (optional) Easing method to use (defaults to easeOut)
7395          * @return {Roo.Element} this
7396          */
7397         autoHeight : function(animate, duration, onComplete, easing){
7398             var oldHeight = this.getHeight();
7399             this.clip();
7400             this.setHeight(1); // force clipping
7401             setTimeout(function(){
7402                 var height = parseInt(this.dom.scrollHeight, 10); // parseInt for Safari
7403                 if(!animate){
7404                     this.setHeight(height);
7405                     this.unclip();
7406                     if(typeof onComplete == "function"){
7407                         onComplete();
7408                     }
7409                 }else{
7410                     this.setHeight(oldHeight); // restore original height
7411                     this.setHeight(height, animate, duration, function(){
7412                         this.unclip();
7413                         if(typeof onComplete == "function") { onComplete(); }
7414                     }.createDelegate(this), easing);
7415                 }
7416             }.createDelegate(this), 0);
7417             return this;
7418         },
7419
7420         /**
7421          * Returns true if this element is an ancestor of the passed element
7422          * @param {HTMLElement/String} el The element to check
7423          * @return {Boolean} True if this element is an ancestor of el, else false
7424          */
7425         contains : function(el){
7426             if(!el){return false;}
7427             return D.isAncestor(this.dom, el.dom ? el.dom : el);
7428         },
7429
7430         /**
7431          * Checks whether the element is currently visible using both visibility and display properties.
7432          * @param {Boolean} deep (optional) True to walk the dom and see if parent elements are hidden (defaults to false)
7433          * @return {Boolean} True if the element is currently visible, else false
7434          */
7435         isVisible : function(deep) {
7436             var vis = !(this.getStyle("visibility") == "hidden" || this.getStyle("display") == "none");
7437             if(deep !== true || !vis){
7438                 return vis;
7439             }
7440             var p = this.dom.parentNode;
7441             while(p && p.tagName.toLowerCase() != "body"){
7442                 if(!Roo.fly(p, '_isVisible').isVisible()){
7443                     return false;
7444                 }
7445                 p = p.parentNode;
7446             }
7447             return true;
7448         },
7449
7450         /**
7451          * Creates a {@link Roo.CompositeElement} for child nodes based on the passed CSS selector (the selector should not contain an id).
7452          * @param {String} selector The CSS selector
7453          * @param {Boolean} unique (optional) True to create a unique Roo.Element for each child (defaults to false, which creates a single shared flyweight object)
7454          * @return {CompositeElement/CompositeElementLite} The composite element
7455          */
7456         select : function(selector, unique){
7457             return El.select(selector, unique, this.dom);
7458         },
7459
7460         /**
7461          * Selects child nodes based on the passed CSS selector (the selector should not contain an id).
7462          * @param {String} selector The CSS selector
7463          * @return {Array} An array of the matched nodes
7464          */
7465         query : function(selector, unique){
7466             return Roo.DomQuery.select(selector, this.dom);
7467         },
7468
7469         /**
7470          * Selects a single child at any depth below this element based on the passed CSS selector (the selector should not contain an id).
7471          * @param {String} selector The CSS selector
7472          * @param {Boolean} returnDom (optional) True to return the DOM node instead of Roo.Element (defaults to false)
7473          * @return {HTMLElement/Roo.Element} The child Roo.Element (or DOM node if returnDom = true)
7474          */
7475         child : function(selector, returnDom){
7476             var n = Roo.DomQuery.selectNode(selector, this.dom);
7477             return returnDom ? n : Roo.get(n);
7478         },
7479
7480         /**
7481          * Selects a single *direct* child based on the passed CSS selector (the selector should not contain an id).
7482          * @param {String} selector The CSS selector
7483          * @param {Boolean} returnDom (optional) True to return the DOM node instead of Roo.Element (defaults to false)
7484          * @return {HTMLElement/Roo.Element} The child Roo.Element (or DOM node if returnDom = true)
7485          */
7486         down : function(selector, returnDom){
7487             var n = Roo.DomQuery.selectNode(" > " + selector, this.dom);
7488             return returnDom ? n : Roo.get(n);
7489         },
7490
7491         /**
7492          * Initializes a {@link Roo.dd.DD} drag drop object for this element.
7493          * @param {String} group The group the DD object is member of
7494          * @param {Object} config The DD config object
7495          * @param {Object} overrides An object containing methods to override/implement on the DD object
7496          * @return {Roo.dd.DD} The DD object
7497          */
7498         initDD : function(group, config, overrides){
7499             var dd = new Roo.dd.DD(Roo.id(this.dom), group, config);
7500             return Roo.apply(dd, overrides);
7501         },
7502
7503         /**
7504          * Initializes a {@link Roo.dd.DDProxy} object for this element.
7505          * @param {String} group The group the DDProxy object is member of
7506          * @param {Object} config The DDProxy config object
7507          * @param {Object} overrides An object containing methods to override/implement on the DDProxy object
7508          * @return {Roo.dd.DDProxy} The DDProxy object
7509          */
7510         initDDProxy : function(group, config, overrides){
7511             var dd = new Roo.dd.DDProxy(Roo.id(this.dom), group, config);
7512             return Roo.apply(dd, overrides);
7513         },
7514
7515         /**
7516          * Initializes a {@link Roo.dd.DDTarget} object for this element.
7517          * @param {String} group The group the DDTarget object is member of
7518          * @param {Object} config The DDTarget config object
7519          * @param {Object} overrides An object containing methods to override/implement on the DDTarget object
7520          * @return {Roo.dd.DDTarget} The DDTarget object
7521          */
7522         initDDTarget : function(group, config, overrides){
7523             var dd = new Roo.dd.DDTarget(Roo.id(this.dom), group, config);
7524             return Roo.apply(dd, overrides);
7525         },
7526
7527         /**
7528          * Sets the visibility of the element (see details). If the visibilityMode is set to Element.DISPLAY, it will use
7529          * the display property to hide the element, otherwise it uses visibility. The default is to hide and show using the visibility property.
7530          * @param {Boolean} visible Whether the element is visible
7531          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7532          * @return {Roo.Element} this
7533          */
7534          setVisible : function(visible, animate){
7535             if(!animate || !A){
7536                 if(this.visibilityMode == El.DISPLAY){
7537                     this.setDisplayed(visible);
7538                 }else{
7539                     this.fixDisplay();
7540                     this.dom.style.visibility = visible ? "visible" : "hidden";
7541                 }
7542             }else{
7543                 // closure for composites
7544                 var dom = this.dom;
7545                 var visMode = this.visibilityMode;
7546                 if(visible){
7547                     this.setOpacity(.01);
7548                     this.setVisible(true);
7549                 }
7550                 this.anim({opacity: { to: (visible?1:0) }},
7551                       this.preanim(arguments, 1),
7552                       null, .35, 'easeIn', function(){
7553                          if(!visible){
7554                              if(visMode == El.DISPLAY){
7555                                  dom.style.display = "none";
7556                              }else{
7557                                  dom.style.visibility = "hidden";
7558                              }
7559                              Roo.get(dom).setOpacity(1);
7560                          }
7561                      });
7562             }
7563             return this;
7564         },
7565
7566         /**
7567          * Returns true if display is not "none"
7568          * @return {Boolean}
7569          */
7570         isDisplayed : function() {
7571             return this.getStyle("display") != "none";
7572         },
7573
7574         /**
7575          * Toggles the element's visibility or display, depending on visibility mode.
7576          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7577          * @return {Roo.Element} this
7578          */
7579         toggle : function(animate){
7580             this.setVisible(!this.isVisible(), this.preanim(arguments, 0));
7581             return this;
7582         },
7583
7584         /**
7585          * Sets the CSS display property. Uses originalDisplay if the specified value is a boolean true.
7586          * @param {Boolean} value Boolean value to display the element using its default display, or a string to set the display directly
7587          * @return {Roo.Element} this
7588          */
7589         setDisplayed : function(value) {
7590             if(typeof value == "boolean"){
7591                value = value ? this.originalDisplay : "none";
7592             }
7593             this.setStyle("display", value);
7594             return this;
7595         },
7596
7597         /**
7598          * Tries to focus the element. Any exceptions are caught and ignored.
7599          * @return {Roo.Element} this
7600          */
7601         focus : function() {
7602             try{
7603                 this.dom.focus();
7604             }catch(e){}
7605             return this;
7606         },
7607
7608         /**
7609          * Tries to blur the element. Any exceptions are caught and ignored.
7610          * @return {Roo.Element} this
7611          */
7612         blur : function() {
7613             try{
7614                 this.dom.blur();
7615             }catch(e){}
7616             return this;
7617         },
7618
7619         /**
7620          * Adds one or more CSS classes to the element. Duplicate classes are automatically filtered out.
7621          * @param {String/Array} className The CSS class to add, or an array of classes
7622          * @return {Roo.Element} this
7623          */
7624         addClass : function(className){
7625             if(className instanceof Array){
7626                 for(var i = 0, len = className.length; i < len; i++) {
7627                     this.addClass(className[i]);
7628                 }
7629             }else{
7630                 if(className && !this.hasClass(className)){
7631                     this.dom.className = this.dom.className + " " + className;
7632                 }
7633             }
7634             return this;
7635         },
7636
7637         /**
7638          * Adds one or more CSS classes to this element and removes the same class(es) from all siblings.
7639          * @param {String/Array} className The CSS class to add, or an array of classes
7640          * @return {Roo.Element} this
7641          */
7642         radioClass : function(className){
7643             var siblings = this.dom.parentNode.childNodes;
7644             for(var i = 0; i < siblings.length; i++) {
7645                 var s = siblings[i];
7646                 if(s.nodeType == 1){
7647                     Roo.get(s).removeClass(className);
7648                 }
7649             }
7650             this.addClass(className);
7651             return this;
7652         },
7653
7654         /**
7655          * Removes one or more CSS classes from the element.
7656          * @param {String/Array} className The CSS class to remove, or an array of classes
7657          * @return {Roo.Element} this
7658          */
7659         removeClass : function(className){
7660             if(!className || !this.dom.className){
7661                 return this;
7662             }
7663             if(className instanceof Array){
7664                 for(var i = 0, len = className.length; i < len; i++) {
7665                     this.removeClass(className[i]);
7666                 }
7667             }else{
7668                 if(this.hasClass(className)){
7669                     var re = this.classReCache[className];
7670                     if (!re) {
7671                        re = new RegExp('(?:^|\\s+)' + className + '(?:\\s+|$)', "g");
7672                        this.classReCache[className] = re;
7673                     }
7674                     this.dom.className =
7675                         this.dom.className.replace(re, " ");
7676                 }
7677             }
7678             return this;
7679         },
7680
7681         // private
7682         classReCache: {},
7683
7684         /**
7685          * Toggles the specified CSS class on this element (removes it if it already exists, otherwise adds it).
7686          * @param {String} className The CSS class to toggle
7687          * @return {Roo.Element} this
7688          */
7689         toggleClass : function(className){
7690             if(this.hasClass(className)){
7691                 this.removeClass(className);
7692             }else{
7693                 this.addClass(className);
7694             }
7695             return this;
7696         },
7697
7698         /**
7699          * Checks if the specified CSS class exists on this element's DOM node.
7700          * @param {String} className The CSS class to check for
7701          * @return {Boolean} True if the class exists, else false
7702          */
7703         hasClass : function(className){
7704             return className && (' '+this.dom.className+' ').indexOf(' '+className+' ') != -1;
7705         },
7706
7707         /**
7708          * Replaces a CSS class on the element with another.  If the old name does not exist, the new name will simply be added.
7709          * @param {String} oldClassName The CSS class to replace
7710          * @param {String} newClassName The replacement CSS class
7711          * @return {Roo.Element} this
7712          */
7713         replaceClass : function(oldClassName, newClassName){
7714             this.removeClass(oldClassName);
7715             this.addClass(newClassName);
7716             return this;
7717         },
7718
7719         /**
7720          * Returns an object with properties matching the styles requested.
7721          * For example, el.getStyles('color', 'font-size', 'width') might return
7722          * {'color': '#FFFFFF', 'font-size': '13px', 'width': '100px'}.
7723          * @param {String} style1 A style name
7724          * @param {String} style2 A style name
7725          * @param {String} etc.
7726          * @return {Object} The style object
7727          */
7728         getStyles : function(){
7729             var a = arguments, len = a.length, r = {};
7730             for(var i = 0; i < len; i++){
7731                 r[a[i]] = this.getStyle(a[i]);
7732             }
7733             return r;
7734         },
7735
7736         /**
7737          * Normalizes currentStyle and computedStyle. This is not YUI getStyle, it is an optimised version.
7738          * @param {String} property The style property whose value is returned.
7739          * @return {String} The current value of the style property for this element.
7740          */
7741         getStyle : function(){
7742             return view && view.getComputedStyle ?
7743                 function(prop){
7744                     var el = this.dom, v, cs, camel;
7745                     if(prop == 'float'){
7746                         prop = "cssFloat";
7747                     }
7748                     if(el.style && (v = el.style[prop])){
7749                         return v;
7750                     }
7751                     if(cs = view.getComputedStyle(el, "")){
7752                         if(!(camel = propCache[prop])){
7753                             camel = propCache[prop] = prop.replace(camelRe, camelFn);
7754                         }
7755                         return cs[camel];
7756                     }
7757                     return null;
7758                 } :
7759                 function(prop){
7760                     var el = this.dom, v, cs, camel;
7761                     if(prop == 'opacity'){
7762                         if(typeof el.style.filter == 'string'){
7763                             var m = el.style.filter.match(/alpha\(opacity=(.*)\)/i);
7764                             if(m){
7765                                 var fv = parseFloat(m[1]);
7766                                 if(!isNaN(fv)){
7767                                     return fv ? fv / 100 : 0;
7768                                 }
7769                             }
7770                         }
7771                         return 1;
7772                     }else if(prop == 'float'){
7773                         prop = "styleFloat";
7774                     }
7775                     if(!(camel = propCache[prop])){
7776                         camel = propCache[prop] = prop.replace(camelRe, camelFn);
7777                     }
7778                     if(v = el.style[camel]){
7779                         return v;
7780                     }
7781                     if(cs = el.currentStyle){
7782                         return cs[camel];
7783                     }
7784                     return null;
7785                 };
7786         }(),
7787
7788         /**
7789          * Wrapper for setting style properties, also takes single object parameter of multiple styles.
7790          * @param {String/Object} property The style property to be set, or an object of multiple styles.
7791          * @param {String} value (optional) The value to apply to the given property, or null if an object was passed.
7792          * @return {Roo.Element} this
7793          */
7794         setStyle : function(prop, value){
7795             if(typeof prop == "string"){
7796                 
7797                 if (prop == 'float') {
7798                     this.setStyle(Roo.isIE ? 'styleFloat'  : 'cssFloat', value);
7799                     return this;
7800                 }
7801                 
7802                 var camel;
7803                 if(!(camel = propCache[prop])){
7804                     camel = propCache[prop] = prop.replace(camelRe, camelFn);
7805                 }
7806                 
7807                 if(camel == 'opacity') {
7808                     this.setOpacity(value);
7809                 }else{
7810                     this.dom.style[camel] = value;
7811                 }
7812             }else{
7813                 for(var style in prop){
7814                     if(typeof prop[style] != "function"){
7815                        this.setStyle(style, prop[style]);
7816                     }
7817                 }
7818             }
7819             return this;
7820         },
7821
7822         /**
7823          * More flexible version of {@link #setStyle} for setting style properties.
7824          * @param {String/Object/Function} styles A style specification string, e.g. "width:100px", or object in the form {width:"100px"}, or
7825          * a function which returns such a specification.
7826          * @return {Roo.Element} this
7827          */
7828         applyStyles : function(style){
7829             Roo.DomHelper.applyStyles(this.dom, style);
7830             return this;
7831         },
7832
7833         /**
7834           * Gets the current X position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7835           * @return {Number} The X position of the element
7836           */
7837         getX : function(){
7838             return D.getX(this.dom);
7839         },
7840
7841         /**
7842           * Gets the current Y position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7843           * @return {Number} The Y position of the element
7844           */
7845         getY : function(){
7846             return D.getY(this.dom);
7847         },
7848
7849         /**
7850           * Gets the current position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7851           * @return {Array} The XY position of the element
7852           */
7853         getXY : function(){
7854             return D.getXY(this.dom);
7855         },
7856
7857         /**
7858          * Sets the X position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7859          * @param {Number} The X position of the element
7860          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7861          * @return {Roo.Element} this
7862          */
7863         setX : function(x, animate){
7864             if(!animate || !A){
7865                 D.setX(this.dom, x);
7866             }else{
7867                 this.setXY([x, this.getY()], this.preanim(arguments, 1));
7868             }
7869             return this;
7870         },
7871
7872         /**
7873          * Sets the Y position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7874          * @param {Number} The Y position of the element
7875          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7876          * @return {Roo.Element} this
7877          */
7878         setY : function(y, animate){
7879             if(!animate || !A){
7880                 D.setY(this.dom, y);
7881             }else{
7882                 this.setXY([this.getX(), y], this.preanim(arguments, 1));
7883             }
7884             return this;
7885         },
7886
7887         /**
7888          * Sets the element's left position directly using CSS style (instead of {@link #setX}).
7889          * @param {String} left The left CSS property value
7890          * @return {Roo.Element} this
7891          */
7892         setLeft : function(left){
7893             this.setStyle("left", this.addUnits(left));
7894             return this;
7895         },
7896
7897         /**
7898          * Sets the element's top position directly using CSS style (instead of {@link #setY}).
7899          * @param {String} top The top CSS property value
7900          * @return {Roo.Element} this
7901          */
7902         setTop : function(top){
7903             this.setStyle("top", this.addUnits(top));
7904             return this;
7905         },
7906
7907         /**
7908          * Sets the element's CSS right style.
7909          * @param {String} right The right CSS property value
7910          * @return {Roo.Element} this
7911          */
7912         setRight : function(right){
7913             this.setStyle("right", this.addUnits(right));
7914             return this;
7915         },
7916
7917         /**
7918          * Sets the element's CSS bottom style.
7919          * @param {String} bottom The bottom CSS property value
7920          * @return {Roo.Element} this
7921          */
7922         setBottom : function(bottom){
7923             this.setStyle("bottom", this.addUnits(bottom));
7924             return this;
7925         },
7926
7927         /**
7928          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7929          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7930          * @param {Array} pos Contains X & Y [x, y] values for new position (coordinates are page-based)
7931          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7932          * @return {Roo.Element} this
7933          */
7934         setXY : function(pos, animate){
7935             if(!animate || !A){
7936                 D.setXY(this.dom, pos);
7937             }else{
7938                 this.anim({points: {to: pos}}, this.preanim(arguments, 1), 'motion');
7939             }
7940             return this;
7941         },
7942
7943         /**
7944          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7945          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7946          * @param {Number} x X value for new position (coordinates are page-based)
7947          * @param {Number} y Y value for new position (coordinates are page-based)
7948          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7949          * @return {Roo.Element} this
7950          */
7951         setLocation : function(x, y, animate){
7952             this.setXY([x, y], this.preanim(arguments, 2));
7953             return this;
7954         },
7955
7956         /**
7957          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7958          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7959          * @param {Number} x X value for new position (coordinates are page-based)
7960          * @param {Number} y Y value for new position (coordinates are page-based)
7961          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7962          * @return {Roo.Element} this
7963          */
7964         moveTo : function(x, y, animate){
7965             this.setXY([x, y], this.preanim(arguments, 2));
7966             return this;
7967         },
7968
7969         /**
7970          * Returns the region of the given element.
7971          * The element must be part of the DOM tree to have a region (display:none or elements not appended return false).
7972          * @return {Region} A Roo.lib.Region containing "top, left, bottom, right" member data.
7973          */
7974         getRegion : function(){
7975             return D.getRegion(this.dom);
7976         },
7977
7978         /**
7979          * Returns the offset height of the element
7980          * @param {Boolean} contentHeight (optional) true to get the height minus borders and padding
7981          * @return {Number} The element's height
7982          */
7983         getHeight : function(contentHeight){
7984             var h = this.dom.offsetHeight || 0;
7985             return contentHeight !== true ? h : h-this.getBorderWidth("tb")-this.getPadding("tb");
7986         },
7987
7988         /**
7989          * Returns the offset width of the element
7990          * @param {Boolean} contentWidth (optional) true to get the width minus borders and padding
7991          * @return {Number} The element's width
7992          */
7993         getWidth : function(contentWidth){
7994             var w = this.dom.offsetWidth || 0;
7995             return contentWidth !== true ? w : w-this.getBorderWidth("lr")-this.getPadding("lr");
7996         },
7997
7998         /**
7999          * Returns either the offsetHeight or the height of this element based on CSS height adjusted by padding or borders
8000          * when needed to simulate offsetHeight when offsets aren't available. This may not work on display:none elements
8001          * if a height has not been set using CSS.
8002          * @return {Number}
8003          */
8004         getComputedHeight : function(){
8005             var h = Math.max(this.dom.offsetHeight, this.dom.clientHeight);
8006             if(!h){
8007                 h = parseInt(this.getStyle('height'), 10) || 0;
8008                 if(!this.isBorderBox()){
8009                     h += this.getFrameWidth('tb');
8010                 }
8011             }
8012             return h;
8013         },
8014
8015         /**
8016          * Returns either the offsetWidth or the width of this element based on CSS width adjusted by padding or borders
8017          * when needed to simulate offsetWidth when offsets aren't available. This may not work on display:none elements
8018          * if a width has not been set using CSS.
8019          * @return {Number}
8020          */
8021         getComputedWidth : function(){
8022             var w = Math.max(this.dom.offsetWidth, this.dom.clientWidth);
8023             if(!w){
8024                 w = parseInt(this.getStyle('width'), 10) || 0;
8025                 if(!this.isBorderBox()){
8026                     w += this.getFrameWidth('lr');
8027                 }
8028             }
8029             return w;
8030         },
8031
8032         /**
8033          * Returns the size of the element.
8034          * @param {Boolean} contentSize (optional) true to get the width/size minus borders and padding
8035          * @return {Object} An object containing the element's size {width: (element width), height: (element height)}
8036          */
8037         getSize : function(contentSize){
8038             return {width: this.getWidth(contentSize), height: this.getHeight(contentSize)};
8039         },
8040
8041         /**
8042          * Returns the width and height of the viewport.
8043          * @return {Object} An object containing the viewport's size {width: (viewport width), height: (viewport height)}
8044          */
8045         getViewSize : function(){
8046             var d = this.dom, doc = document, aw = 0, ah = 0;
8047             if(d == doc || d == doc.body){
8048                 return {width : D.getViewWidth(), height: D.getViewHeight()};
8049             }else{
8050                 return {
8051                     width : d.clientWidth,
8052                     height: d.clientHeight
8053                 };
8054             }
8055         },
8056
8057         /**
8058          * Returns the value of the "value" attribute
8059          * @param {Boolean} asNumber true to parse the value as a number
8060          * @return {String/Number}
8061          */
8062         getValue : function(asNumber){
8063             return asNumber ? parseInt(this.dom.value, 10) : this.dom.value;
8064         },
8065
8066         // private
8067         adjustWidth : function(width){
8068             if(typeof width == "number"){
8069                 if(this.autoBoxAdjust && !this.isBorderBox()){
8070                    width -= (this.getBorderWidth("lr") + this.getPadding("lr"));
8071                 }
8072                 if(width < 0){
8073                     width = 0;
8074                 }
8075             }
8076             return width;
8077         },
8078
8079         // private
8080         adjustHeight : function(height){
8081             if(typeof height == "number"){
8082                if(this.autoBoxAdjust && !this.isBorderBox()){
8083                    height -= (this.getBorderWidth("tb") + this.getPadding("tb"));
8084                }
8085                if(height < 0){
8086                    height = 0;
8087                }
8088             }
8089             return height;
8090         },
8091
8092         /**
8093          * Set the width of the element
8094          * @param {Number} width The new width
8095          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8096          * @return {Roo.Element} this
8097          */
8098         setWidth : function(width, animate){
8099             width = this.adjustWidth(width);
8100             if(!animate || !A){
8101                 this.dom.style.width = this.addUnits(width);
8102             }else{
8103                 this.anim({width: {to: width}}, this.preanim(arguments, 1));
8104             }
8105             return this;
8106         },
8107
8108         /**
8109          * Set the height of the element
8110          * @param {Number} height The new height
8111          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8112          * @return {Roo.Element} this
8113          */
8114          setHeight : function(height, animate){
8115             height = this.adjustHeight(height);
8116             if(!animate || !A){
8117                 this.dom.style.height = this.addUnits(height);
8118             }else{
8119                 this.anim({height: {to: height}}, this.preanim(arguments, 1));
8120             }
8121             return this;
8122         },
8123
8124         /**
8125          * Set the size of the element. If animation is true, both width an height will be animated concurrently.
8126          * @param {Number} width The new width
8127          * @param {Number} height The new height
8128          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8129          * @return {Roo.Element} this
8130          */
8131          setSize : function(width, height, animate){
8132             if(typeof width == "object"){ // in case of object from getSize()
8133                 height = width.height; width = width.width;
8134             }
8135             width = this.adjustWidth(width); height = this.adjustHeight(height);
8136             if(!animate || !A){
8137                 this.dom.style.width = this.addUnits(width);
8138                 this.dom.style.height = this.addUnits(height);
8139             }else{
8140                 this.anim({width: {to: width}, height: {to: height}}, this.preanim(arguments, 2));
8141             }
8142             return this;
8143         },
8144
8145         /**
8146          * Sets the element's position and size in one shot. If animation is true then width, height, x and y will be animated concurrently.
8147          * @param {Number} x X value for new position (coordinates are page-based)
8148          * @param {Number} y Y value for new position (coordinates are page-based)
8149          * @param {Number} width The new width
8150          * @param {Number} height The new height
8151          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8152          * @return {Roo.Element} this
8153          */
8154         setBounds : function(x, y, width, height, animate){
8155             if(!animate || !A){
8156                 this.setSize(width, height);
8157                 this.setLocation(x, y);
8158             }else{
8159                 width = this.adjustWidth(width); height = this.adjustHeight(height);
8160                 this.anim({points: {to: [x, y]}, width: {to: width}, height: {to: height}},
8161                               this.preanim(arguments, 4), 'motion');
8162             }
8163             return this;
8164         },
8165
8166         /**
8167          * Sets the element's position and size the the specified region. If animation is true then width, height, x and y will be animated concurrently.
8168          * @param {Roo.lib.Region} region The region to fill
8169          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8170          * @return {Roo.Element} this
8171          */
8172         setRegion : function(region, animate){
8173             this.setBounds(region.left, region.top, region.right-region.left, region.bottom-region.top, this.preanim(arguments, 1));
8174             return this;
8175         },
8176
8177         /**
8178          * Appends an event handler
8179          *
8180          * @param {String}   eventName     The type of event to append
8181          * @param {Function} fn        The method the event invokes
8182          * @param {Object} scope       (optional) The scope (this object) of the fn
8183          * @param {Object}   options   (optional)An object with standard {@link Roo.EventManager#addListener} options
8184          */
8185         addListener : function(eventName, fn, scope, options){
8186             if (this.dom) {
8187                 Roo.EventManager.on(this.dom,  eventName, fn, scope || this, options);
8188             }
8189         },
8190
8191         /**
8192          * Removes an event handler from this element
8193          * @param {String} eventName the type of event to remove
8194          * @param {Function} fn the method the event invokes
8195          * @return {Roo.Element} this
8196          */
8197         removeListener : function(eventName, fn){
8198             Roo.EventManager.removeListener(this.dom,  eventName, fn);
8199             return this;
8200         },
8201
8202         /**
8203          * Removes all previous added listeners from this element
8204          * @return {Roo.Element} this
8205          */
8206         removeAllListeners : function(){
8207             E.purgeElement(this.dom);
8208             return this;
8209         },
8210
8211         relayEvent : function(eventName, observable){
8212             this.on(eventName, function(e){
8213                 observable.fireEvent(eventName, e);
8214             });
8215         },
8216
8217         /**
8218          * Set the opacity of the element
8219          * @param {Float} opacity The new opacity. 0 = transparent, .5 = 50% visibile, 1 = fully visible, etc
8220          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8221          * @return {Roo.Element} this
8222          */
8223          setOpacity : function(opacity, animate){
8224             if(!animate || !A){
8225                 var s = this.dom.style;
8226                 if(Roo.isIE){
8227                     s.zoom = 1;
8228                     s.filter = (s.filter || '').replace(/alpha\([^\)]*\)/gi,"") +
8229                                (opacity == 1 ? "" : "alpha(opacity=" + opacity * 100 + ")");
8230                 }else{
8231                     s.opacity = opacity;
8232                 }
8233             }else{
8234                 this.anim({opacity: {to: opacity}}, this.preanim(arguments, 1), null, .35, 'easeIn');
8235             }
8236             return this;
8237         },
8238
8239         /**
8240          * Gets the left X coordinate
8241          * @param {Boolean} local True to get the local css position instead of page coordinate
8242          * @return {Number}
8243          */
8244         getLeft : function(local){
8245             if(!local){
8246                 return this.getX();
8247             }else{
8248                 return parseInt(this.getStyle("left"), 10) || 0;
8249             }
8250         },
8251
8252         /**
8253          * Gets the right X coordinate of the element (element X position + element width)
8254          * @param {Boolean} local True to get the local css position instead of page coordinate
8255          * @return {Number}
8256          */
8257         getRight : function(local){
8258             if(!local){
8259                 return this.getX() + this.getWidth();
8260             }else{
8261                 return (this.getLeft(true) + this.getWidth()) || 0;
8262             }
8263         },
8264
8265         /**
8266          * Gets the top Y coordinate
8267          * @param {Boolean} local True to get the local css position instead of page coordinate
8268          * @return {Number}
8269          */
8270         getTop : function(local) {
8271             if(!local){
8272                 return this.getY();
8273             }else{
8274                 return parseInt(this.getStyle("top"), 10) || 0;
8275             }
8276         },
8277
8278         /**
8279          * Gets the bottom Y coordinate of the element (element Y position + element height)
8280          * @param {Boolean} local True to get the local css position instead of page coordinate
8281          * @return {Number}
8282          */
8283         getBottom : function(local){
8284             if(!local){
8285                 return this.getY() + this.getHeight();
8286             }else{
8287                 return (this.getTop(true) + this.getHeight()) || 0;
8288             }
8289         },
8290
8291         /**
8292         * Initializes positioning on this element. If a desired position is not passed, it will make the
8293         * the element positioned relative IF it is not already positioned.
8294         * @param {String} pos (optional) Positioning to use "relative", "absolute" or "fixed"
8295         * @param {Number} zIndex (optional) The zIndex to apply
8296         * @param {Number} x (optional) Set the page X position
8297         * @param {Number} y (optional) Set the page Y position
8298         */
8299         position : function(pos, zIndex, x, y){
8300             if(!pos){
8301                if(this.getStyle('position') == 'static'){
8302                    this.setStyle('position', 'relative');
8303                }
8304             }else{
8305                 this.setStyle("position", pos);
8306             }
8307             if(zIndex){
8308                 this.setStyle("z-index", zIndex);
8309             }
8310             if(x !== undefined && y !== undefined){
8311                 this.setXY([x, y]);
8312             }else if(x !== undefined){
8313                 this.setX(x);
8314             }else if(y !== undefined){
8315                 this.setY(y);
8316             }
8317         },
8318
8319         /**
8320         * Clear positioning back to the default when the document was loaded
8321         * @param {String} value (optional) The value to use for the left,right,top,bottom, defaults to '' (empty string). You could use 'auto'.
8322         * @return {Roo.Element} this
8323          */
8324         clearPositioning : function(value){
8325             value = value ||'';
8326             this.setStyle({
8327                 "left": value,
8328                 "right": value,
8329                 "top": value,
8330                 "bottom": value,
8331                 "z-index": "",
8332                 "position" : "static"
8333             });
8334             return this;
8335         },
8336
8337         /**
8338         * Gets an object with all CSS positioning properties. Useful along with setPostioning to get
8339         * snapshot before performing an update and then restoring the element.
8340         * @return {Object}
8341         */
8342         getPositioning : function(){
8343             var l = this.getStyle("left");
8344             var t = this.getStyle("top");
8345             return {
8346                 "position" : this.getStyle("position"),
8347                 "left" : l,
8348                 "right" : l ? "" : this.getStyle("right"),
8349                 "top" : t,
8350                 "bottom" : t ? "" : this.getStyle("bottom"),
8351                 "z-index" : this.getStyle("z-index")
8352             };
8353         },
8354
8355         /**
8356          * Gets the width of the border(s) for the specified side(s)
8357          * @param {String} side Can be t, l, r, b or any combination of those to add multiple values. For example,
8358          * passing lr would get the border (l)eft width + the border (r)ight width.
8359          * @return {Number} The width of the sides passed added together
8360          */
8361         getBorderWidth : function(side){
8362             return this.addStyles(side, El.borders);
8363         },
8364
8365         /**
8366          * Gets the width of the padding(s) for the specified side(s)
8367          * @param {String} side Can be t, l, r, b or any combination of those to add multiple values. For example,
8368          * passing lr would get the padding (l)eft + the padding (r)ight.
8369          * @return {Number} The padding of the sides passed added together
8370          */
8371         getPadding : function(side){
8372             return this.addStyles(side, El.paddings);
8373         },
8374
8375         /**
8376         * Set positioning with an object returned by getPositioning().
8377         * @param {Object} posCfg
8378         * @return {Roo.Element} this
8379          */
8380         setPositioning : function(pc){
8381             this.applyStyles(pc);
8382             if(pc.right == "auto"){
8383                 this.dom.style.right = "";
8384             }
8385             if(pc.bottom == "auto"){
8386                 this.dom.style.bottom = "";
8387             }
8388             return this;
8389         },
8390
8391         // private
8392         fixDisplay : function(){
8393             if(this.getStyle("display") == "none"){
8394                 this.setStyle("visibility", "hidden");
8395                 this.setStyle("display", this.originalDisplay); // first try reverting to default
8396                 if(this.getStyle("display") == "none"){ // if that fails, default to block
8397                     this.setStyle("display", "block");
8398                 }
8399             }
8400         },
8401
8402         /**
8403          * Quick set left and top adding default units
8404          * @param {String} left The left CSS property value
8405          * @param {String} top The top CSS property value
8406          * @return {Roo.Element} this
8407          */
8408          setLeftTop : function(left, top){
8409             this.dom.style.left = this.addUnits(left);
8410             this.dom.style.top = this.addUnits(top);
8411             return this;
8412         },
8413
8414         /**
8415          * Move this element relative to its current position.
8416          * @param {String} direction Possible values are: "l","left" - "r","right" - "t","top","up" - "b","bottom","down".
8417          * @param {Number} distance How far to move the element in pixels
8418          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8419          * @return {Roo.Element} this
8420          */
8421          move : function(direction, distance, animate){
8422             var xy = this.getXY();
8423             direction = direction.toLowerCase();
8424             switch(direction){
8425                 case "l":
8426                 case "left":
8427                     this.moveTo(xy[0]-distance, xy[1], this.preanim(arguments, 2));
8428                     break;
8429                case "r":
8430                case "right":
8431                     this.moveTo(xy[0]+distance, xy[1], this.preanim(arguments, 2));
8432                     break;
8433                case "t":
8434                case "top":
8435                case "up":
8436                     this.moveTo(xy[0], xy[1]-distance, this.preanim(arguments, 2));
8437                     break;
8438                case "b":
8439                case "bottom":
8440                case "down":
8441                     this.moveTo(xy[0], xy[1]+distance, this.preanim(arguments, 2));
8442                     break;
8443             }
8444             return this;
8445         },
8446
8447         /**
8448          *  Store the current overflow setting and clip overflow on the element - use {@link #unclip} to remove
8449          * @return {Roo.Element} this
8450          */
8451         clip : function(){
8452             if(!this.isClipped){
8453                this.isClipped = true;
8454                this.originalClip = {
8455                    "o": this.getStyle("overflow"),
8456                    "x": this.getStyle("overflow-x"),
8457                    "y": this.getStyle("overflow-y")
8458                };
8459                this.setStyle("overflow", "hidden");
8460                this.setStyle("overflow-x", "hidden");
8461                this.setStyle("overflow-y", "hidden");
8462             }
8463             return this;
8464         },
8465
8466         /**
8467          *  Return clipping (overflow) to original clipping before clip() was called
8468          * @return {Roo.Element} this
8469          */
8470         unclip : function(){
8471             if(this.isClipped){
8472                 this.isClipped = false;
8473                 var o = this.originalClip;
8474                 if(o.o){this.setStyle("overflow", o.o);}
8475                 if(o.x){this.setStyle("overflow-x", o.x);}
8476                 if(o.y){this.setStyle("overflow-y", o.y);}
8477             }
8478             return this;
8479         },
8480
8481
8482         /**
8483          * Gets the x,y coordinates specified by the anchor position on the element.
8484          * @param {String} anchor (optional) The specified anchor position (defaults to "c").  See {@link #alignTo} for details on supported anchor positions.
8485          * @param {Object} size (optional) An object containing the size to use for calculating anchor position
8486          *                       {width: (target width), height: (target height)} (defaults to the element's current size)
8487          * @param {Boolean} local (optional) True to get the local (element top/left-relative) anchor position instead of page coordinates
8488          * @return {Array} [x, y] An array containing the element's x and y coordinates
8489          */
8490         getAnchorXY : function(anchor, local, s){
8491             //Passing a different size is useful for pre-calculating anchors,
8492             //especially for anchored animations that change the el size.
8493
8494             var w, h, vp = false;
8495             if(!s){
8496                 var d = this.dom;
8497                 if(d == document.body || d == document){
8498                     vp = true;
8499                     w = D.getViewWidth(); h = D.getViewHeight();
8500                 }else{
8501                     w = this.getWidth(); h = this.getHeight();
8502                 }
8503             }else{
8504                 w = s.width;  h = s.height;
8505             }
8506             var x = 0, y = 0, r = Math.round;
8507             switch((anchor || "tl").toLowerCase()){
8508                 case "c":
8509                     x = r(w*.5);
8510                     y = r(h*.5);
8511                 break;
8512                 case "t":
8513                     x = r(w*.5);
8514                     y = 0;
8515                 break;
8516                 case "l":
8517                     x = 0;
8518                     y = r(h*.5);
8519                 break;
8520                 case "r":
8521                     x = w;
8522                     y = r(h*.5);
8523                 break;
8524                 case "b":
8525                     x = r(w*.5);
8526                     y = h;
8527                 break;
8528                 case "tl":
8529                     x = 0;
8530                     y = 0;
8531                 break;
8532                 case "bl":
8533                     x = 0;
8534                     y = h;
8535                 break;
8536                 case "br":
8537                     x = w;
8538                     y = h;
8539                 break;
8540                 case "tr":
8541                     x = w;
8542                     y = 0;
8543                 break;
8544             }
8545             if(local === true){
8546                 return [x, y];
8547             }
8548             if(vp){
8549                 var sc = this.getScroll();
8550                 return [x + sc.left, y + sc.top];
8551             }
8552             //Add the element's offset xy
8553             var o = this.getXY();
8554             return [x+o[0], y+o[1]];
8555         },
8556
8557         /**
8558          * Gets the x,y coordinates to align this element with another element. See {@link #alignTo} for more info on the
8559          * supported position values.
8560          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8561          * @param {String} position The position to align to.
8562          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8563          * @return {Array} [x, y]
8564          */
8565         getAlignToXY : function(el, p, o){
8566             el = Roo.get(el);
8567             var d = this.dom;
8568             if(!el.dom){
8569                 throw "Element.alignTo with an element that doesn't exist";
8570             }
8571             var c = false; //constrain to viewport
8572             var p1 = "", p2 = "";
8573             o = o || [0,0];
8574
8575             if(!p){
8576                 p = "tl-bl";
8577             }else if(p == "?"){
8578                 p = "tl-bl?";
8579             }else if(p.indexOf("-") == -1){
8580                 p = "tl-" + p;
8581             }
8582             p = p.toLowerCase();
8583             var m = p.match(/^([a-z]+)-([a-z]+)(\?)?$/);
8584             if(!m){
8585                throw "Element.alignTo with an invalid alignment " + p;
8586             }
8587             p1 = m[1]; p2 = m[2]; c = !!m[3];
8588
8589             //Subtract the aligned el's internal xy from the target's offset xy
8590             //plus custom offset to get the aligned el's new offset xy
8591             var a1 = this.getAnchorXY(p1, true);
8592             var a2 = el.getAnchorXY(p2, false);
8593             var x = a2[0] - a1[0] + o[0];
8594             var y = a2[1] - a1[1] + o[1];
8595             if(c){
8596                 //constrain the aligned el to viewport if necessary
8597                 var w = this.getWidth(), h = this.getHeight(), r = el.getRegion();
8598                 // 5px of margin for ie
8599                 var dw = D.getViewWidth()-5, dh = D.getViewHeight()-5;
8600
8601                 //If we are at a viewport boundary and the aligned el is anchored on a target border that is
8602                 //perpendicular to the vp border, allow the aligned el to slide on that border,
8603                 //otherwise swap the aligned el to the opposite border of the target.
8604                 var p1y = p1.charAt(0), p1x = p1.charAt(p1.length-1);
8605                var p2y = p2.charAt(0), p2x = p2.charAt(p2.length-1);
8606                var swapY = ((p1y=="t" && p2y=="b") || (p1y=="b" && p2y=="t"));
8607                var swapX = ((p1x=="r" && p2x=="l") || (p1x=="l" && p2x=="r"));
8608
8609                var doc = document;
8610                var scrollX = (doc.documentElement.scrollLeft || doc.body.scrollLeft || 0)+5;
8611                var scrollY = (doc.documentElement.scrollTop || doc.body.scrollTop || 0)+5;
8612
8613                if((x+w) > dw + scrollX){
8614                     x = swapX ? r.left-w : dw+scrollX-w;
8615                 }
8616                if(x < scrollX){
8617                    x = swapX ? r.right : scrollX;
8618                }
8619                if((y+h) > dh + scrollY){
8620                     y = swapY ? r.top-h : dh+scrollY-h;
8621                 }
8622                if (y < scrollY){
8623                    y = swapY ? r.bottom : scrollY;
8624                }
8625             }
8626             return [x,y];
8627         },
8628
8629         // private
8630         getConstrainToXY : function(){
8631             var os = {top:0, left:0, bottom:0, right: 0};
8632
8633             return function(el, local, offsets, proposedXY){
8634                 el = Roo.get(el);
8635                 offsets = offsets ? Roo.applyIf(offsets, os) : os;
8636
8637                 var vw, vh, vx = 0, vy = 0;
8638                 if(el.dom == document.body || el.dom == document){
8639                     vw = Roo.lib.Dom.getViewWidth();
8640                     vh = Roo.lib.Dom.getViewHeight();
8641                 }else{
8642                     vw = el.dom.clientWidth;
8643                     vh = el.dom.clientHeight;
8644                     if(!local){
8645                         var vxy = el.getXY();
8646                         vx = vxy[0];
8647                         vy = vxy[1];
8648                     }
8649                 }
8650
8651                 var s = el.getScroll();
8652
8653                 vx += offsets.left + s.left;
8654                 vy += offsets.top + s.top;
8655
8656                 vw -= offsets.right;
8657                 vh -= offsets.bottom;
8658
8659                 var vr = vx+vw;
8660                 var vb = vy+vh;
8661
8662                 var xy = proposedXY || (!local ? this.getXY() : [this.getLeft(true), this.getTop(true)]);
8663                 var x = xy[0], y = xy[1];
8664                 var w = this.dom.offsetWidth, h = this.dom.offsetHeight;
8665
8666                 // only move it if it needs it
8667                 var moved = false;
8668
8669                 // first validate right/bottom
8670                 if((x + w) > vr){
8671                     x = vr - w;
8672                     moved = true;
8673                 }
8674                 if((y + h) > vb){
8675                     y = vb - h;
8676                     moved = true;
8677                 }
8678                 // then make sure top/left isn't negative
8679                 if(x < vx){
8680                     x = vx;
8681                     moved = true;
8682                 }
8683                 if(y < vy){
8684                     y = vy;
8685                     moved = true;
8686                 }
8687                 return moved ? [x, y] : false;
8688             };
8689         }(),
8690
8691         // private
8692         adjustForConstraints : function(xy, parent, offsets){
8693             return this.getConstrainToXY(parent || document, false, offsets, xy) ||  xy;
8694         },
8695
8696         /**
8697          * Aligns this element with another element relative to the specified anchor points. If the other element is the
8698          * document it aligns it to the viewport.
8699          * The position parameter is optional, and can be specified in any one of the following formats:
8700          * <ul>
8701          *   <li><b>Blank</b>: Defaults to aligning the element's top-left corner to the target's bottom-left corner ("tl-bl").</li>
8702          *   <li><b>One anchor (deprecated)</b>: The passed anchor position is used as the target element's anchor point.
8703          *       The element being aligned will position its top-left corner (tl) to that point.  <i>This method has been
8704          *       deprecated in favor of the newer two anchor syntax below</i>.</li>
8705          *   <li><b>Two anchors</b>: If two values from the table below are passed separated by a dash, the first value is used as the
8706          *       element's anchor point, and the second value is used as the target's anchor point.</li>
8707          * </ul>
8708          * In addition to the anchor points, the position parameter also supports the "?" character.  If "?" is passed at the end of
8709          * the position string, the element will attempt to align as specified, but the position will be adjusted to constrain to
8710          * the viewport if necessary.  Note that the element being aligned might be swapped to align to a different position than
8711          * that specified in order to enforce the viewport constraints.
8712          * Following are all of the supported anchor positions:
8713     <pre>
8714     Value  Description
8715     -----  -----------------------------
8716     tl     The top left corner (default)
8717     t      The center of the top edge
8718     tr     The top right corner
8719     l      The center of the left edge
8720     c      In the center of the element
8721     r      The center of the right edge
8722     bl     The bottom left corner
8723     b      The center of the bottom edge
8724     br     The bottom right corner
8725     </pre>
8726     Example Usage:
8727     <pre><code>
8728     // align el to other-el using the default positioning ("tl-bl", non-constrained)
8729     el.alignTo("other-el");
8730
8731     // align the top left corner of el with the top right corner of other-el (constrained to viewport)
8732     el.alignTo("other-el", "tr?");
8733
8734     // align the bottom right corner of el with the center left edge of other-el
8735     el.alignTo("other-el", "br-l?");
8736
8737     // align the center of el with the bottom left corner of other-el and
8738     // adjust the x position by -6 pixels (and the y position by 0)
8739     el.alignTo("other-el", "c-bl", [-6, 0]);
8740     </code></pre>
8741          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8742          * @param {String} position The position to align to.
8743          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8744          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8745          * @return {Roo.Element} this
8746          */
8747         alignTo : function(element, position, offsets, animate){
8748             var xy = this.getAlignToXY(element, position, offsets);
8749             this.setXY(xy, this.preanim(arguments, 3));
8750             return this;
8751         },
8752
8753         /**
8754          * Anchors an element to another element and realigns it when the window is resized.
8755          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8756          * @param {String} position The position to align to.
8757          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8758          * @param {Boolean/Object} animate (optional) True for the default animation or a standard Element animation config object
8759          * @param {Boolean/Number} monitorScroll (optional) True to monitor body scroll and reposition. If this parameter
8760          * is a number, it is used as the buffer delay (defaults to 50ms).
8761          * @param {Function} callback The function to call after the animation finishes
8762          * @return {Roo.Element} this
8763          */
8764         anchorTo : function(el, alignment, offsets, animate, monitorScroll, callback){
8765             var action = function(){
8766                 this.alignTo(el, alignment, offsets, animate);
8767                 Roo.callback(callback, this);
8768             };
8769             Roo.EventManager.onWindowResize(action, this);
8770             var tm = typeof monitorScroll;
8771             if(tm != 'undefined'){
8772                 Roo.EventManager.on(window, 'scroll', action, this,
8773                     {buffer: tm == 'number' ? monitorScroll : 50});
8774             }
8775             action.call(this); // align immediately
8776             return this;
8777         },
8778         /**
8779          * Clears any opacity settings from this element. Required in some cases for IE.
8780          * @return {Roo.Element} this
8781          */
8782         clearOpacity : function(){
8783             if (window.ActiveXObject) {
8784                 if(typeof this.dom.style.filter == 'string' && (/alpha/i).test(this.dom.style.filter)){
8785                     this.dom.style.filter = "";
8786                 }
8787             } else {
8788                 this.dom.style.opacity = "";
8789                 this.dom.style["-moz-opacity"] = "";
8790                 this.dom.style["-khtml-opacity"] = "";
8791             }
8792             return this;
8793         },
8794
8795         /**
8796          * Hide this element - Uses display mode to determine whether to use "display" or "visibility". See {@link #setVisible}.
8797          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8798          * @return {Roo.Element} this
8799          */
8800         hide : function(animate){
8801             this.setVisible(false, this.preanim(arguments, 0));
8802             return this;
8803         },
8804
8805         /**
8806         * Show this element - Uses display mode to determine whether to use "display" or "visibility". See {@link #setVisible}.
8807         * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8808          * @return {Roo.Element} this
8809          */
8810         show : function(animate){
8811             this.setVisible(true, this.preanim(arguments, 0));
8812             return this;
8813         },
8814
8815         /**
8816          * @private Test if size has a unit, otherwise appends the default
8817          */
8818         addUnits : function(size){
8819             return Roo.Element.addUnits(size, this.defaultUnit);
8820         },
8821
8822         /**
8823          * Temporarily enables offsets (width,height,x,y) for an element with display:none, use endMeasure() when done.
8824          * @return {Roo.Element} this
8825          */
8826         beginMeasure : function(){
8827             var el = this.dom;
8828             if(el.offsetWidth || el.offsetHeight){
8829                 return this; // offsets work already
8830             }
8831             var changed = [];
8832             var p = this.dom, b = document.body; // start with this element
8833             while((!el.offsetWidth && !el.offsetHeight) && p && p.tagName && p != b){
8834                 var pe = Roo.get(p);
8835                 if(pe.getStyle('display') == 'none'){
8836                     changed.push({el: p, visibility: pe.getStyle("visibility")});
8837                     p.style.visibility = "hidden";
8838                     p.style.display = "block";
8839                 }
8840                 p = p.parentNode;
8841             }
8842             this._measureChanged = changed;
8843             return this;
8844
8845         },
8846
8847         /**
8848          * Restores displays to before beginMeasure was called
8849          * @return {Roo.Element} this
8850          */
8851         endMeasure : function(){
8852             var changed = this._measureChanged;
8853             if(changed){
8854                 for(var i = 0, len = changed.length; i < len; i++) {
8855                     var r = changed[i];
8856                     r.el.style.visibility = r.visibility;
8857                     r.el.style.display = "none";
8858                 }
8859                 this._measureChanged = null;
8860             }
8861             return this;
8862         },
8863
8864         /**
8865         * Update the innerHTML of this element, optionally searching for and processing scripts
8866         * @param {String} html The new HTML
8867         * @param {Boolean} loadScripts (optional) true to look for and process scripts
8868         * @param {Function} callback For async script loading you can be noticed when the update completes
8869         * @return {Roo.Element} this
8870          */
8871         update : function(html, loadScripts, callback){
8872             if(typeof html == "undefined"){
8873                 html = "";
8874             }
8875             if(loadScripts !== true){
8876                 this.dom.innerHTML = html;
8877                 if(typeof callback == "function"){
8878                     callback();
8879                 }
8880                 return this;
8881             }
8882             var id = Roo.id();
8883             var dom = this.dom;
8884
8885             html += '<span id="' + id + '"></span>';
8886
8887             E.onAvailable(id, function(){
8888                 var hd = document.getElementsByTagName("head")[0];
8889                 var re = /(?:<script([^>]*)?>)((\n|\r|.)*?)(?:<\/script>)/ig;
8890                 var srcRe = /\ssrc=([\'\"])(.*?)\1/i;
8891                 var typeRe = /\stype=([\'\"])(.*?)\1/i;
8892
8893                 var match;
8894                 while(match = re.exec(html)){
8895                     var attrs = match[1];
8896                     var srcMatch = attrs ? attrs.match(srcRe) : false;
8897                     if(srcMatch && srcMatch[2]){
8898                        var s = document.createElement("script");
8899                        s.src = srcMatch[2];
8900                        var typeMatch = attrs.match(typeRe);
8901                        if(typeMatch && typeMatch[2]){
8902                            s.type = typeMatch[2];
8903                        }
8904                        hd.appendChild(s);
8905                     }else if(match[2] && match[2].length > 0){
8906                         if(window.execScript) {
8907                            window.execScript(match[2]);
8908                         } else {
8909                             /**
8910                              * eval:var:id
8911                              * eval:var:dom
8912                              * eval:var:html
8913                              * 
8914                              */
8915                            window.eval(match[2]);
8916                         }
8917                     }
8918                 }
8919                 var el = document.getElementById(id);
8920                 if(el){el.parentNode.removeChild(el);}
8921                 if(typeof callback == "function"){
8922                     callback();
8923                 }
8924             });
8925             dom.innerHTML = html.replace(/(?:<script.*?>)((\n|\r|.)*?)(?:<\/script>)/ig, "");
8926             return this;
8927         },
8928
8929         /**
8930          * Direct access to the UpdateManager update() method (takes the same parameters).
8931          * @param {String/Function} url The url for this request or a function to call to get the url
8932          * @param {String/Object} params (optional) The parameters to pass as either a url encoded string "param1=1&amp;param2=2" or an object {param1: 1, param2: 2}
8933          * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
8934          * @param {Boolean} discardUrl (optional) By default when you execute an update the defaultUrl is changed to the last used url. If true, it will not store the url.
8935          * @return {Roo.Element} this
8936          */
8937         load : function(){
8938             var um = this.getUpdateManager();
8939             um.update.apply(um, arguments);
8940             return this;
8941         },
8942
8943         /**
8944         * Gets this element's UpdateManager
8945         * @return {Roo.UpdateManager} The UpdateManager
8946         */
8947         getUpdateManager : function(){
8948             if(!this.updateManager){
8949                 this.updateManager = new Roo.UpdateManager(this);
8950             }
8951             return this.updateManager;
8952         },
8953
8954         /**
8955          * Disables text selection for this element (normalized across browsers)
8956          * @return {Roo.Element} this
8957          */
8958         unselectable : function(){
8959             this.dom.unselectable = "on";
8960             this.swallowEvent("selectstart", true);
8961             this.applyStyles("-moz-user-select:none;-khtml-user-select:none;");
8962             this.addClass("x-unselectable");
8963             return this;
8964         },
8965
8966         /**
8967         * Calculates the x, y to center this element on the screen
8968         * @return {Array} The x, y values [x, y]
8969         */
8970         getCenterXY : function(){
8971             return this.getAlignToXY(document, 'c-c');
8972         },
8973
8974         /**
8975         * Centers the Element in either the viewport, or another Element.
8976         * @param {String/HTMLElement/Roo.Element} centerIn (optional) The element in which to center the element.
8977         */
8978         center : function(centerIn){
8979             this.alignTo(centerIn || document, 'c-c');
8980             return this;
8981         },
8982
8983         /**
8984          * Tests various css rules/browsers to determine if this element uses a border box
8985          * @return {Boolean}
8986          */
8987         isBorderBox : function(){
8988             return noBoxAdjust[this.dom.tagName.toLowerCase()] || Roo.isBorderBox;
8989         },
8990
8991         /**
8992          * Return a box {x, y, width, height} that can be used to set another elements
8993          * size/location to match this element.
8994          * @param {Boolean} contentBox (optional) If true a box for the content of the element is returned.
8995          * @param {Boolean} local (optional) If true the element's left and top are returned instead of page x/y.
8996          * @return {Object} box An object in the format {x, y, width, height}
8997          */
8998         getBox : function(contentBox, local){
8999             var xy;
9000             if(!local){
9001                 xy = this.getXY();
9002             }else{
9003                 var left = parseInt(this.getStyle("left"), 10) || 0;
9004                 var top = parseInt(this.getStyle("top"), 10) || 0;
9005                 xy = [left, top];
9006             }
9007             var el = this.dom, w = el.offsetWidth, h = el.offsetHeight, bx;
9008             if(!contentBox){
9009                 bx = {x: xy[0], y: xy[1], 0: xy[0], 1: xy[1], width: w, height: h};
9010             }else{
9011                 var l = this.getBorderWidth("l")+this.getPadding("l");
9012                 var r = this.getBorderWidth("r")+this.getPadding("r");
9013                 var t = this.getBorderWidth("t")+this.getPadding("t");
9014                 var b = this.getBorderWidth("b")+this.getPadding("b");
9015                 bx = {x: xy[0]+l, y: xy[1]+t, 0: xy[0]+l, 1: xy[1]+t, width: w-(l+r), height: h-(t+b)};
9016             }
9017             bx.right = bx.x + bx.width;
9018             bx.bottom = bx.y + bx.height;
9019             return bx;
9020         },
9021
9022         /**
9023          * Returns the sum width of the padding and borders for the passed "sides". See getBorderWidth()
9024          for more information about the sides.
9025          * @param {String} sides
9026          * @return {Number}
9027          */
9028         getFrameWidth : function(sides, onlyContentBox){
9029             return onlyContentBox && Roo.isBorderBox ? 0 : (this.getPadding(sides) + this.getBorderWidth(sides));
9030         },
9031
9032         /**
9033          * Sets the element's box. Use getBox() on another element to get a box obj. If animate is true then width, height, x and y will be animated concurrently.
9034          * @param {Object} box The box to fill {x, y, width, height}
9035          * @param {Boolean} adjust (optional) Whether to adjust for box-model issues automatically
9036          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9037          * @return {Roo.Element} this
9038          */
9039         setBox : function(box, adjust, animate){
9040             var w = box.width, h = box.height;
9041             if((adjust && !this.autoBoxAdjust) && !this.isBorderBox()){
9042                w -= (this.getBorderWidth("lr") + this.getPadding("lr"));
9043                h -= (this.getBorderWidth("tb") + this.getPadding("tb"));
9044             }
9045             this.setBounds(box.x, box.y, w, h, this.preanim(arguments, 2));
9046             return this;
9047         },
9048
9049         /**
9050          * Forces the browser to repaint this element
9051          * @return {Roo.Element} this
9052          */
9053          repaint : function(){
9054             var dom = this.dom;
9055             this.addClass("x-repaint");
9056             setTimeout(function(){
9057                 Roo.get(dom).removeClass("x-repaint");
9058             }, 1);
9059             return this;
9060         },
9061
9062         /**
9063          * Returns an object with properties top, left, right and bottom representing the margins of this element unless sides is passed,
9064          * then it returns the calculated width of the sides (see getPadding)
9065          * @param {String} sides (optional) Any combination of l, r, t, b to get the sum of those sides
9066          * @return {Object/Number}
9067          */
9068         getMargins : function(side){
9069             if(!side){
9070                 return {
9071                     top: parseInt(this.getStyle("margin-top"), 10) || 0,
9072                     left: parseInt(this.getStyle("margin-left"), 10) || 0,
9073                     bottom: parseInt(this.getStyle("margin-bottom"), 10) || 0,
9074                     right: parseInt(this.getStyle("margin-right"), 10) || 0
9075                 };
9076             }else{
9077                 return this.addStyles(side, El.margins);
9078              }
9079         },
9080
9081         // private
9082         addStyles : function(sides, styles){
9083             var val = 0, v, w;
9084             for(var i = 0, len = sides.length; i < len; i++){
9085                 v = this.getStyle(styles[sides.charAt(i)]);
9086                 if(v){
9087                      w = parseInt(v, 10);
9088                      if(w){ val += w; }
9089                 }
9090             }
9091             return val;
9092         },
9093
9094         /**
9095          * Creates a proxy element of this element
9096          * @param {String/Object} config The class name of the proxy element or a DomHelper config object
9097          * @param {String/HTMLElement} renderTo (optional) The element or element id to render the proxy to (defaults to document.body)
9098          * @param {Boolean} matchBox (optional) True to align and size the proxy to this element now (defaults to false)
9099          * @return {Roo.Element} The new proxy element
9100          */
9101         createProxy : function(config, renderTo, matchBox){
9102             if(renderTo){
9103                 renderTo = Roo.getDom(renderTo);
9104             }else{
9105                 renderTo = document.body;
9106             }
9107             config = typeof config == "object" ?
9108                 config : {tag : "div", cls: config};
9109             var proxy = Roo.DomHelper.append(renderTo, config, true);
9110             if(matchBox){
9111                proxy.setBox(this.getBox());
9112             }
9113             return proxy;
9114         },
9115
9116         /**
9117          * Puts a mask over this element to disable user interaction. Requires core.css.
9118          * This method can only be applied to elements which accept child nodes.
9119          * @param {String} msg (optional) A message to display in the mask
9120          * @param {String} msgCls (optional) A css class to apply to the msg element
9121          * @return {Element} The mask  element
9122          */
9123         mask : function(msg, msgCls)
9124         {
9125             if(this.getStyle("position") == "static" && this.dom.tagName !== 'BODY'){
9126                 this.setStyle("position", "relative");
9127             }
9128             if(!this._mask){
9129                 this._mask = Roo.DomHelper.append(this.dom, {cls:"roo-el-mask"}, true);
9130             }
9131             
9132             this.addClass("x-masked");
9133             this._mask.setDisplayed(true);
9134             
9135             // we wander
9136             var z = 0;
9137             var dom = this.dom;
9138             while (dom && dom.style) {
9139                 if (!isNaN(parseInt(dom.style.zIndex))) {
9140                     z = Math.max(z, parseInt(dom.style.zIndex));
9141                 }
9142                 dom = dom.parentNode;
9143             }
9144             // if we are masking the body - then it hides everything..
9145             if (this.dom == document.body) {
9146                 z = 1000000;
9147                 this._mask.setWidth(Roo.lib.Dom.getDocumentWidth());
9148                 this._mask.setHeight(Roo.lib.Dom.getDocumentHeight());
9149             }
9150            
9151             if(typeof msg == 'string'){
9152                 if(!this._maskMsg){
9153                     this._maskMsg = Roo.DomHelper.append(this.dom, {
9154                         cls: "roo-el-mask-msg", 
9155                         cn: [
9156                             {
9157                                 tag: 'i',
9158                                 cls: 'fa fa-spinner fa-spin'
9159                             },
9160                             {
9161                                 tag: 'div'
9162                             }   
9163                         ]
9164                     }, true);
9165                 }
9166                 var mm = this._maskMsg;
9167                 mm.dom.className = msgCls ? "roo-el-mask-msg " + msgCls : "roo-el-mask-msg";
9168                 if (mm.dom.lastChild) { // weird IE issue?
9169                     mm.dom.lastChild.innerHTML = msg;
9170                 }
9171                 mm.setDisplayed(true);
9172                 mm.center(this);
9173                 mm.setStyle('z-index', z + 102);
9174             }
9175             if(Roo.isIE && !(Roo.isIE7 && Roo.isStrict) && this.getStyle('height') == 'auto'){ // ie will not expand full height automatically
9176                 this._mask.setHeight(this.getHeight());
9177             }
9178             this._mask.setStyle('z-index', z + 100);
9179             
9180             return this._mask;
9181         },
9182
9183         /**
9184          * Removes a previously applied mask. If removeEl is true the mask overlay is destroyed, otherwise
9185          * it is cached for reuse.
9186          */
9187         unmask : function(removeEl){
9188             if(this._mask){
9189                 if(removeEl === true){
9190                     this._mask.remove();
9191                     delete this._mask;
9192                     if(this._maskMsg){
9193                         this._maskMsg.remove();
9194                         delete this._maskMsg;
9195                     }
9196                 }else{
9197                     this._mask.setDisplayed(false);
9198                     if(this._maskMsg){
9199                         this._maskMsg.setDisplayed(false);
9200                     }
9201                 }
9202             }
9203             this.removeClass("x-masked");
9204         },
9205
9206         /**
9207          * Returns true if this element is masked
9208          * @return {Boolean}
9209          */
9210         isMasked : function(){
9211             return this._mask && this._mask.isVisible();
9212         },
9213
9214         /**
9215          * Creates an iframe shim for this element to keep selects and other windowed objects from
9216          * showing through.
9217          * @return {Roo.Element} The new shim element
9218          */
9219         createShim : function(){
9220             var el = document.createElement('iframe');
9221             el.frameBorder = 'no';
9222             el.className = 'roo-shim';
9223             if(Roo.isIE && Roo.isSecure){
9224                 el.src = Roo.SSL_SECURE_URL;
9225             }
9226             var shim = Roo.get(this.dom.parentNode.insertBefore(el, this.dom));
9227             shim.autoBoxAdjust = false;
9228             return shim;
9229         },
9230
9231         /**
9232          * Removes this element from the DOM and deletes it from the cache
9233          */
9234         remove : function(){
9235             if(this.dom.parentNode){
9236                 this.dom.parentNode.removeChild(this.dom);
9237             }
9238             delete El.cache[this.dom.id];
9239         },
9240
9241         /**
9242          * Sets up event handlers to add and remove a css class when the mouse is over this element
9243          * @param {String} className
9244          * @param {Boolean} preventFlicker (optional) If set to true, it prevents flickering by filtering
9245          * mouseout events for children elements
9246          * @return {Roo.Element} this
9247          */
9248         addClassOnOver : function(className, preventFlicker){
9249             this.on("mouseover", function(){
9250                 Roo.fly(this, '_internal').addClass(className);
9251             }, this.dom);
9252             var removeFn = function(e){
9253                 if(preventFlicker !== true || !e.within(this, true)){
9254                     Roo.fly(this, '_internal').removeClass(className);
9255                 }
9256             };
9257             this.on("mouseout", removeFn, this.dom);
9258             return this;
9259         },
9260
9261         /**
9262          * Sets up event handlers to add and remove a css class when this element has the focus
9263          * @param {String} className
9264          * @return {Roo.Element} this
9265          */
9266         addClassOnFocus : function(className){
9267             this.on("focus", function(){
9268                 Roo.fly(this, '_internal').addClass(className);
9269             }, this.dom);
9270             this.on("blur", function(){
9271                 Roo.fly(this, '_internal').removeClass(className);
9272             }, this.dom);
9273             return this;
9274         },
9275         /**
9276          * Sets up event handlers to add and remove a css class when the mouse is down and then up on this element (a click effect)
9277          * @param {String} className
9278          * @return {Roo.Element} this
9279          */
9280         addClassOnClick : function(className){
9281             var dom = this.dom;
9282             this.on("mousedown", function(){
9283                 Roo.fly(dom, '_internal').addClass(className);
9284                 var d = Roo.get(document);
9285                 var fn = function(){
9286                     Roo.fly(dom, '_internal').removeClass(className);
9287                     d.removeListener("mouseup", fn);
9288                 };
9289                 d.on("mouseup", fn);
9290             });
9291             return this;
9292         },
9293
9294         /**
9295          * Stops the specified event from bubbling and optionally prevents the default action
9296          * @param {String} eventName
9297          * @param {Boolean} preventDefault (optional) true to prevent the default action too
9298          * @return {Roo.Element} this
9299          */
9300         swallowEvent : function(eventName, preventDefault){
9301             var fn = function(e){
9302                 e.stopPropagation();
9303                 if(preventDefault){
9304                     e.preventDefault();
9305                 }
9306             };
9307             if(eventName instanceof Array){
9308                 for(var i = 0, len = eventName.length; i < len; i++){
9309                      this.on(eventName[i], fn);
9310                 }
9311                 return this;
9312             }
9313             this.on(eventName, fn);
9314             return this;
9315         },
9316
9317         /**
9318          * @private
9319          */
9320       fitToParentDelegate : Roo.emptyFn, // keep a reference to the fitToParent delegate
9321
9322         /**
9323          * Sizes this element to its parent element's dimensions performing
9324          * neccessary box adjustments.
9325          * @param {Boolean} monitorResize (optional) If true maintains the fit when the browser window is resized.
9326          * @param {String/HTMLElment/Element} targetParent (optional) The target parent, default to the parentNode.
9327          * @return {Roo.Element} this
9328          */
9329         fitToParent : function(monitorResize, targetParent) {
9330           Roo.EventManager.removeResizeListener(this.fitToParentDelegate); // always remove previous fitToParent delegate from onWindowResize
9331           this.fitToParentDelegate = Roo.emptyFn; // remove reference to previous delegate
9332           if (monitorResize === true && !this.dom.parentNode) { // check if this Element still exists
9333             return;
9334           }
9335           var p = Roo.get(targetParent || this.dom.parentNode);
9336           this.setSize(p.getComputedWidth() - p.getFrameWidth('lr'), p.getComputedHeight() - p.getFrameWidth('tb'));
9337           if (monitorResize === true) {
9338             this.fitToParentDelegate = this.fitToParent.createDelegate(this, [true, targetParent]);
9339             Roo.EventManager.onWindowResize(this.fitToParentDelegate);
9340           }
9341           return this;
9342         },
9343
9344         /**
9345          * Gets the next sibling, skipping text nodes
9346          * @return {HTMLElement} The next sibling or null
9347          */
9348         getNextSibling : function(){
9349             var n = this.dom.nextSibling;
9350             while(n && n.nodeType != 1){
9351                 n = n.nextSibling;
9352             }
9353             return n;
9354         },
9355
9356         /**
9357          * Gets the previous sibling, skipping text nodes
9358          * @return {HTMLElement} The previous sibling or null
9359          */
9360         getPrevSibling : function(){
9361             var n = this.dom.previousSibling;
9362             while(n && n.nodeType != 1){
9363                 n = n.previousSibling;
9364             }
9365             return n;
9366         },
9367
9368
9369         /**
9370          * Appends the passed element(s) to this element
9371          * @param {String/HTMLElement/Array/Element/CompositeElement} el
9372          * @return {Roo.Element} this
9373          */
9374         appendChild: function(el){
9375             el = Roo.get(el);
9376             el.appendTo(this);
9377             return this;
9378         },
9379
9380         /**
9381          * Creates the passed DomHelper config and appends it to this element or optionally inserts it before the passed child element.
9382          * @param {Object} config DomHelper element config object.  If no tag is specified (e.g., {tag:'input'}) then a div will be
9383          * automatically generated with the specified attributes.
9384          * @param {HTMLElement} insertBefore (optional) a child element of this element
9385          * @param {Boolean} returnDom (optional) true to return the dom node instead of creating an Element
9386          * @return {Roo.Element} The new child element
9387          */
9388         createChild: function(config, insertBefore, returnDom){
9389             config = config || {tag:'div'};
9390             if(insertBefore){
9391                 return Roo.DomHelper.insertBefore(insertBefore, config, returnDom !== true);
9392             }
9393             return Roo.DomHelper[!this.dom.firstChild ? 'overwrite' : 'append'](this.dom, config,  returnDom !== true);
9394         },
9395
9396         /**
9397          * Appends this element to the passed element
9398          * @param {String/HTMLElement/Element} el The new parent element
9399          * @return {Roo.Element} this
9400          */
9401         appendTo: function(el){
9402             el = Roo.getDom(el);
9403             el.appendChild(this.dom);
9404             return this;
9405         },
9406
9407         /**
9408          * Inserts this element before the passed element in the DOM
9409          * @param {String/HTMLElement/Element} el The element to insert before
9410          * @return {Roo.Element} this
9411          */
9412         insertBefore: function(el){
9413             el = Roo.getDom(el);
9414             el.parentNode.insertBefore(this.dom, el);
9415             return this;
9416         },
9417
9418         /**
9419          * Inserts this element after the passed element in the DOM
9420          * @param {String/HTMLElement/Element} el The element to insert after
9421          * @return {Roo.Element} this
9422          */
9423         insertAfter: function(el){
9424             el = Roo.getDom(el);
9425             el.parentNode.insertBefore(this.dom, el.nextSibling);
9426             return this;
9427         },
9428
9429         /**
9430          * Inserts (or creates) an element (or DomHelper config) as the first child of the this element
9431          * @param {String/HTMLElement/Element/Object} el The id or element to insert or a DomHelper config to create and insert
9432          * @return {Roo.Element} The new child
9433          */
9434         insertFirst: function(el, returnDom){
9435             el = el || {};
9436             if(typeof el == 'object' && !el.nodeType){ // dh config
9437                 return this.createChild(el, this.dom.firstChild, returnDom);
9438             }else{
9439                 el = Roo.getDom(el);
9440                 this.dom.insertBefore(el, this.dom.firstChild);
9441                 return !returnDom ? Roo.get(el) : el;
9442             }
9443         },
9444
9445         /**
9446          * Inserts (or creates) the passed element (or DomHelper config) as a sibling of this element
9447          * @param {String/HTMLElement/Element/Object} el The id or element to insert or a DomHelper config to create and insert
9448          * @param {String} where (optional) 'before' or 'after' defaults to before
9449          * @param {Boolean} returnDom (optional) True to return the raw DOM element instead of Roo.Element
9450          * @return {Roo.Element} the inserted Element
9451          */
9452         insertSibling: function(el, where, returnDom){
9453             where = where ? where.toLowerCase() : 'before';
9454             el = el || {};
9455             var rt, refNode = where == 'before' ? this.dom : this.dom.nextSibling;
9456
9457             if(typeof el == 'object' && !el.nodeType){ // dh config
9458                 if(where == 'after' && !this.dom.nextSibling){
9459                     rt = Roo.DomHelper.append(this.dom.parentNode, el, !returnDom);
9460                 }else{
9461                     rt = Roo.DomHelper[where == 'after' ? 'insertAfter' : 'insertBefore'](this.dom, el, !returnDom);
9462                 }
9463
9464             }else{
9465                 rt = this.dom.parentNode.insertBefore(Roo.getDom(el),
9466                             where == 'before' ? this.dom : this.dom.nextSibling);
9467                 if(!returnDom){
9468                     rt = Roo.get(rt);
9469                 }
9470             }
9471             return rt;
9472         },
9473
9474         /**
9475          * Creates and wraps this element with another element
9476          * @param {Object} config (optional) DomHelper element config object for the wrapper element or null for an empty div
9477          * @param {Boolean} returnDom (optional) True to return the raw DOM element instead of Roo.Element
9478          * @return {HTMLElement/Element} The newly created wrapper element
9479          */
9480         wrap: function(config, returnDom){
9481             if(!config){
9482                 config = {tag: "div"};
9483             }
9484             var newEl = Roo.DomHelper.insertBefore(this.dom, config, !returnDom);
9485             newEl.dom ? newEl.dom.appendChild(this.dom) : newEl.appendChild(this.dom);
9486             return newEl;
9487         },
9488
9489         /**
9490          * Replaces the passed element with this element
9491          * @param {String/HTMLElement/Element} el The element to replace
9492          * @return {Roo.Element} this
9493          */
9494         replace: function(el){
9495             el = Roo.get(el);
9496             this.insertBefore(el);
9497             el.remove();
9498             return this;
9499         },
9500
9501         /**
9502          * Inserts an html fragment into this element
9503          * @param {String} where Where to insert the html in relation to the this element - beforeBegin, afterBegin, beforeEnd, afterEnd.
9504          * @param {String} html The HTML fragment
9505          * @param {Boolean} returnEl True to return an Roo.Element
9506          * @return {HTMLElement/Roo.Element} The inserted node (or nearest related if more than 1 inserted)
9507          */
9508         insertHtml : function(where, html, returnEl){
9509             var el = Roo.DomHelper.insertHtml(where, this.dom, html);
9510             return returnEl ? Roo.get(el) : el;
9511         },
9512
9513         /**
9514          * Sets the passed attributes as attributes of this element (a style attribute can be a string, object or function)
9515          * @param {Object} o The object with the attributes
9516          * @param {Boolean} useSet (optional) false to override the default setAttribute to use expandos.
9517          * @return {Roo.Element} this
9518          */
9519         set : function(o, useSet){
9520             var el = this.dom;
9521             useSet = typeof useSet == 'undefined' ? (el.setAttribute ? true : false) : useSet;
9522             for(var attr in o){
9523                 if(attr == "style" || typeof o[attr] == "function")  { continue; }
9524                 if(attr=="cls"){
9525                     el.className = o["cls"];
9526                 }else{
9527                     if(useSet) {
9528                         el.setAttribute(attr, o[attr]);
9529                     } else {
9530                         el[attr] = o[attr];
9531                     }
9532                 }
9533             }
9534             if(o.style){
9535                 Roo.DomHelper.applyStyles(el, o.style);
9536             }
9537             return this;
9538         },
9539
9540         /**
9541          * Convenience method for constructing a KeyMap
9542          * @param {Number/Array/Object/String} key Either a string with the keys to listen for, the numeric key code, array of key codes or an object with the following options:
9543          *                                  {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
9544          * @param {Function} fn The function to call
9545          * @param {Object} scope (optional) The scope of the function
9546          * @return {Roo.KeyMap} The KeyMap created
9547          */
9548         addKeyListener : function(key, fn, scope){
9549             var config;
9550             if(typeof key != "object" || key instanceof Array){
9551                 config = {
9552                     key: key,
9553                     fn: fn,
9554                     scope: scope
9555                 };
9556             }else{
9557                 config = {
9558                     key : key.key,
9559                     shift : key.shift,
9560                     ctrl : key.ctrl,
9561                     alt : key.alt,
9562                     fn: fn,
9563                     scope: scope
9564                 };
9565             }
9566             return new Roo.KeyMap(this, config);
9567         },
9568
9569         /**
9570          * Creates a KeyMap for this element
9571          * @param {Object} config The KeyMap config. See {@link Roo.KeyMap} for more details
9572          * @return {Roo.KeyMap} The KeyMap created
9573          */
9574         addKeyMap : function(config){
9575             return new Roo.KeyMap(this, config);
9576         },
9577
9578         /**
9579          * Returns true if this element is scrollable.
9580          * @return {Boolean}
9581          */
9582          isScrollable : function(){
9583             var dom = this.dom;
9584             return dom.scrollHeight > dom.clientHeight || dom.scrollWidth > dom.clientWidth;
9585         },
9586
9587         /**
9588          * Scrolls this element the specified scroll point. It does NOT do bounds checking so if you scroll to a weird value it will try to do it. For auto bounds checking, use scroll().
9589          * @param {String} side Either "left" for scrollLeft values or "top" for scrollTop values.
9590          * @param {Number} value The new scroll value
9591          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9592          * @return {Element} this
9593          */
9594
9595         scrollTo : function(side, value, animate){
9596             var prop = side.toLowerCase() == "left" ? "scrollLeft" : "scrollTop";
9597             if(!animate || !A){
9598                 this.dom[prop] = value;
9599             }else{
9600                 var to = prop == "scrollLeft" ? [value, this.dom.scrollTop] : [this.dom.scrollLeft, value];
9601                 this.anim({scroll: {"to": to}}, this.preanim(arguments, 2), 'scroll');
9602             }
9603             return this;
9604         },
9605
9606         /**
9607          * Scrolls this element the specified direction. Does bounds checking to make sure the scroll is
9608          * within this element's scrollable range.
9609          * @param {String} direction Possible values are: "l","left" - "r","right" - "t","top","up" - "b","bottom","down".
9610          * @param {Number} distance How far to scroll the element in pixels
9611          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9612          * @return {Boolean} Returns true if a scroll was triggered or false if the element
9613          * was scrolled as far as it could go.
9614          */
9615          scroll : function(direction, distance, animate){
9616              if(!this.isScrollable()){
9617                  return;
9618              }
9619              var el = this.dom;
9620              var l = el.scrollLeft, t = el.scrollTop;
9621              var w = el.scrollWidth, h = el.scrollHeight;
9622              var cw = el.clientWidth, ch = el.clientHeight;
9623              direction = direction.toLowerCase();
9624              var scrolled = false;
9625              var a = this.preanim(arguments, 2);
9626              switch(direction){
9627                  case "l":
9628                  case "left":
9629                      if(w - l > cw){
9630                          var v = Math.min(l + distance, w-cw);
9631                          this.scrollTo("left", v, a);
9632                          scrolled = true;
9633                      }
9634                      break;
9635                 case "r":
9636                 case "right":
9637                      if(l > 0){
9638                          var v = Math.max(l - distance, 0);
9639                          this.scrollTo("left", v, a);
9640                          scrolled = true;
9641                      }
9642                      break;
9643                 case "t":
9644                 case "top":
9645                 case "up":
9646                      if(t > 0){
9647                          var v = Math.max(t - distance, 0);
9648                          this.scrollTo("top", v, a);
9649                          scrolled = true;
9650                      }
9651                      break;
9652                 case "b":
9653                 case "bottom":
9654                 case "down":
9655                      if(h - t > ch){
9656                          var v = Math.min(t + distance, h-ch);
9657                          this.scrollTo("top", v, a);
9658                          scrolled = true;
9659                      }
9660                      break;
9661              }
9662              return scrolled;
9663         },
9664
9665         /**
9666          * Translates the passed page coordinates into left/top css values for this element
9667          * @param {Number/Array} x The page x or an array containing [x, y]
9668          * @param {Number} y The page y
9669          * @return {Object} An object with left and top properties. e.g. {left: (value), top: (value)}
9670          */
9671         translatePoints : function(x, y){
9672             if(typeof x == 'object' || x instanceof Array){
9673                 y = x[1]; x = x[0];
9674             }
9675             var p = this.getStyle('position');
9676             var o = this.getXY();
9677
9678             var l = parseInt(this.getStyle('left'), 10);
9679             var t = parseInt(this.getStyle('top'), 10);
9680
9681             if(isNaN(l)){
9682                 l = (p == "relative") ? 0 : this.dom.offsetLeft;
9683             }
9684             if(isNaN(t)){
9685                 t = (p == "relative") ? 0 : this.dom.offsetTop;
9686             }
9687
9688             return {left: (x - o[0] + l), top: (y - o[1] + t)};
9689         },
9690
9691         /**
9692          * Returns the current scroll position of the element.
9693          * @return {Object} An object containing the scroll position in the format {left: (scrollLeft), top: (scrollTop)}
9694          */
9695         getScroll : function(){
9696             var d = this.dom, doc = document;
9697             if(d == doc || d == doc.body){
9698                 var l = window.pageXOffset || doc.documentElement.scrollLeft || doc.body.scrollLeft || 0;
9699                 var t = window.pageYOffset || doc.documentElement.scrollTop || doc.body.scrollTop || 0;
9700                 return {left: l, top: t};
9701             }else{
9702                 return {left: d.scrollLeft, top: d.scrollTop};
9703             }
9704         },
9705
9706         /**
9707          * Return the CSS color for the specified CSS attribute. rgb, 3 digit (like #fff) and valid values
9708          * are convert to standard 6 digit hex color.
9709          * @param {String} attr The css attribute
9710          * @param {String} defaultValue The default value to use when a valid color isn't found
9711          * @param {String} prefix (optional) defaults to #. Use an empty string when working with
9712          * YUI color anims.
9713          */
9714         getColor : function(attr, defaultValue, prefix){
9715             var v = this.getStyle(attr);
9716             if(!v || v == "transparent" || v == "inherit") {
9717                 return defaultValue;
9718             }
9719             var color = typeof prefix == "undefined" ? "#" : prefix;
9720             if(v.substr(0, 4) == "rgb("){
9721                 var rvs = v.slice(4, v.length -1).split(",");
9722                 for(var i = 0; i < 3; i++){
9723                     var h = parseInt(rvs[i]).toString(16);
9724                     if(h < 16){
9725                         h = "0" + h;
9726                     }
9727                     color += h;
9728                 }
9729             } else {
9730                 if(v.substr(0, 1) == "#"){
9731                     if(v.length == 4) {
9732                         for(var i = 1; i < 4; i++){
9733                             var c = v.charAt(i);
9734                             color +=  c + c;
9735                         }
9736                     }else if(v.length == 7){
9737                         color += v.substr(1);
9738                     }
9739                 }
9740             }
9741             return(color.length > 5 ? color.toLowerCase() : defaultValue);
9742         },
9743
9744         /**
9745          * Wraps the specified element with a special markup/CSS block that renders by default as a gray container with a
9746          * gradient background, rounded corners and a 4-way shadow.
9747          * @param {String} class (optional) A base CSS class to apply to the containing wrapper element (defaults to 'x-box').
9748          * Note that there are a number of CSS rules that are dependent on this name to make the overall effect work,
9749          * so if you supply an alternate base class, make sure you also supply all of the necessary rules.
9750          * @return {Roo.Element} this
9751          */
9752         boxWrap : function(cls){
9753             cls = cls || 'x-box';
9754             var el = Roo.get(this.insertHtml('beforeBegin', String.format('<div class="{0}">'+El.boxMarkup+'</div>', cls)));
9755             el.child('.'+cls+'-mc').dom.appendChild(this.dom);
9756             return el;
9757         },
9758
9759         /**
9760          * Returns the value of a namespaced attribute from the element's underlying DOM node.
9761          * @param {String} namespace The namespace in which to look for the attribute
9762          * @param {String} name The attribute name
9763          * @return {String} The attribute value
9764          */
9765         getAttributeNS : Roo.isIE ? function(ns, name){
9766             var d = this.dom;
9767             var type = typeof d[ns+":"+name];
9768             if(type != 'undefined' && type != 'unknown'){
9769                 return d[ns+":"+name];
9770             }
9771             return d[name];
9772         } : function(ns, name){
9773             var d = this.dom;
9774             return d.getAttributeNS(ns, name) || d.getAttribute(ns+":"+name) || d.getAttribute(name) || d[name];
9775         },
9776         
9777         
9778         /**
9779          * Sets or Returns the value the dom attribute value
9780          * @param {String|Object} name The attribute name (or object to set multiple attributes)
9781          * @param {String} value (optional) The value to set the attribute to
9782          * @return {String} The attribute value
9783          */
9784         attr : function(name){
9785             if (arguments.length > 1) {
9786                 this.dom.setAttribute(name, arguments[1]);
9787                 return arguments[1];
9788             }
9789             if (typeof(name) == 'object') {
9790                 for(var i in name) {
9791                     this.attr(i, name[i]);
9792                 }
9793                 return name;
9794             }
9795             
9796             
9797             if (!this.dom.hasAttribute(name)) {
9798                 return undefined;
9799             }
9800             return this.dom.getAttribute(name);
9801         }
9802         
9803         
9804         
9805     };
9806
9807     var ep = El.prototype;
9808
9809     /**
9810      * Appends an event handler (Shorthand for addListener)
9811      * @param {String}   eventName     The type of event to append
9812      * @param {Function} fn        The method the event invokes
9813      * @param {Object} scope       (optional) The scope (this object) of the fn
9814      * @param {Object}   options   (optional)An object with standard {@link Roo.EventManager#addListener} options
9815      * @method
9816      */
9817     ep.on = ep.addListener;
9818         // backwards compat
9819     ep.mon = ep.addListener;
9820
9821     /**
9822      * Removes an event handler from this element (shorthand for removeListener)
9823      * @param {String} eventName the type of event to remove
9824      * @param {Function} fn the method the event invokes
9825      * @return {Roo.Element} this
9826      * @method
9827      */
9828     ep.un = ep.removeListener;
9829
9830     /**
9831      * true to automatically adjust width and height settings for box-model issues (default to true)
9832      */
9833     ep.autoBoxAdjust = true;
9834
9835     // private
9836     El.unitPattern = /\d+(px|em|%|en|ex|pt|in|cm|mm|pc)$/i;
9837
9838     // private
9839     El.addUnits = function(v, defaultUnit){
9840         if(v === "" || v == "auto"){
9841             return v;
9842         }
9843         if(v === undefined){
9844             return '';
9845         }
9846         if(typeof v == "number" || !El.unitPattern.test(v)){
9847             return v + (defaultUnit || 'px');
9848         }
9849         return v;
9850     };
9851
9852     // special markup used throughout Roo when box wrapping elements
9853     El.boxMarkup = '<div class="{0}-tl"><div class="{0}-tr"><div class="{0}-tc"></div></div></div><div class="{0}-ml"><div class="{0}-mr"><div class="{0}-mc"></div></div></div><div class="{0}-bl"><div class="{0}-br"><div class="{0}-bc"></div></div></div>';
9854     /**
9855      * Visibility mode constant - Use visibility to hide element
9856      * @static
9857      * @type Number
9858      */
9859     El.VISIBILITY = 1;
9860     /**
9861      * Visibility mode constant - Use display to hide element
9862      * @static
9863      * @type Number
9864      */
9865     El.DISPLAY = 2;
9866
9867     El.borders = {l: "border-left-width", r: "border-right-width", t: "border-top-width", b: "border-bottom-width"};
9868     El.paddings = {l: "padding-left", r: "padding-right", t: "padding-top", b: "padding-bottom"};
9869     El.margins = {l: "margin-left", r: "margin-right", t: "margin-top", b: "margin-bottom"};
9870
9871
9872
9873     /**
9874      * @private
9875      */
9876     El.cache = {};
9877
9878     var docEl;
9879
9880     /**
9881      * Static method to retrieve Element objects. Uses simple caching to consistently return the same object.
9882      * Automatically fixes if an object was recreated with the same id via AJAX or DOM.
9883      * @param {String/HTMLElement/Element} el The id of the node, a DOM Node or an existing Element.
9884      * @return {Element} The Element object
9885      * @static
9886      */
9887     El.get = function(el){
9888         var ex, elm, id;
9889         if(!el){ return null; }
9890         if(typeof el == "string"){ // element id
9891             if(!(elm = document.getElementById(el))){
9892                 return null;
9893             }
9894             if(ex = El.cache[el]){
9895                 ex.dom = elm;
9896             }else{
9897                 ex = El.cache[el] = new El(elm);
9898             }
9899             return ex;
9900         }else if(el.tagName){ // dom element
9901             if(!(id = el.id)){
9902                 id = Roo.id(el);
9903             }
9904             if(ex = El.cache[id]){
9905                 ex.dom = el;
9906             }else{
9907                 ex = El.cache[id] = new El(el);
9908             }
9909             return ex;
9910         }else if(el instanceof El){
9911             if(el != docEl){
9912                 el.dom = document.getElementById(el.id) || el.dom; // refresh dom element in case no longer valid,
9913                                                               // catch case where it hasn't been appended
9914                 El.cache[el.id] = el; // in case it was created directly with Element(), let's cache it
9915             }
9916             return el;
9917         }else if(el.isComposite){
9918             return el;
9919         }else if(el instanceof Array){
9920             return El.select(el);
9921         }else if(el == document){
9922             // create a bogus element object representing the document object
9923             if(!docEl){
9924                 var f = function(){};
9925                 f.prototype = El.prototype;
9926                 docEl = new f();
9927                 docEl.dom = document;
9928             }
9929             return docEl;
9930         }
9931         return null;
9932     };
9933
9934     // private
9935     El.uncache = function(el){
9936         for(var i = 0, a = arguments, len = a.length; i < len; i++) {
9937             if(a[i]){
9938                 delete El.cache[a[i].id || a[i]];
9939             }
9940         }
9941     };
9942
9943     // private
9944     // Garbage collection - uncache elements/purge listeners on orphaned elements
9945     // so we don't hold a reference and cause the browser to retain them
9946     El.garbageCollect = function(){
9947         if(!Roo.enableGarbageCollector){
9948             clearInterval(El.collectorThread);
9949             return;
9950         }
9951         for(var eid in El.cache){
9952             var el = El.cache[eid], d = el.dom;
9953             // -------------------------------------------------------
9954             // Determining what is garbage:
9955             // -------------------------------------------------------
9956             // !d
9957             // dom node is null, definitely garbage
9958             // -------------------------------------------------------
9959             // !d.parentNode
9960             // no parentNode == direct orphan, definitely garbage
9961             // -------------------------------------------------------
9962             // !d.offsetParent && !document.getElementById(eid)
9963             // display none elements have no offsetParent so we will
9964             // also try to look it up by it's id. However, check
9965             // offsetParent first so we don't do unneeded lookups.
9966             // This enables collection of elements that are not orphans
9967             // directly, but somewhere up the line they have an orphan
9968             // parent.
9969             // -------------------------------------------------------
9970             if(!d || !d.parentNode || (!d.offsetParent && !document.getElementById(eid))){
9971                 delete El.cache[eid];
9972                 if(d && Roo.enableListenerCollection){
9973                     E.purgeElement(d);
9974                 }
9975             }
9976         }
9977     }
9978     El.collectorThreadId = setInterval(El.garbageCollect, 30000);
9979
9980
9981     // dom is optional
9982     El.Flyweight = function(dom){
9983         this.dom = dom;
9984     };
9985     El.Flyweight.prototype = El.prototype;
9986
9987     El._flyweights = {};
9988     /**
9989      * Gets the globally shared flyweight Element, with the passed node as the active element. Do not store a reference to this element -
9990      * the dom node can be overwritten by other code.
9991      * @param {String/HTMLElement} el The dom node or id
9992      * @param {String} named (optional) Allows for creation of named reusable flyweights to
9993      *                                  prevent conflicts (e.g. internally Roo uses "_internal")
9994      * @static
9995      * @return {Element} The shared Element object
9996      */
9997     El.fly = function(el, named){
9998         named = named || '_global';
9999         el = Roo.getDom(el);
10000         if(!el){
10001             return null;
10002         }
10003         if(!El._flyweights[named]){
10004             El._flyweights[named] = new El.Flyweight();
10005         }
10006         El._flyweights[named].dom = el;
10007         return El._flyweights[named];
10008     };
10009
10010     /**
10011      * Static method to retrieve Element objects. Uses simple caching to consistently return the same object.
10012      * Automatically fixes if an object was recreated with the same id via AJAX or DOM.
10013      * Shorthand of {@link Roo.Element#get}
10014      * @param {String/HTMLElement/Element} el The id of the node, a DOM Node or an existing Element.
10015      * @return {Element} The Element object
10016      * @member Roo
10017      * @method get
10018      */
10019     Roo.get = El.get;
10020     /**
10021      * Gets the globally shared flyweight Element, with the passed node as the active element. Do not store a reference to this element -
10022      * the dom node can be overwritten by other code.
10023      * Shorthand of {@link Roo.Element#fly}
10024      * @param {String/HTMLElement} el The dom node or id
10025      * @param {String} named (optional) Allows for creation of named reusable flyweights to
10026      *                                  prevent conflicts (e.g. internally Roo uses "_internal")
10027      * @static
10028      * @return {Element} The shared Element object
10029      * @member Roo
10030      * @method fly
10031      */
10032     Roo.fly = El.fly;
10033
10034     // speedy lookup for elements never to box adjust
10035     var noBoxAdjust = Roo.isStrict ? {
10036         select:1
10037     } : {
10038         input:1, select:1, textarea:1
10039     };
10040     if(Roo.isIE || Roo.isGecko){
10041         noBoxAdjust['button'] = 1;
10042     }
10043
10044
10045     Roo.EventManager.on(window, 'unload', function(){
10046         delete El.cache;
10047         delete El._flyweights;
10048     });
10049 })();
10050
10051
10052
10053
10054 if(Roo.DomQuery){
10055     Roo.Element.selectorFunction = Roo.DomQuery.select;
10056 }
10057
10058 Roo.Element.select = function(selector, unique, root){
10059     var els;
10060     if(typeof selector == "string"){
10061         els = Roo.Element.selectorFunction(selector, root);
10062     }else if(selector.length !== undefined){
10063         els = selector;
10064     }else{
10065         throw "Invalid selector";
10066     }
10067     if(unique === true){
10068         return new Roo.CompositeElement(els);
10069     }else{
10070         return new Roo.CompositeElementLite(els);
10071     }
10072 };
10073 /**
10074  * Selects elements based on the passed CSS selector to enable working on them as 1.
10075  * @param {String/Array} selector The CSS selector or an array of elements
10076  * @param {Boolean} unique (optional) true to create a unique Roo.Element for each element (defaults to a shared flyweight object)
10077  * @param {HTMLElement/String} root (optional) The root element of the query or id of the root
10078  * @return {CompositeElementLite/CompositeElement}
10079  * @member Roo
10080  * @method select
10081  */
10082 Roo.select = Roo.Element.select;
10083
10084
10085
10086
10087
10088
10089
10090
10091
10092
10093
10094
10095
10096
10097 /*
10098  * Based on:
10099  * Ext JS Library 1.1.1
10100  * Copyright(c) 2006-2007, Ext JS, LLC.
10101  *
10102  * Originally Released Under LGPL - original licence link has changed is not relivant.
10103  *
10104  * Fork - LGPL
10105  * <script type="text/javascript">
10106  */
10107
10108
10109
10110 //Notifies Element that fx methods are available
10111 Roo.enableFx = true;
10112
10113 /**
10114  * @class Roo.Fx
10115  * <p>A class to provide basic animation and visual effects support.  <b>Note:</b> This class is automatically applied
10116  * to the {@link Roo.Element} interface when included, so all effects calls should be performed via Element.
10117  * Conversely, since the effects are not actually defined in Element, Roo.Fx <b>must</b> be included in order for the 
10118  * Element effects to work.</p><br/>
10119  *
10120  * <p>It is important to note that although the Fx methods and many non-Fx Element methods support "method chaining" in that
10121  * they return the Element object itself as the method return value, it is not always possible to mix the two in a single
10122  * method chain.  The Fx methods use an internal effects queue so that each effect can be properly timed and sequenced.
10123  * Non-Fx methods, on the other hand, have no such internal queueing and will always execute immediately.  For this reason,
10124  * while it may be possible to mix certain Fx and non-Fx method calls in a single chain, it may not always provide the
10125  * expected results and should be done with care.</p><br/>
10126  *
10127  * <p>Motion effects support 8-way anchoring, meaning that you can choose one of 8 different anchor points on the Element
10128  * that will serve as either the start or end point of the animation.  Following are all of the supported anchor positions:</p>
10129 <pre>
10130 Value  Description
10131 -----  -----------------------------
10132 tl     The top left corner
10133 t      The center of the top edge
10134 tr     The top right corner
10135 l      The center of the left edge
10136 r      The center of the right edge
10137 bl     The bottom left corner
10138 b      The center of the bottom edge
10139 br     The bottom right corner
10140 </pre>
10141  * <b>Although some Fx methods accept specific custom config parameters, the ones shown in the Config Options section
10142  * below are common options that can be passed to any Fx method.</b>
10143  * @cfg {Function} callback A function called when the effect is finished
10144  * @cfg {Object} scope The scope of the effect function
10145  * @cfg {String} easing A valid Easing value for the effect
10146  * @cfg {String} afterCls A css class to apply after the effect
10147  * @cfg {Number} duration The length of time (in seconds) that the effect should last
10148  * @cfg {Boolean} remove Whether the Element should be removed from the DOM and destroyed after the effect finishes
10149  * @cfg {Boolean} useDisplay Whether to use the <i>display</i> CSS property instead of <i>visibility</i> when hiding Elements (only applies to 
10150  * effects that end with the element being visually hidden, ignored otherwise)
10151  * @cfg {String/Object/Function} afterStyle A style specification string, e.g. "width:100px", or an object in the form {width:"100px"}, or
10152  * a function which returns such a specification that will be applied to the Element after the effect finishes
10153  * @cfg {Boolean} block Whether the effect should block other effects from queueing while it runs
10154  * @cfg {Boolean} concurrent Whether to allow subsequently-queued effects to run at the same time as the current effect, or to ensure that they run in sequence
10155  * @cfg {Boolean} stopFx Whether subsequent effects should be stopped and removed after the current effect finishes
10156  */
10157 Roo.Fx = {
10158         /**
10159          * Slides the element into view.  An anchor point can be optionally passed to set the point of
10160          * origin for the slide effect.  This function automatically handles wrapping the element with
10161          * a fixed-size container if needed.  See the Fx class overview for valid anchor point options.
10162          * Usage:
10163          *<pre><code>
10164 // default: slide the element in from the top
10165 el.slideIn();
10166
10167 // custom: slide the element in from the right with a 2-second duration
10168 el.slideIn('r', { duration: 2 });
10169
10170 // common config options shown with default values
10171 el.slideIn('t', {
10172     easing: 'easeOut',
10173     duration: .5
10174 });
10175 </code></pre>
10176          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to top: 't')
10177          * @param {Object} options (optional) Object literal with any of the Fx config options
10178          * @return {Roo.Element} The Element
10179          */
10180     slideIn : function(anchor, o){
10181         var el = this.getFxEl();
10182         o = o || {};
10183
10184         el.queueFx(o, function(){
10185
10186             anchor = anchor || "t";
10187
10188             // fix display to visibility
10189             this.fixDisplay();
10190
10191             // restore values after effect
10192             var r = this.getFxRestore();
10193             var b = this.getBox();
10194             // fixed size for slide
10195             this.setSize(b);
10196
10197             // wrap if needed
10198             var wrap = this.fxWrap(r.pos, o, "hidden");
10199
10200             var st = this.dom.style;
10201             st.visibility = "visible";
10202             st.position = "absolute";
10203
10204             // clear out temp styles after slide and unwrap
10205             var after = function(){
10206                 el.fxUnwrap(wrap, r.pos, o);
10207                 st.width = r.width;
10208                 st.height = r.height;
10209                 el.afterFx(o);
10210             };
10211             // time to calc the positions
10212             var a, pt = {to: [b.x, b.y]}, bw = {to: b.width}, bh = {to: b.height};
10213
10214             switch(anchor.toLowerCase()){
10215                 case "t":
10216                     wrap.setSize(b.width, 0);
10217                     st.left = st.bottom = "0";
10218                     a = {height: bh};
10219                 break;
10220                 case "l":
10221                     wrap.setSize(0, b.height);
10222                     st.right = st.top = "0";
10223                     a = {width: bw};
10224                 break;
10225                 case "r":
10226                     wrap.setSize(0, b.height);
10227                     wrap.setX(b.right);
10228                     st.left = st.top = "0";
10229                     a = {width: bw, points: pt};
10230                 break;
10231                 case "b":
10232                     wrap.setSize(b.width, 0);
10233                     wrap.setY(b.bottom);
10234                     st.left = st.top = "0";
10235                     a = {height: bh, points: pt};
10236                 break;
10237                 case "tl":
10238                     wrap.setSize(0, 0);
10239                     st.right = st.bottom = "0";
10240                     a = {width: bw, height: bh};
10241                 break;
10242                 case "bl":
10243                     wrap.setSize(0, 0);
10244                     wrap.setY(b.y+b.height);
10245                     st.right = st.top = "0";
10246                     a = {width: bw, height: bh, points: pt};
10247                 break;
10248                 case "br":
10249                     wrap.setSize(0, 0);
10250                     wrap.setXY([b.right, b.bottom]);
10251                     st.left = st.top = "0";
10252                     a = {width: bw, height: bh, points: pt};
10253                 break;
10254                 case "tr":
10255                     wrap.setSize(0, 0);
10256                     wrap.setX(b.x+b.width);
10257                     st.left = st.bottom = "0";
10258                     a = {width: bw, height: bh, points: pt};
10259                 break;
10260             }
10261             this.dom.style.visibility = "visible";
10262             wrap.show();
10263
10264             arguments.callee.anim = wrap.fxanim(a,
10265                 o,
10266                 'motion',
10267                 .5,
10268                 'easeOut', after);
10269         });
10270         return this;
10271     },
10272     
10273         /**
10274          * Slides the element out of view.  An anchor point can be optionally passed to set the end point
10275          * for the slide effect.  When the effect is completed, the element will be hidden (visibility = 
10276          * 'hidden') but block elements will still take up space in the document.  The element must be removed
10277          * from the DOM using the 'remove' config option if desired.  This function automatically handles 
10278          * wrapping the element with a fixed-size container if needed.  See the Fx class overview for valid anchor point options.
10279          * Usage:
10280          *<pre><code>
10281 // default: slide the element out to the top
10282 el.slideOut();
10283
10284 // custom: slide the element out to the right with a 2-second duration
10285 el.slideOut('r', { duration: 2 });
10286
10287 // common config options shown with default values
10288 el.slideOut('t', {
10289     easing: 'easeOut',
10290     duration: .5,
10291     remove: false,
10292     useDisplay: false
10293 });
10294 </code></pre>
10295          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to top: 't')
10296          * @param {Object} options (optional) Object literal with any of the Fx config options
10297          * @return {Roo.Element} The Element
10298          */
10299     slideOut : function(anchor, o){
10300         var el = this.getFxEl();
10301         o = o || {};
10302
10303         el.queueFx(o, function(){
10304
10305             anchor = anchor || "t";
10306
10307             // restore values after effect
10308             var r = this.getFxRestore();
10309             
10310             var b = this.getBox();
10311             // fixed size for slide
10312             this.setSize(b);
10313
10314             // wrap if needed
10315             var wrap = this.fxWrap(r.pos, o, "visible");
10316
10317             var st = this.dom.style;
10318             st.visibility = "visible";
10319             st.position = "absolute";
10320
10321             wrap.setSize(b);
10322
10323             var after = function(){
10324                 if(o.useDisplay){
10325                     el.setDisplayed(false);
10326                 }else{
10327                     el.hide();
10328                 }
10329
10330                 el.fxUnwrap(wrap, r.pos, o);
10331
10332                 st.width = r.width;
10333                 st.height = r.height;
10334
10335                 el.afterFx(o);
10336             };
10337
10338             var a, zero = {to: 0};
10339             switch(anchor.toLowerCase()){
10340                 case "t":
10341                     st.left = st.bottom = "0";
10342                     a = {height: zero};
10343                 break;
10344                 case "l":
10345                     st.right = st.top = "0";
10346                     a = {width: zero};
10347                 break;
10348                 case "r":
10349                     st.left = st.top = "0";
10350                     a = {width: zero, points: {to:[b.right, b.y]}};
10351                 break;
10352                 case "b":
10353                     st.left = st.top = "0";
10354                     a = {height: zero, points: {to:[b.x, b.bottom]}};
10355                 break;
10356                 case "tl":
10357                     st.right = st.bottom = "0";
10358                     a = {width: zero, height: zero};
10359                 break;
10360                 case "bl":
10361                     st.right = st.top = "0";
10362                     a = {width: zero, height: zero, points: {to:[b.x, b.bottom]}};
10363                 break;
10364                 case "br":
10365                     st.left = st.top = "0";
10366                     a = {width: zero, height: zero, points: {to:[b.x+b.width, b.bottom]}};
10367                 break;
10368                 case "tr":
10369                     st.left = st.bottom = "0";
10370                     a = {width: zero, height: zero, points: {to:[b.right, b.y]}};
10371                 break;
10372             }
10373
10374             arguments.callee.anim = wrap.fxanim(a,
10375                 o,
10376                 'motion',
10377                 .5,
10378                 "easeOut", after);
10379         });
10380         return this;
10381     },
10382
10383         /**
10384          * Fades the element out while slowly expanding it in all directions.  When the effect is completed, the 
10385          * element will be hidden (visibility = 'hidden') but block elements will still take up space in the document. 
10386          * The element must be removed from the DOM using the 'remove' config option if desired.
10387          * Usage:
10388          *<pre><code>
10389 // default
10390 el.puff();
10391
10392 // common config options shown with default values
10393 el.puff({
10394     easing: 'easeOut',
10395     duration: .5,
10396     remove: false,
10397     useDisplay: false
10398 });
10399 </code></pre>
10400          * @param {Object} options (optional) Object literal with any of the Fx config options
10401          * @return {Roo.Element} The Element
10402          */
10403     puff : function(o){
10404         var el = this.getFxEl();
10405         o = o || {};
10406
10407         el.queueFx(o, function(){
10408             this.clearOpacity();
10409             this.show();
10410
10411             // restore values after effect
10412             var r = this.getFxRestore();
10413             var st = this.dom.style;
10414
10415             var after = function(){
10416                 if(o.useDisplay){
10417                     el.setDisplayed(false);
10418                 }else{
10419                     el.hide();
10420                 }
10421
10422                 el.clearOpacity();
10423
10424                 el.setPositioning(r.pos);
10425                 st.width = r.width;
10426                 st.height = r.height;
10427                 st.fontSize = '';
10428                 el.afterFx(o);
10429             };
10430
10431             var width = this.getWidth();
10432             var height = this.getHeight();
10433
10434             arguments.callee.anim = this.fxanim({
10435                     width : {to: this.adjustWidth(width * 2)},
10436                     height : {to: this.adjustHeight(height * 2)},
10437                     points : {by: [-(width * .5), -(height * .5)]},
10438                     opacity : {to: 0},
10439                     fontSize: {to:200, unit: "%"}
10440                 },
10441                 o,
10442                 'motion',
10443                 .5,
10444                 "easeOut", after);
10445         });
10446         return this;
10447     },
10448
10449         /**
10450          * Blinks the element as if it was clicked and then collapses on its center (similar to switching off a television).
10451          * When the effect is completed, the element will be hidden (visibility = 'hidden') but block elements will still 
10452          * take up space in the document. The element must be removed from the DOM using the 'remove' config option if desired.
10453          * Usage:
10454          *<pre><code>
10455 // default
10456 el.switchOff();
10457
10458 // all config options shown with default values
10459 el.switchOff({
10460     easing: 'easeIn',
10461     duration: .3,
10462     remove: false,
10463     useDisplay: false
10464 });
10465 </code></pre>
10466          * @param {Object} options (optional) Object literal with any of the Fx config options
10467          * @return {Roo.Element} The Element
10468          */
10469     switchOff : function(o){
10470         var el = this.getFxEl();
10471         o = o || {};
10472
10473         el.queueFx(o, function(){
10474             this.clearOpacity();
10475             this.clip();
10476
10477             // restore values after effect
10478             var r = this.getFxRestore();
10479             var st = this.dom.style;
10480
10481             var after = function(){
10482                 if(o.useDisplay){
10483                     el.setDisplayed(false);
10484                 }else{
10485                     el.hide();
10486                 }
10487
10488                 el.clearOpacity();
10489                 el.setPositioning(r.pos);
10490                 st.width = r.width;
10491                 st.height = r.height;
10492
10493                 el.afterFx(o);
10494             };
10495
10496             this.fxanim({opacity:{to:0.3}}, null, null, .1, null, function(){
10497                 this.clearOpacity();
10498                 (function(){
10499                     this.fxanim({
10500                         height:{to:1},
10501                         points:{by:[0, this.getHeight() * .5]}
10502                     }, o, 'motion', 0.3, 'easeIn', after);
10503                 }).defer(100, this);
10504             });
10505         });
10506         return this;
10507     },
10508
10509     /**
10510      * Highlights the Element by setting a color (applies to the background-color by default, but can be
10511      * changed using the "attr" config option) and then fading back to the original color. If no original
10512      * color is available, you should provide the "endColor" config option which will be cleared after the animation.
10513      * Usage:
10514 <pre><code>
10515 // default: highlight background to yellow
10516 el.highlight();
10517
10518 // custom: highlight foreground text to blue for 2 seconds
10519 el.highlight("0000ff", { attr: 'color', duration: 2 });
10520
10521 // common config options shown with default values
10522 el.highlight("ffff9c", {
10523     attr: "background-color", //can be any valid CSS property (attribute) that supports a color value
10524     endColor: (current color) or "ffffff",
10525     easing: 'easeIn',
10526     duration: 1
10527 });
10528 </code></pre>
10529      * @param {String} color (optional) The highlight color. Should be a 6 char hex color without the leading # (defaults to yellow: 'ffff9c')
10530      * @param {Object} options (optional) Object literal with any of the Fx config options
10531      * @return {Roo.Element} The Element
10532      */ 
10533     highlight : function(color, o){
10534         var el = this.getFxEl();
10535         o = o || {};
10536
10537         el.queueFx(o, function(){
10538             color = color || "ffff9c";
10539             attr = o.attr || "backgroundColor";
10540
10541             this.clearOpacity();
10542             this.show();
10543
10544             var origColor = this.getColor(attr);
10545             var restoreColor = this.dom.style[attr];
10546             endColor = (o.endColor || origColor) || "ffffff";
10547
10548             var after = function(){
10549                 el.dom.style[attr] = restoreColor;
10550                 el.afterFx(o);
10551             };
10552
10553             var a = {};
10554             a[attr] = {from: color, to: endColor};
10555             arguments.callee.anim = this.fxanim(a,
10556                 o,
10557                 'color',
10558                 1,
10559                 'easeIn', after);
10560         });
10561         return this;
10562     },
10563
10564    /**
10565     * Shows a ripple of exploding, attenuating borders to draw attention to an Element.
10566     * Usage:
10567 <pre><code>
10568 // default: a single light blue ripple
10569 el.frame();
10570
10571 // custom: 3 red ripples lasting 3 seconds total
10572 el.frame("ff0000", 3, { duration: 3 });
10573
10574 // common config options shown with default values
10575 el.frame("C3DAF9", 1, {
10576     duration: 1 //duration of entire animation (not each individual ripple)
10577     // Note: Easing is not configurable and will be ignored if included
10578 });
10579 </code></pre>
10580     * @param {String} color (optional) The color of the border.  Should be a 6 char hex color without the leading # (defaults to light blue: 'C3DAF9').
10581     * @param {Number} count (optional) The number of ripples to display (defaults to 1)
10582     * @param {Object} options (optional) Object literal with any of the Fx config options
10583     * @return {Roo.Element} The Element
10584     */
10585     frame : function(color, count, o){
10586         var el = this.getFxEl();
10587         o = o || {};
10588
10589         el.queueFx(o, function(){
10590             color = color || "#C3DAF9";
10591             if(color.length == 6){
10592                 color = "#" + color;
10593             }
10594             count = count || 1;
10595             duration = o.duration || 1;
10596             this.show();
10597
10598             var b = this.getBox();
10599             var animFn = function(){
10600                 var proxy = this.createProxy({
10601
10602                      style:{
10603                         visbility:"hidden",
10604                         position:"absolute",
10605                         "z-index":"35000", // yee haw
10606                         border:"0px solid " + color
10607                      }
10608                   });
10609                 var scale = Roo.isBorderBox ? 2 : 1;
10610                 proxy.animate({
10611                     top:{from:b.y, to:b.y - 20},
10612                     left:{from:b.x, to:b.x - 20},
10613                     borderWidth:{from:0, to:10},
10614                     opacity:{from:1, to:0},
10615                     height:{from:b.height, to:(b.height + (20*scale))},
10616                     width:{from:b.width, to:(b.width + (20*scale))}
10617                 }, duration, function(){
10618                     proxy.remove();
10619                 });
10620                 if(--count > 0){
10621                      animFn.defer((duration/2)*1000, this);
10622                 }else{
10623                     el.afterFx(o);
10624                 }
10625             };
10626             animFn.call(this);
10627         });
10628         return this;
10629     },
10630
10631    /**
10632     * Creates a pause before any subsequent queued effects begin.  If there are
10633     * no effects queued after the pause it will have no effect.
10634     * Usage:
10635 <pre><code>
10636 el.pause(1);
10637 </code></pre>
10638     * @param {Number} seconds The length of time to pause (in seconds)
10639     * @return {Roo.Element} The Element
10640     */
10641     pause : function(seconds){
10642         var el = this.getFxEl();
10643         var o = {};
10644
10645         el.queueFx(o, function(){
10646             setTimeout(function(){
10647                 el.afterFx(o);
10648             }, seconds * 1000);
10649         });
10650         return this;
10651     },
10652
10653    /**
10654     * Fade an element in (from transparent to opaque).  The ending opacity can be specified
10655     * using the "endOpacity" config option.
10656     * Usage:
10657 <pre><code>
10658 // default: fade in from opacity 0 to 100%
10659 el.fadeIn();
10660
10661 // custom: fade in from opacity 0 to 75% over 2 seconds
10662 el.fadeIn({ endOpacity: .75, duration: 2});
10663
10664 // common config options shown with default values
10665 el.fadeIn({
10666     endOpacity: 1, //can be any value between 0 and 1 (e.g. .5)
10667     easing: 'easeOut',
10668     duration: .5
10669 });
10670 </code></pre>
10671     * @param {Object} options (optional) Object literal with any of the Fx config options
10672     * @return {Roo.Element} The Element
10673     */
10674     fadeIn : function(o){
10675         var el = this.getFxEl();
10676         o = o || {};
10677         el.queueFx(o, function(){
10678             this.setOpacity(0);
10679             this.fixDisplay();
10680             this.dom.style.visibility = 'visible';
10681             var to = o.endOpacity || 1;
10682             arguments.callee.anim = this.fxanim({opacity:{to:to}},
10683                 o, null, .5, "easeOut", function(){
10684                 if(to == 1){
10685                     this.clearOpacity();
10686                 }
10687                 el.afterFx(o);
10688             });
10689         });
10690         return this;
10691     },
10692
10693    /**
10694     * Fade an element out (from opaque to transparent).  The ending opacity can be specified
10695     * using the "endOpacity" config option.
10696     * Usage:
10697 <pre><code>
10698 // default: fade out from the element's current opacity to 0
10699 el.fadeOut();
10700
10701 // custom: fade out from the element's current opacity to 25% over 2 seconds
10702 el.fadeOut({ endOpacity: .25, duration: 2});
10703
10704 // common config options shown with default values
10705 el.fadeOut({
10706     endOpacity: 0, //can be any value between 0 and 1 (e.g. .5)
10707     easing: 'easeOut',
10708     duration: .5
10709     remove: false,
10710     useDisplay: false
10711 });
10712 </code></pre>
10713     * @param {Object} options (optional) Object literal with any of the Fx config options
10714     * @return {Roo.Element} The Element
10715     */
10716     fadeOut : function(o){
10717         var el = this.getFxEl();
10718         o = o || {};
10719         el.queueFx(o, function(){
10720             arguments.callee.anim = this.fxanim({opacity:{to:o.endOpacity || 0}},
10721                 o, null, .5, "easeOut", function(){
10722                 if(this.visibilityMode == Roo.Element.DISPLAY || o.useDisplay){
10723                      this.dom.style.display = "none";
10724                 }else{
10725                      this.dom.style.visibility = "hidden";
10726                 }
10727                 this.clearOpacity();
10728                 el.afterFx(o);
10729             });
10730         });
10731         return this;
10732     },
10733
10734    /**
10735     * Animates the transition of an element's dimensions from a starting height/width
10736     * to an ending height/width.
10737     * Usage:
10738 <pre><code>
10739 // change height and width to 100x100 pixels
10740 el.scale(100, 100);
10741
10742 // common config options shown with default values.  The height and width will default to
10743 // the element's existing values if passed as null.
10744 el.scale(
10745     [element's width],
10746     [element's height], {
10747     easing: 'easeOut',
10748     duration: .35
10749 });
10750 </code></pre>
10751     * @param {Number} width  The new width (pass undefined to keep the original width)
10752     * @param {Number} height  The new height (pass undefined to keep the original height)
10753     * @param {Object} options (optional) Object literal with any of the Fx config options
10754     * @return {Roo.Element} The Element
10755     */
10756     scale : function(w, h, o){
10757         this.shift(Roo.apply({}, o, {
10758             width: w,
10759             height: h
10760         }));
10761         return this;
10762     },
10763
10764    /**
10765     * Animates the transition of any combination of an element's dimensions, xy position and/or opacity.
10766     * Any of these properties not specified in the config object will not be changed.  This effect 
10767     * requires that at least one new dimension, position or opacity setting must be passed in on
10768     * the config object in order for the function to have any effect.
10769     * Usage:
10770 <pre><code>
10771 // slide the element horizontally to x position 200 while changing the height and opacity
10772 el.shift({ x: 200, height: 50, opacity: .8 });
10773
10774 // common config options shown with default values.
10775 el.shift({
10776     width: [element's width],
10777     height: [element's height],
10778     x: [element's x position],
10779     y: [element's y position],
10780     opacity: [element's opacity],
10781     easing: 'easeOut',
10782     duration: .35
10783 });
10784 </code></pre>
10785     * @param {Object} options  Object literal with any of the Fx config options
10786     * @return {Roo.Element} The Element
10787     */
10788     shift : function(o){
10789         var el = this.getFxEl();
10790         o = o || {};
10791         el.queueFx(o, function(){
10792             var a = {}, w = o.width, h = o.height, x = o.x, y = o.y,  op = o.opacity;
10793             if(w !== undefined){
10794                 a.width = {to: this.adjustWidth(w)};
10795             }
10796             if(h !== undefined){
10797                 a.height = {to: this.adjustHeight(h)};
10798             }
10799             if(x !== undefined || y !== undefined){
10800                 a.points = {to: [
10801                     x !== undefined ? x : this.getX(),
10802                     y !== undefined ? y : this.getY()
10803                 ]};
10804             }
10805             if(op !== undefined){
10806                 a.opacity = {to: op};
10807             }
10808             if(o.xy !== undefined){
10809                 a.points = {to: o.xy};
10810             }
10811             arguments.callee.anim = this.fxanim(a,
10812                 o, 'motion', .35, "easeOut", function(){
10813                 el.afterFx(o);
10814             });
10815         });
10816         return this;
10817     },
10818
10819         /**
10820          * Slides the element while fading it out of view.  An anchor point can be optionally passed to set the 
10821          * ending point of the effect.
10822          * Usage:
10823          *<pre><code>
10824 // default: slide the element downward while fading out
10825 el.ghost();
10826
10827 // custom: slide the element out to the right with a 2-second duration
10828 el.ghost('r', { duration: 2 });
10829
10830 // common config options shown with default values
10831 el.ghost('b', {
10832     easing: 'easeOut',
10833     duration: .5
10834     remove: false,
10835     useDisplay: false
10836 });
10837 </code></pre>
10838          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to bottom: 'b')
10839          * @param {Object} options (optional) Object literal with any of the Fx config options
10840          * @return {Roo.Element} The Element
10841          */
10842     ghost : function(anchor, o){
10843         var el = this.getFxEl();
10844         o = o || {};
10845
10846         el.queueFx(o, function(){
10847             anchor = anchor || "b";
10848
10849             // restore values after effect
10850             var r = this.getFxRestore();
10851             var w = this.getWidth(),
10852                 h = this.getHeight();
10853
10854             var st = this.dom.style;
10855
10856             var after = function(){
10857                 if(o.useDisplay){
10858                     el.setDisplayed(false);
10859                 }else{
10860                     el.hide();
10861                 }
10862
10863                 el.clearOpacity();
10864                 el.setPositioning(r.pos);
10865                 st.width = r.width;
10866                 st.height = r.height;
10867
10868                 el.afterFx(o);
10869             };
10870
10871             var a = {opacity: {to: 0}, points: {}}, pt = a.points;
10872             switch(anchor.toLowerCase()){
10873                 case "t":
10874                     pt.by = [0, -h];
10875                 break;
10876                 case "l":
10877                     pt.by = [-w, 0];
10878                 break;
10879                 case "r":
10880                     pt.by = [w, 0];
10881                 break;
10882                 case "b":
10883                     pt.by = [0, h];
10884                 break;
10885                 case "tl":
10886                     pt.by = [-w, -h];
10887                 break;
10888                 case "bl":
10889                     pt.by = [-w, h];
10890                 break;
10891                 case "br":
10892                     pt.by = [w, h];
10893                 break;
10894                 case "tr":
10895                     pt.by = [w, -h];
10896                 break;
10897             }
10898
10899             arguments.callee.anim = this.fxanim(a,
10900                 o,
10901                 'motion',
10902                 .5,
10903                 "easeOut", after);
10904         });
10905         return this;
10906     },
10907
10908         /**
10909          * Ensures that all effects queued after syncFx is called on the element are
10910          * run concurrently.  This is the opposite of {@link #sequenceFx}.
10911          * @return {Roo.Element} The Element
10912          */
10913     syncFx : function(){
10914         this.fxDefaults = Roo.apply(this.fxDefaults || {}, {
10915             block : false,
10916             concurrent : true,
10917             stopFx : false
10918         });
10919         return this;
10920     },
10921
10922         /**
10923          * Ensures that all effects queued after sequenceFx is called on the element are
10924          * run in sequence.  This is the opposite of {@link #syncFx}.
10925          * @return {Roo.Element} The Element
10926          */
10927     sequenceFx : function(){
10928         this.fxDefaults = Roo.apply(this.fxDefaults || {}, {
10929             block : false,
10930             concurrent : false,
10931             stopFx : false
10932         });
10933         return this;
10934     },
10935
10936         /* @private */
10937     nextFx : function(){
10938         var ef = this.fxQueue[0];
10939         if(ef){
10940             ef.call(this);
10941         }
10942     },
10943
10944         /**
10945          * Returns true if the element has any effects actively running or queued, else returns false.
10946          * @return {Boolean} True if element has active effects, else false
10947          */
10948     hasActiveFx : function(){
10949         return this.fxQueue && this.fxQueue[0];
10950     },
10951
10952         /**
10953          * Stops any running effects and clears the element's internal effects queue if it contains
10954          * any additional effects that haven't started yet.
10955          * @return {Roo.Element} The Element
10956          */
10957     stopFx : function(){
10958         if(this.hasActiveFx()){
10959             var cur = this.fxQueue[0];
10960             if(cur && cur.anim && cur.anim.isAnimated()){
10961                 this.fxQueue = [cur]; // clear out others
10962                 cur.anim.stop(true);
10963             }
10964         }
10965         return this;
10966     },
10967
10968         /* @private */
10969     beforeFx : function(o){
10970         if(this.hasActiveFx() && !o.concurrent){
10971            if(o.stopFx){
10972                this.stopFx();
10973                return true;
10974            }
10975            return false;
10976         }
10977         return true;
10978     },
10979
10980         /**
10981          * Returns true if the element is currently blocking so that no other effect can be queued
10982          * until this effect is finished, else returns false if blocking is not set.  This is commonly
10983          * used to ensure that an effect initiated by a user action runs to completion prior to the
10984          * same effect being restarted (e.g., firing only one effect even if the user clicks several times).
10985          * @return {Boolean} True if blocking, else false
10986          */
10987     hasFxBlock : function(){
10988         var q = this.fxQueue;
10989         return q && q[0] && q[0].block;
10990     },
10991
10992         /* @private */
10993     queueFx : function(o, fn){
10994         if(!this.fxQueue){
10995             this.fxQueue = [];
10996         }
10997         if(!this.hasFxBlock()){
10998             Roo.applyIf(o, this.fxDefaults);
10999             if(!o.concurrent){
11000                 var run = this.beforeFx(o);
11001                 fn.block = o.block;
11002                 this.fxQueue.push(fn);
11003                 if(run){
11004                     this.nextFx();
11005                 }
11006             }else{
11007                 fn.call(this);
11008             }
11009         }
11010         return this;
11011     },
11012
11013         /* @private */
11014     fxWrap : function(pos, o, vis){
11015         var wrap;
11016         if(!o.wrap || !(wrap = Roo.get(o.wrap))){
11017             var wrapXY;
11018             if(o.fixPosition){
11019                 wrapXY = this.getXY();
11020             }
11021             var div = document.createElement("div");
11022             div.style.visibility = vis;
11023             wrap = Roo.get(this.dom.parentNode.insertBefore(div, this.dom));
11024             wrap.setPositioning(pos);
11025             if(wrap.getStyle("position") == "static"){
11026                 wrap.position("relative");
11027             }
11028             this.clearPositioning('auto');
11029             wrap.clip();
11030             wrap.dom.appendChild(this.dom);
11031             if(wrapXY){
11032                 wrap.setXY(wrapXY);
11033             }
11034         }
11035         return wrap;
11036     },
11037
11038         /* @private */
11039     fxUnwrap : function(wrap, pos, o){
11040         this.clearPositioning();
11041         this.setPositioning(pos);
11042         if(!o.wrap){
11043             wrap.dom.parentNode.insertBefore(this.dom, wrap.dom);
11044             wrap.remove();
11045         }
11046     },
11047
11048         /* @private */
11049     getFxRestore : function(){
11050         var st = this.dom.style;
11051         return {pos: this.getPositioning(), width: st.width, height : st.height};
11052     },
11053
11054         /* @private */
11055     afterFx : function(o){
11056         if(o.afterStyle){
11057             this.applyStyles(o.afterStyle);
11058         }
11059         if(o.afterCls){
11060             this.addClass(o.afterCls);
11061         }
11062         if(o.remove === true){
11063             this.remove();
11064         }
11065         Roo.callback(o.callback, o.scope, [this]);
11066         if(!o.concurrent){
11067             this.fxQueue.shift();
11068             this.nextFx();
11069         }
11070     },
11071
11072         /* @private */
11073     getFxEl : function(){ // support for composite element fx
11074         return Roo.get(this.dom);
11075     },
11076
11077         /* @private */
11078     fxanim : function(args, opt, animType, defaultDur, defaultEase, cb){
11079         animType = animType || 'run';
11080         opt = opt || {};
11081         var anim = Roo.lib.Anim[animType](
11082             this.dom, args,
11083             (opt.duration || defaultDur) || .35,
11084             (opt.easing || defaultEase) || 'easeOut',
11085             function(){
11086                 Roo.callback(cb, this);
11087             },
11088             this
11089         );
11090         opt.anim = anim;
11091         return anim;
11092     }
11093 };
11094
11095 // backwords compat
11096 Roo.Fx.resize = Roo.Fx.scale;
11097
11098 //When included, Roo.Fx is automatically applied to Element so that all basic
11099 //effects are available directly via the Element API
11100 Roo.apply(Roo.Element.prototype, Roo.Fx);/*
11101  * Based on:
11102  * Ext JS Library 1.1.1
11103  * Copyright(c) 2006-2007, Ext JS, LLC.
11104  *
11105  * Originally Released Under LGPL - original licence link has changed is not relivant.
11106  *
11107  * Fork - LGPL
11108  * <script type="text/javascript">
11109  */
11110
11111
11112 /**
11113  * @class Roo.CompositeElement
11114  * Standard composite class. Creates a Roo.Element for every element in the collection.
11115  * <br><br>
11116  * <b>NOTE: Although they are not listed, this class supports all of the set/update methods of Roo.Element. All Roo.Element
11117  * actions will be performed on all the elements in this collection.</b>
11118  * <br><br>
11119  * All methods return <i>this</i> and can be chained.
11120  <pre><code>
11121  var els = Roo.select("#some-el div.some-class", true);
11122  // or select directly from an existing element
11123  var el = Roo.get('some-el');
11124  el.select('div.some-class', true);
11125
11126  els.setWidth(100); // all elements become 100 width
11127  els.hide(true); // all elements fade out and hide
11128  // or
11129  els.setWidth(100).hide(true);
11130  </code></pre>
11131  */
11132 Roo.CompositeElement = function(els){
11133     this.elements = [];
11134     this.addElements(els);
11135 };
11136 Roo.CompositeElement.prototype = {
11137     isComposite: true,
11138     addElements : function(els){
11139         if(!els) {
11140             return this;
11141         }
11142         if(typeof els == "string"){
11143             els = Roo.Element.selectorFunction(els);
11144         }
11145         var yels = this.elements;
11146         var index = yels.length-1;
11147         for(var i = 0, len = els.length; i < len; i++) {
11148                 yels[++index] = Roo.get(els[i]);
11149         }
11150         return this;
11151     },
11152
11153     /**
11154     * Clears this composite and adds the elements returned by the passed selector.
11155     * @param {String/Array} els A string CSS selector, an array of elements or an element
11156     * @return {CompositeElement} this
11157     */
11158     fill : function(els){
11159         this.elements = [];
11160         this.add(els);
11161         return this;
11162     },
11163
11164     /**
11165     * Filters this composite to only elements that match the passed selector.
11166     * @param {String} selector A string CSS selector
11167     * @param {Boolean} inverse return inverse filter (not matches)
11168     * @return {CompositeElement} this
11169     */
11170     filter : function(selector, inverse){
11171         var els = [];
11172         inverse = inverse || false;
11173         this.each(function(el){
11174             var match = inverse ? !el.is(selector) : el.is(selector);
11175             if(match){
11176                 els[els.length] = el.dom;
11177             }
11178         });
11179         this.fill(els);
11180         return this;
11181     },
11182
11183     invoke : function(fn, args){
11184         var els = this.elements;
11185         for(var i = 0, len = els.length; i < len; i++) {
11186                 Roo.Element.prototype[fn].apply(els[i], args);
11187         }
11188         return this;
11189     },
11190     /**
11191     * Adds elements to this composite.
11192     * @param {String/Array} els A string CSS selector, an array of elements or an element
11193     * @return {CompositeElement} this
11194     */
11195     add : function(els){
11196         if(typeof els == "string"){
11197             this.addElements(Roo.Element.selectorFunction(els));
11198         }else if(els.length !== undefined){
11199             this.addElements(els);
11200         }else{
11201             this.addElements([els]);
11202         }
11203         return this;
11204     },
11205     /**
11206     * Calls the passed function passing (el, this, index) for each element in this composite.
11207     * @param {Function} fn The function to call
11208     * @param {Object} scope (optional) The <i>this</i> object (defaults to the element)
11209     * @return {CompositeElement} this
11210     */
11211     each : function(fn, scope){
11212         var els = this.elements;
11213         for(var i = 0, len = els.length; i < len; i++){
11214             if(fn.call(scope || els[i], els[i], this, i) === false) {
11215                 break;
11216             }
11217         }
11218         return this;
11219     },
11220
11221     /**
11222      * Returns the Element object at the specified index
11223      * @param {Number} index
11224      * @return {Roo.Element}
11225      */
11226     item : function(index){
11227         return this.elements[index] || null;
11228     },
11229
11230     /**
11231      * Returns the first Element
11232      * @return {Roo.Element}
11233      */
11234     first : function(){
11235         return this.item(0);
11236     },
11237
11238     /**
11239      * Returns the last Element
11240      * @return {Roo.Element}
11241      */
11242     last : function(){
11243         return this.item(this.elements.length-1);
11244     },
11245
11246     /**
11247      * Returns the number of elements in this composite
11248      * @return Number
11249      */
11250     getCount : function(){
11251         return this.elements.length;
11252     },
11253
11254     /**
11255      * Returns true if this composite contains the passed element
11256      * @return Boolean
11257      */
11258     contains : function(el){
11259         return this.indexOf(el) !== -1;
11260     },
11261
11262     /**
11263      * Returns true if this composite contains the passed element
11264      * @return Boolean
11265      */
11266     indexOf : function(el){
11267         return this.elements.indexOf(Roo.get(el));
11268     },
11269
11270
11271     /**
11272     * Removes the specified element(s).
11273     * @param {Mixed} el The id of an element, the Element itself, the index of the element in this composite
11274     * or an array of any of those.
11275     * @param {Boolean} removeDom (optional) True to also remove the element from the document
11276     * @return {CompositeElement} this
11277     */
11278     removeElement : function(el, removeDom){
11279         if(el instanceof Array){
11280             for(var i = 0, len = el.length; i < len; i++){
11281                 this.removeElement(el[i]);
11282             }
11283             return this;
11284         }
11285         var index = typeof el == 'number' ? el : this.indexOf(el);
11286         if(index !== -1){
11287             if(removeDom){
11288                 var d = this.elements[index];
11289                 if(d.dom){
11290                     d.remove();
11291                 }else{
11292                     d.parentNode.removeChild(d);
11293                 }
11294             }
11295             this.elements.splice(index, 1);
11296         }
11297         return this;
11298     },
11299
11300     /**
11301     * Replaces the specified element with the passed element.
11302     * @param {String/HTMLElement/Element/Number} el The id of an element, the Element itself, the index of the element in this composite
11303     * to replace.
11304     * @param {String/HTMLElement/Element} replacement The id of an element or the Element itself.
11305     * @param {Boolean} domReplace (Optional) True to remove and replace the element in the document too.
11306     * @return {CompositeElement} this
11307     */
11308     replaceElement : function(el, replacement, domReplace){
11309         var index = typeof el == 'number' ? el : this.indexOf(el);
11310         if(index !== -1){
11311             if(domReplace){
11312                 this.elements[index].replaceWith(replacement);
11313             }else{
11314                 this.elements.splice(index, 1, Roo.get(replacement))
11315             }
11316         }
11317         return this;
11318     },
11319
11320     /**
11321      * Removes all elements.
11322      */
11323     clear : function(){
11324         this.elements = [];
11325     }
11326 };
11327 (function(){
11328     Roo.CompositeElement.createCall = function(proto, fnName){
11329         if(!proto[fnName]){
11330             proto[fnName] = function(){
11331                 return this.invoke(fnName, arguments);
11332             };
11333         }
11334     };
11335     for(var fnName in Roo.Element.prototype){
11336         if(typeof Roo.Element.prototype[fnName] == "function"){
11337             Roo.CompositeElement.createCall(Roo.CompositeElement.prototype, fnName);
11338         }
11339     };
11340 })();
11341 /*
11342  * Based on:
11343  * Ext JS Library 1.1.1
11344  * Copyright(c) 2006-2007, Ext JS, LLC.
11345  *
11346  * Originally Released Under LGPL - original licence link has changed is not relivant.
11347  *
11348  * Fork - LGPL
11349  * <script type="text/javascript">
11350  */
11351
11352 /**
11353  * @class Roo.CompositeElementLite
11354  * @extends Roo.CompositeElement
11355  * Flyweight composite class. Reuses the same Roo.Element for element operations.
11356  <pre><code>
11357  var els = Roo.select("#some-el div.some-class");
11358  // or select directly from an existing element
11359  var el = Roo.get('some-el');
11360  el.select('div.some-class');
11361
11362  els.setWidth(100); // all elements become 100 width
11363  els.hide(true); // all elements fade out and hide
11364  // or
11365  els.setWidth(100).hide(true);
11366  </code></pre><br><br>
11367  * <b>NOTE: Although they are not listed, this class supports all of the set/update methods of Roo.Element. All Roo.Element
11368  * actions will be performed on all the elements in this collection.</b>
11369  */
11370 Roo.CompositeElementLite = function(els){
11371     Roo.CompositeElementLite.superclass.constructor.call(this, els);
11372     this.el = new Roo.Element.Flyweight();
11373 };
11374 Roo.extend(Roo.CompositeElementLite, Roo.CompositeElement, {
11375     addElements : function(els){
11376         if(els){
11377             if(els instanceof Array){
11378                 this.elements = this.elements.concat(els);
11379             }else{
11380                 var yels = this.elements;
11381                 var index = yels.length-1;
11382                 for(var i = 0, len = els.length; i < len; i++) {
11383                     yels[++index] = els[i];
11384                 }
11385             }
11386         }
11387         return this;
11388     },
11389     invoke : function(fn, args){
11390         var els = this.elements;
11391         var el = this.el;
11392         for(var i = 0, len = els.length; i < len; i++) {
11393             el.dom = els[i];
11394                 Roo.Element.prototype[fn].apply(el, args);
11395         }
11396         return this;
11397     },
11398     /**
11399      * Returns a flyweight Element of the dom element object at the specified index
11400      * @param {Number} index
11401      * @return {Roo.Element}
11402      */
11403     item : function(index){
11404         if(!this.elements[index]){
11405             return null;
11406         }
11407         this.el.dom = this.elements[index];
11408         return this.el;
11409     },
11410
11411     // fixes scope with flyweight
11412     addListener : function(eventName, handler, scope, opt){
11413         var els = this.elements;
11414         for(var i = 0, len = els.length; i < len; i++) {
11415             Roo.EventManager.on(els[i], eventName, handler, scope || els[i], opt);
11416         }
11417         return this;
11418     },
11419
11420     /**
11421     * Calls the passed function passing (el, this, index) for each element in this composite. <b>The element
11422     * passed is the flyweight (shared) Roo.Element instance, so if you require a
11423     * a reference to the dom node, use el.dom.</b>
11424     * @param {Function} fn The function to call
11425     * @param {Object} scope (optional) The <i>this</i> object (defaults to the element)
11426     * @return {CompositeElement} this
11427     */
11428     each : function(fn, scope){
11429         var els = this.elements;
11430         var el = this.el;
11431         for(var i = 0, len = els.length; i < len; i++){
11432             el.dom = els[i];
11433                 if(fn.call(scope || el, el, this, i) === false){
11434                 break;
11435             }
11436         }
11437         return this;
11438     },
11439
11440     indexOf : function(el){
11441         return this.elements.indexOf(Roo.getDom(el));
11442     },
11443
11444     replaceElement : function(el, replacement, domReplace){
11445         var index = typeof el == 'number' ? el : this.indexOf(el);
11446         if(index !== -1){
11447             replacement = Roo.getDom(replacement);
11448             if(domReplace){
11449                 var d = this.elements[index];
11450                 d.parentNode.insertBefore(replacement, d);
11451                 d.parentNode.removeChild(d);
11452             }
11453             this.elements.splice(index, 1, replacement);
11454         }
11455         return this;
11456     }
11457 });
11458 Roo.CompositeElementLite.prototype.on = Roo.CompositeElementLite.prototype.addListener;
11459
11460 /*
11461  * Based on:
11462  * Ext JS Library 1.1.1
11463  * Copyright(c) 2006-2007, Ext JS, LLC.
11464  *
11465  * Originally Released Under LGPL - original licence link has changed is not relivant.
11466  *
11467  * Fork - LGPL
11468  * <script type="text/javascript">
11469  */
11470
11471  
11472
11473 /**
11474  * @class Roo.data.Connection
11475  * @extends Roo.util.Observable
11476  * The class encapsulates a connection to the page's originating domain, allowing requests to be made
11477  * either to a configured URL, or to a URL specified at request time. 
11478  * 
11479  * Requests made by this class are asynchronous, and will return immediately. No data from
11480  * the server will be available to the statement immediately following the {@link #request} call.
11481  * To process returned data, use a callback in the request options object, or an event listener.
11482  * 
11483  * Note: If you are doing a file upload, you will not get a normal response object sent back to
11484  * your callback or event handler.  Since the upload is handled via in IFRAME, there is no XMLHttpRequest.
11485  * The response object is created using the innerHTML of the IFRAME's document as the responseText
11486  * property and, if present, the IFRAME's XML document as the responseXML property.
11487  * 
11488  * This means that a valid XML or HTML document must be returned. If JSON data is required, it is suggested
11489  * that it be placed either inside a &lt;textarea> in an HTML document and retrieved from the responseText
11490  * using a regex, or inside a CDATA section in an XML document and retrieved from the responseXML using
11491  * standard DOM methods.
11492  * @constructor
11493  * @param {Object} config a configuration object.
11494  */
11495 Roo.data.Connection = function(config){
11496     Roo.apply(this, config);
11497     this.addEvents({
11498         /**
11499          * @event beforerequest
11500          * Fires before a network request is made to retrieve a data object.
11501          * @param {Connection} conn This Connection object.
11502          * @param {Object} options The options config object passed to the {@link #request} method.
11503          */
11504         "beforerequest" : true,
11505         /**
11506          * @event requestcomplete
11507          * Fires if the request was successfully completed.
11508          * @param {Connection} conn This Connection object.
11509          * @param {Object} response The XHR object containing the response data.
11510          * See {@link http://www.w3.org/TR/XMLHttpRequest/} for details.
11511          * @param {Object} options The options config object passed to the {@link #request} method.
11512          */
11513         "requestcomplete" : true,
11514         /**
11515          * @event requestexception
11516          * Fires if an error HTTP status was returned from the server.
11517          * See {@link http://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html} for details of HTTP status codes.
11518          * @param {Connection} conn This Connection object.
11519          * @param {Object} response The XHR object containing the response data.
11520          * See {@link http://www.w3.org/TR/XMLHttpRequest/} for details.
11521          * @param {Object} options The options config object passed to the {@link #request} method.
11522          */
11523         "requestexception" : true
11524     });
11525     Roo.data.Connection.superclass.constructor.call(this);
11526 };
11527
11528 Roo.extend(Roo.data.Connection, Roo.util.Observable, {
11529     /**
11530      * @cfg {String} url (Optional) The default URL to be used for requests to the server. (defaults to undefined)
11531      */
11532     /**
11533      * @cfg {Object} extraParams (Optional) An object containing properties which are used as
11534      * extra parameters to each request made by this object. (defaults to undefined)
11535      */
11536     /**
11537      * @cfg {Object} defaultHeaders (Optional) An object containing request headers which are added
11538      *  to each request made by this object. (defaults to undefined)
11539      */
11540     /**
11541      * @cfg {String} method (Optional) The default HTTP method to be used for requests. (defaults to undefined; if not set but parms are present will use POST, otherwise GET)
11542      */
11543     /**
11544      * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
11545      */
11546     timeout : 30000,
11547     /**
11548      * @cfg {Boolean} autoAbort (Optional) Whether this request should abort any pending requests. (defaults to false)
11549      * @type Boolean
11550      */
11551     autoAbort:false,
11552
11553     /**
11554      * @cfg {Boolean} disableCaching (Optional) True to add a unique cache-buster param to GET requests. (defaults to true)
11555      * @type Boolean
11556      */
11557     disableCaching: true,
11558
11559     /**
11560      * Sends an HTTP request to a remote server.
11561      * @param {Object} options An object which may contain the following properties:<ul>
11562      * <li><b>url</b> {String} (Optional) The URL to which to send the request. Defaults to configured URL</li>
11563      * <li><b>params</b> {Object/String/Function} (Optional) An object containing properties which are used as parameters to the
11564      * request, a url encoded string or a function to call to get either.</li>
11565      * <li><b>method</b> {String} (Optional) The HTTP method to use for the request. Defaults to the configured method, or
11566      * if no method was configured, "GET" if no parameters are being sent, and "POST" if parameters are being sent.</li>
11567      * <li><b>callback</b> {Function} (Optional) The function to be called upon receipt of the HTTP response.
11568      * The callback is called regardless of success or failure and is passed the following parameters:<ul>
11569      * <li>options {Object} The parameter to the request call.</li>
11570      * <li>success {Boolean} True if the request succeeded.</li>
11571      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11572      * </ul></li>
11573      * <li><b>success</b> {Function} (Optional) The function to be called upon success of the request.
11574      * The callback is passed the following parameters:<ul>
11575      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11576      * <li>options {Object} The parameter to the request call.</li>
11577      * </ul></li>
11578      * <li><b>failure</b> {Function} (Optional) The function to be called upon failure of the request.
11579      * The callback is passed the following parameters:<ul>
11580      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11581      * <li>options {Object} The parameter to the request call.</li>
11582      * </ul></li>
11583      * <li><b>scope</b> {Object} (Optional) The scope in which to execute the callbacks: The "this" object
11584      * for the callback function. Defaults to the browser window.</li>
11585      * <li><b>form</b> {Object/String} (Optional) A form object or id to pull parameters from.</li>
11586      * <li><b>isUpload</b> {Boolean} (Optional) True if the form object is a file upload (will usually be automatically detected).</li>
11587      * <li><b>headers</b> {Object} (Optional) Request headers to set for the request.</li>
11588      * <li><b>xmlData</b> {Object} (Optional) XML document to use for the post. Note: This will be used instead of
11589      * params for the post data. Any params will be appended to the URL.</li>
11590      * <li><b>disableCaching</b> {Boolean} (Optional) True to add a unique cache-buster param to GET requests.</li>
11591      * </ul>
11592      * @return {Number} transactionId
11593      */
11594     request : function(o){
11595         if(this.fireEvent("beforerequest", this, o) !== false){
11596             var p = o.params;
11597
11598             if(typeof p == "function"){
11599                 p = p.call(o.scope||window, o);
11600             }
11601             if(typeof p == "object"){
11602                 p = Roo.urlEncode(o.params);
11603             }
11604             if(this.extraParams){
11605                 var extras = Roo.urlEncode(this.extraParams);
11606                 p = p ? (p + '&' + extras) : extras;
11607             }
11608
11609             var url = o.url || this.url;
11610             if(typeof url == 'function'){
11611                 url = url.call(o.scope||window, o);
11612             }
11613
11614             if(o.form){
11615                 var form = Roo.getDom(o.form);
11616                 url = url || form.action;
11617
11618                 var enctype = form.getAttribute("enctype");
11619                 
11620                 if (o.formData) {
11621                     return this.doFormDataUpload(o,p,url);
11622                 }
11623                 
11624                 if(o.isUpload || (enctype && enctype.toLowerCase() == 'multipart/form-data')){
11625                     return this.doFormUpload(o, p, url);
11626                 }
11627                 var f = Roo.lib.Ajax.serializeForm(form);
11628                 p = p ? (p + '&' + f) : f;
11629             }
11630
11631             var hs = o.headers;
11632             if(this.defaultHeaders){
11633                 hs = Roo.apply(hs || {}, this.defaultHeaders);
11634                 if(!o.headers){
11635                     o.headers = hs;
11636                 }
11637             }
11638
11639             var cb = {
11640                 success: this.handleResponse,
11641                 failure: this.handleFailure,
11642                 scope: this,
11643                 argument: {options: o},
11644                 timeout : o.timeout || this.timeout
11645             };
11646
11647             var method = o.method||this.method||(p ? "POST" : "GET");
11648
11649             if(method == 'GET' && (this.disableCaching && o.disableCaching !== false) || o.disableCaching === true){
11650                 url += (url.indexOf('?') != -1 ? '&' : '?') + '_dc=' + (new Date().getTime());
11651             }
11652
11653             if(typeof o.autoAbort == 'boolean'){ // options gets top priority
11654                 if(o.autoAbort){
11655                     this.abort();
11656                 }
11657             }else if(this.autoAbort !== false){
11658                 this.abort();
11659             }
11660
11661             if((method == 'GET' && p) || o.xmlData){
11662                 url += (url.indexOf('?') != -1 ? '&' : '?') + p;
11663                 p = '';
11664             }
11665             Roo.lib.Ajax.useDefaultHeader = typeof(o.headers) == 'undefined' || typeof(o.headers['Content-Type']) == 'undefined';
11666             this.transId = Roo.lib.Ajax.request(method, url, cb, p, o);
11667             Roo.lib.Ajax.useDefaultHeader == true;
11668             return this.transId;
11669         }else{
11670             Roo.callback(o.callback, o.scope, [o, null, null]);
11671             return null;
11672         }
11673     },
11674
11675     /**
11676      * Determine whether this object has a request outstanding.
11677      * @param {Number} transactionId (Optional) defaults to the last transaction
11678      * @return {Boolean} True if there is an outstanding request.
11679      */
11680     isLoading : function(transId){
11681         if(transId){
11682             return Roo.lib.Ajax.isCallInProgress(transId);
11683         }else{
11684             return this.transId ? true : false;
11685         }
11686     },
11687
11688     /**
11689      * Aborts any outstanding request.
11690      * @param {Number} transactionId (Optional) defaults to the last transaction
11691      */
11692     abort : function(transId){
11693         if(transId || this.isLoading()){
11694             Roo.lib.Ajax.abort(transId || this.transId);
11695         }
11696     },
11697
11698     // private
11699     handleResponse : function(response){
11700         this.transId = false;
11701         var options = response.argument.options;
11702         response.argument = options ? options.argument : null;
11703         this.fireEvent("requestcomplete", this, response, options);
11704         Roo.callback(options.success, options.scope, [response, options]);
11705         Roo.callback(options.callback, options.scope, [options, true, response]);
11706     },
11707
11708     // private
11709     handleFailure : function(response, e){
11710         this.transId = false;
11711         var options = response.argument.options;
11712         response.argument = options ? options.argument : null;
11713         this.fireEvent("requestexception", this, response, options, e);
11714         Roo.callback(options.failure, options.scope, [response, options]);
11715         Roo.callback(options.callback, options.scope, [options, false, response]);
11716     },
11717
11718     // private
11719     doFormUpload : function(o, ps, url){
11720         var id = Roo.id();
11721         var frame = document.createElement('iframe');
11722         frame.id = id;
11723         frame.name = id;
11724         frame.className = 'x-hidden';
11725         if(Roo.isIE){
11726             frame.src = Roo.SSL_SECURE_URL;
11727         }
11728         document.body.appendChild(frame);
11729
11730         if(Roo.isIE){
11731            document.frames[id].name = id;
11732         }
11733
11734         var form = Roo.getDom(o.form);
11735         form.target = id;
11736         form.method = 'POST';
11737         form.enctype = form.encoding = 'multipart/form-data';
11738         if(url){
11739             form.action = url;
11740         }
11741
11742         var hiddens, hd;
11743         if(ps){ // add dynamic params
11744             hiddens = [];
11745             ps = Roo.urlDecode(ps, false);
11746             for(var k in ps){
11747                 if(ps.hasOwnProperty(k)){
11748                     hd = document.createElement('input');
11749                     hd.type = 'hidden';
11750                     hd.name = k;
11751                     hd.value = ps[k];
11752                     form.appendChild(hd);
11753                     hiddens.push(hd);
11754                 }
11755             }
11756         }
11757
11758         function cb(){
11759             var r = {  // bogus response object
11760                 responseText : '',
11761                 responseXML : null
11762             };
11763
11764             r.argument = o ? o.argument : null;
11765
11766             try { //
11767                 var doc;
11768                 if(Roo.isIE){
11769                     doc = frame.contentWindow.document;
11770                 }else {
11771                     doc = (frame.contentDocument || window.frames[id].document);
11772                 }
11773                 if(doc && doc.body){
11774                     r.responseText = doc.body.innerHTML;
11775                 }
11776                 if(doc && doc.XMLDocument){
11777                     r.responseXML = doc.XMLDocument;
11778                 }else {
11779                     r.responseXML = doc;
11780                 }
11781             }
11782             catch(e) {
11783                 // ignore
11784             }
11785
11786             Roo.EventManager.removeListener(frame, 'load', cb, this);
11787
11788             this.fireEvent("requestcomplete", this, r, o);
11789             Roo.callback(o.success, o.scope, [r, o]);
11790             Roo.callback(o.callback, o.scope, [o, true, r]);
11791
11792             setTimeout(function(){document.body.removeChild(frame);}, 100);
11793         }
11794
11795         Roo.EventManager.on(frame, 'load', cb, this);
11796         form.submit();
11797
11798         if(hiddens){ // remove dynamic params
11799             for(var i = 0, len = hiddens.length; i < len; i++){
11800                 form.removeChild(hiddens[i]);
11801             }
11802         }
11803     },
11804     // this is a 'formdata version???'
11805     
11806     
11807     doFormDataUpload : function(o, ps, url)
11808     {
11809         var form = Roo.getDom(o.form);
11810         form.enctype = form.encoding = 'multipart/form-data';
11811         var formData = o.formData === true ? new FormData(form) : o.formData;
11812       
11813         var cb = {
11814             success: this.handleResponse,
11815             failure: this.handleFailure,
11816             scope: this,
11817             argument: {options: o},
11818             timeout : o.timeout || this.timeout
11819         };
11820  
11821         if(typeof o.autoAbort == 'boolean'){ // options gets top priority
11822             if(o.autoAbort){
11823                 this.abort();
11824             }
11825         }else if(this.autoAbort !== false){
11826             this.abort();
11827         }
11828
11829         //Roo.lib.Ajax.defaultPostHeader = null;
11830         Roo.lib.Ajax.useDefaultHeader = false;
11831         this.transId = Roo.lib.Ajax.request( "POST", url, cb, o.formData, o);
11832         Roo.lib.Ajax.useDefaultHeader = true;
11833  
11834          
11835     }
11836     
11837 });
11838 /*
11839  * Based on:
11840  * Ext JS Library 1.1.1
11841  * Copyright(c) 2006-2007, Ext JS, LLC.
11842  *
11843  * Originally Released Under LGPL - original licence link has changed is not relivant.
11844  *
11845  * Fork - LGPL
11846  * <script type="text/javascript">
11847  */
11848  
11849 /**
11850  * Global Ajax request class.
11851  * 
11852  * @class Roo.Ajax
11853  * @extends Roo.data.Connection
11854  * @static
11855  * 
11856  * @cfg {String} url  The default URL to be used for requests to the server. (defaults to undefined)
11857  * @cfg {Object} extraParams  An object containing properties which are used as extra parameters to each request made by this object. (defaults to undefined)
11858  * @cfg {Object} defaultHeaders  An object containing request headers which are added to each request made by this object. (defaults to undefined)
11859  * @cfg {String} method (Optional)  The default HTTP method to be used for requests. (defaults to undefined; if not set but parms are present will use POST, otherwise GET)
11860  * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
11861  * @cfg {Boolean} autoAbort (Optional) Whether a new request should abort any pending requests. (defaults to false)
11862  * @cfg {Boolean} disableCaching (Optional)   True to add a unique cache-buster param to GET requests. (defaults to true)
11863  */
11864 Roo.Ajax = new Roo.data.Connection({
11865     // fix up the docs
11866     /**
11867      * @scope Roo.Ajax
11868      * @type {Boolear} 
11869      */
11870     autoAbort : false,
11871
11872     /**
11873      * Serialize the passed form into a url encoded string
11874      * @scope Roo.Ajax
11875      * @param {String/HTMLElement} form
11876      * @return {String}
11877      */
11878     serializeForm : function(form){
11879         return Roo.lib.Ajax.serializeForm(form);
11880     }
11881 });/*
11882  * Based on:
11883  * Ext JS Library 1.1.1
11884  * Copyright(c) 2006-2007, Ext JS, LLC.
11885  *
11886  * Originally Released Under LGPL - original licence link has changed is not relivant.
11887  *
11888  * Fork - LGPL
11889  * <script type="text/javascript">
11890  */
11891
11892  
11893 /**
11894  * @class Roo.UpdateManager
11895  * @extends Roo.util.Observable
11896  * Provides AJAX-style update for Element object.<br><br>
11897  * Usage:<br>
11898  * <pre><code>
11899  * // Get it from a Roo.Element object
11900  * var el = Roo.get("foo");
11901  * var mgr = el.getUpdateManager();
11902  * mgr.update("http://myserver.com/index.php", "param1=1&amp;param2=2");
11903  * ...
11904  * mgr.formUpdate("myFormId", "http://myserver.com/index.php");
11905  * <br>
11906  * // or directly (returns the same UpdateManager instance)
11907  * var mgr = new Roo.UpdateManager("myElementId");
11908  * mgr.startAutoRefresh(60, "http://myserver.com/index.php");
11909  * mgr.on("update", myFcnNeedsToKnow);
11910  * <br>
11911    // short handed call directly from the element object
11912    Roo.get("foo").load({
11913         url: "bar.php",
11914         scripts:true,
11915         params: "for=bar",
11916         text: "Loading Foo..."
11917    });
11918  * </code></pre>
11919  * @constructor
11920  * Create new UpdateManager directly.
11921  * @param {String/HTMLElement/Roo.Element} el The element to update
11922  * @param {Boolean} forceNew (optional) By default the constructor checks to see if the passed element already has an UpdateManager and if it does it returns the same instance. This will skip that check (useful for extending this class).
11923  */
11924 Roo.UpdateManager = function(el, forceNew){
11925     el = Roo.get(el);
11926     if(!forceNew && el.updateManager){
11927         return el.updateManager;
11928     }
11929     /**
11930      * The Element object
11931      * @type Roo.Element
11932      */
11933     this.el = el;
11934     /**
11935      * Cached url to use for refreshes. Overwritten every time update() is called unless "discardUrl" param is set to true.
11936      * @type String
11937      */
11938     this.defaultUrl = null;
11939
11940     this.addEvents({
11941         /**
11942          * @event beforeupdate
11943          * Fired before an update is made, return false from your handler and the update is cancelled.
11944          * @param {Roo.Element} el
11945          * @param {String/Object/Function} url
11946          * @param {String/Object} params
11947          */
11948         "beforeupdate": true,
11949         /**
11950          * @event update
11951          * Fired after successful update is made.
11952          * @param {Roo.Element} el
11953          * @param {Object} oResponseObject The response Object
11954          */
11955         "update": true,
11956         /**
11957          * @event failure
11958          * Fired on update failure.
11959          * @param {Roo.Element} el
11960          * @param {Object} oResponseObject The response Object
11961          */
11962         "failure": true
11963     });
11964     var d = Roo.UpdateManager.defaults;
11965     /**
11966      * Blank page URL to use with SSL file uploads (Defaults to Roo.UpdateManager.defaults.sslBlankUrl or "about:blank").
11967      * @type String
11968      */
11969     this.sslBlankUrl = d.sslBlankUrl;
11970     /**
11971      * Whether to append unique parameter on get request to disable caching (Defaults to Roo.UpdateManager.defaults.disableCaching or false).
11972      * @type Boolean
11973      */
11974     this.disableCaching = d.disableCaching;
11975     /**
11976      * Text for loading indicator (Defaults to Roo.UpdateManager.defaults.indicatorText or '&lt;div class="loading-indicator"&gt;Loading...&lt;/div&gt;').
11977      * @type String
11978      */
11979     this.indicatorText = d.indicatorText;
11980     /**
11981      * Whether to show indicatorText when loading (Defaults to Roo.UpdateManager.defaults.showLoadIndicator or true).
11982      * @type String
11983      */
11984     this.showLoadIndicator = d.showLoadIndicator;
11985     /**
11986      * Timeout for requests or form posts in seconds (Defaults to Roo.UpdateManager.defaults.timeout or 30 seconds).
11987      * @type Number
11988      */
11989     this.timeout = d.timeout;
11990
11991     /**
11992      * True to process scripts in the output (Defaults to Roo.UpdateManager.defaults.loadScripts (false)).
11993      * @type Boolean
11994      */
11995     this.loadScripts = d.loadScripts;
11996
11997     /**
11998      * Transaction object of current executing transaction
11999      */
12000     this.transaction = null;
12001
12002     /**
12003      * @private
12004      */
12005     this.autoRefreshProcId = null;
12006     /**
12007      * Delegate for refresh() prebound to "this", use myUpdater.refreshDelegate.createCallback(arg1, arg2) to bind arguments
12008      * @type Function
12009      */
12010     this.refreshDelegate = this.refresh.createDelegate(this);
12011     /**
12012      * Delegate for update() prebound to "this", use myUpdater.updateDelegate.createCallback(arg1, arg2) to bind arguments
12013      * @type Function
12014      */
12015     this.updateDelegate = this.update.createDelegate(this);
12016     /**
12017      * Delegate for formUpdate() prebound to "this", use myUpdater.formUpdateDelegate.createCallback(arg1, arg2) to bind arguments
12018      * @type Function
12019      */
12020     this.formUpdateDelegate = this.formUpdate.createDelegate(this);
12021     /**
12022      * @private
12023      */
12024     this.successDelegate = this.processSuccess.createDelegate(this);
12025     /**
12026      * @private
12027      */
12028     this.failureDelegate = this.processFailure.createDelegate(this);
12029
12030     if(!this.renderer){
12031      /**
12032       * The renderer for this UpdateManager. Defaults to {@link Roo.UpdateManager.BasicRenderer}.
12033       */
12034     this.renderer = new Roo.UpdateManager.BasicRenderer();
12035     }
12036     
12037     Roo.UpdateManager.superclass.constructor.call(this);
12038 };
12039
12040 Roo.extend(Roo.UpdateManager, Roo.util.Observable, {
12041     /**
12042      * Get the Element this UpdateManager is bound to
12043      * @return {Roo.Element} The element
12044      */
12045     getEl : function(){
12046         return this.el;
12047     },
12048     /**
12049      * Performs an async request, updating this element with the response. If params are specified it uses POST, otherwise it uses GET.
12050      * @param {Object/String/Function} url The url for this request or a function to call to get the url or a config object containing any of the following options:
12051 <pre><code>
12052 um.update({<br/>
12053     url: "your-url.php",<br/>
12054     params: {param1: "foo", param2: "bar"}, // or a URL encoded string<br/>
12055     callback: yourFunction,<br/>
12056     scope: yourObject, //(optional scope)  <br/>
12057     discardUrl: false, <br/>
12058     nocache: false,<br/>
12059     text: "Loading...",<br/>
12060     timeout: 30,<br/>
12061     scripts: false<br/>
12062 });
12063 </code></pre>
12064      * The only required property is url. The optional properties nocache, text and scripts
12065      * are shorthand for disableCaching, indicatorText and loadScripts and are used to set their associated property on this UpdateManager instance.
12066      * @param {String/Object} params (optional) The parameters to pass as either a url encoded string "param1=1&amp;param2=2" or an object {param1: 1, param2: 2}
12067      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess, oResponse)
12068      * @param {Boolean} discardUrl (optional) By default when you execute an update the defaultUrl is changed to the last used url. If true, it will not store the url.
12069      */
12070     update : function(url, params, callback, discardUrl){
12071         if(this.fireEvent("beforeupdate", this.el, url, params) !== false){
12072             var method = this.method,
12073                 cfg;
12074             if(typeof url == "object"){ // must be config object
12075                 cfg = url;
12076                 url = cfg.url;
12077                 params = params || cfg.params;
12078                 callback = callback || cfg.callback;
12079                 discardUrl = discardUrl || cfg.discardUrl;
12080                 if(callback && cfg.scope){
12081                     callback = callback.createDelegate(cfg.scope);
12082                 }
12083                 if(typeof cfg.method != "undefined"){method = cfg.method;};
12084                 if(typeof cfg.nocache != "undefined"){this.disableCaching = cfg.nocache;};
12085                 if(typeof cfg.text != "undefined"){this.indicatorText = '<div class="loading-indicator">'+cfg.text+"</div>";};
12086                 if(typeof cfg.scripts != "undefined"){this.loadScripts = cfg.scripts;};
12087                 if(typeof cfg.timeout != "undefined"){this.timeout = cfg.timeout;};
12088             }
12089             this.showLoading();
12090             if(!discardUrl){
12091                 this.defaultUrl = url;
12092             }
12093             if(typeof url == "function"){
12094                 url = url.call(this);
12095             }
12096
12097             method = method || (params ? "POST" : "GET");
12098             if(method == "GET"){
12099                 url = this.prepareUrl(url);
12100             }
12101
12102             var o = Roo.apply(cfg ||{}, {
12103                 url : url,
12104                 params: params,
12105                 success: this.successDelegate,
12106                 failure: this.failureDelegate,
12107                 callback: undefined,
12108                 timeout: (this.timeout*1000),
12109                 argument: {"url": url, "form": null, "callback": callback, "params": params}
12110             });
12111             Roo.log("updated manager called with timeout of " + o.timeout);
12112             this.transaction = Roo.Ajax.request(o);
12113         }
12114     },
12115
12116     /**
12117      * Performs an async form post, updating this element with the response. If the form has the attribute enctype="multipart/form-data", it assumes it's a file upload.
12118      * Uses this.sslBlankUrl for SSL file uploads to prevent IE security warning.
12119      * @param {String/HTMLElement} form The form Id or form element
12120      * @param {String} url (optional) The url to pass the form to. If omitted the action attribute on the form will be used.
12121      * @param {Boolean} reset (optional) Whether to try to reset the form after the update
12122      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess, oResponse)
12123      */
12124     formUpdate : function(form, url, reset, callback){
12125         if(this.fireEvent("beforeupdate", this.el, form, url) !== false){
12126             if(typeof url == "function"){
12127                 url = url.call(this);
12128             }
12129             form = Roo.getDom(form);
12130             this.transaction = Roo.Ajax.request({
12131                 form: form,
12132                 url:url,
12133                 success: this.successDelegate,
12134                 failure: this.failureDelegate,
12135                 timeout: (this.timeout*1000),
12136                 argument: {"url": url, "form": form, "callback": callback, "reset": reset}
12137             });
12138             this.showLoading.defer(1, this);
12139         }
12140     },
12141
12142     /**
12143      * Refresh the element with the last used url or defaultUrl. If there is no url, it returns immediately
12144      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
12145      */
12146     refresh : function(callback){
12147         if(this.defaultUrl == null){
12148             return;
12149         }
12150         this.update(this.defaultUrl, null, callback, true);
12151     },
12152
12153     /**
12154      * Set this element to auto refresh.
12155      * @param {Number} interval How often to update (in seconds).
12156      * @param {String/Function} url (optional) The url for this request or a function to call to get the url (Defaults to the last used url)
12157      * @param {String/Object} params (optional) The parameters to pass as either a url encoded string "&param1=1&param2=2" or as an object {param1: 1, param2: 2}
12158      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
12159      * @param {Boolean} refreshNow (optional) Whether to execute the refresh now, or wait the interval
12160      */
12161     startAutoRefresh : function(interval, url, params, callback, refreshNow){
12162         if(refreshNow){
12163             this.update(url || this.defaultUrl, params, callback, true);
12164         }
12165         if(this.autoRefreshProcId){
12166             clearInterval(this.autoRefreshProcId);
12167         }
12168         this.autoRefreshProcId = setInterval(this.update.createDelegate(this, [url || this.defaultUrl, params, callback, true]), interval*1000);
12169     },
12170
12171     /**
12172      * Stop auto refresh on this element.
12173      */
12174      stopAutoRefresh : function(){
12175         if(this.autoRefreshProcId){
12176             clearInterval(this.autoRefreshProcId);
12177             delete this.autoRefreshProcId;
12178         }
12179     },
12180
12181     isAutoRefreshing : function(){
12182        return this.autoRefreshProcId ? true : false;
12183     },
12184     /**
12185      * Called to update the element to "Loading" state. Override to perform custom action.
12186      */
12187     showLoading : function(){
12188         if(this.showLoadIndicator){
12189             this.el.update(this.indicatorText);
12190         }
12191     },
12192
12193     /**
12194      * Adds unique parameter to query string if disableCaching = true
12195      * @private
12196      */
12197     prepareUrl : function(url){
12198         if(this.disableCaching){
12199             var append = "_dc=" + (new Date().getTime());
12200             if(url.indexOf("?") !== -1){
12201                 url += "&" + append;
12202             }else{
12203                 url += "?" + append;
12204             }
12205         }
12206         return url;
12207     },
12208
12209     /**
12210      * @private
12211      */
12212     processSuccess : function(response){
12213         this.transaction = null;
12214         if(response.argument.form && response.argument.reset){
12215             try{ // put in try/catch since some older FF releases had problems with this
12216                 response.argument.form.reset();
12217             }catch(e){}
12218         }
12219         if(this.loadScripts){
12220             this.renderer.render(this.el, response, this,
12221                 this.updateComplete.createDelegate(this, [response]));
12222         }else{
12223             this.renderer.render(this.el, response, this);
12224             this.updateComplete(response);
12225         }
12226     },
12227
12228     updateComplete : function(response){
12229         this.fireEvent("update", this.el, response);
12230         if(typeof response.argument.callback == "function"){
12231             response.argument.callback(this.el, true, response);
12232         }
12233     },
12234
12235     /**
12236      * @private
12237      */
12238     processFailure : function(response){
12239         this.transaction = null;
12240         this.fireEvent("failure", this.el, response);
12241         if(typeof response.argument.callback == "function"){
12242             response.argument.callback(this.el, false, response);
12243         }
12244     },
12245
12246     /**
12247      * Set the content renderer for this UpdateManager. See {@link Roo.UpdateManager.BasicRenderer#render} for more details.
12248      * @param {Object} renderer The object implementing the render() method
12249      */
12250     setRenderer : function(renderer){
12251         this.renderer = renderer;
12252     },
12253
12254     getRenderer : function(){
12255        return this.renderer;
12256     },
12257
12258     /**
12259      * Set the defaultUrl used for updates
12260      * @param {String/Function} defaultUrl The url or a function to call to get the url
12261      */
12262     setDefaultUrl : function(defaultUrl){
12263         this.defaultUrl = defaultUrl;
12264     },
12265
12266     /**
12267      * Aborts the executing transaction
12268      */
12269     abort : function(){
12270         if(this.transaction){
12271             Roo.Ajax.abort(this.transaction);
12272         }
12273     },
12274
12275     /**
12276      * Returns true if an update is in progress
12277      * @return {Boolean}
12278      */
12279     isUpdating : function(){
12280         if(this.transaction){
12281             return Roo.Ajax.isLoading(this.transaction);
12282         }
12283         return false;
12284     }
12285 });
12286
12287 /**
12288  * @class Roo.UpdateManager.defaults
12289  * @static (not really - but it helps the doc tool)
12290  * The defaults collection enables customizing the default properties of UpdateManager
12291  */
12292    Roo.UpdateManager.defaults = {
12293        /**
12294          * Timeout for requests or form posts in seconds (Defaults 30 seconds).
12295          * @type Number
12296          */
12297          timeout : 30,
12298
12299          /**
12300          * True to process scripts by default (Defaults to false).
12301          * @type Boolean
12302          */
12303         loadScripts : false,
12304
12305         /**
12306         * Blank page URL to use with SSL file uploads (Defaults to "javascript:false").
12307         * @type String
12308         */
12309         sslBlankUrl : (Roo.SSL_SECURE_URL || "javascript:false"),
12310         /**
12311          * Whether to append unique parameter on get request to disable caching (Defaults to false).
12312          * @type Boolean
12313          */
12314         disableCaching : false,
12315         /**
12316          * Whether to show indicatorText when loading (Defaults to true).
12317          * @type Boolean
12318          */
12319         showLoadIndicator : true,
12320         /**
12321          * Text for loading indicator (Defaults to '&lt;div class="loading-indicator"&gt;Loading...&lt;/div&gt;').
12322          * @type String
12323          */
12324         indicatorText : '<div class="loading-indicator">Loading...</div>'
12325    };
12326
12327 /**
12328  * Static convenience method. This method is deprecated in favor of el.load({url:'foo.php', ...}).
12329  *Usage:
12330  * <pre><code>Roo.UpdateManager.updateElement("my-div", "stuff.php");</code></pre>
12331  * @param {String/HTMLElement/Roo.Element} el The element to update
12332  * @param {String} url The url
12333  * @param {String/Object} params (optional) Url encoded param string or an object of name/value pairs
12334  * @param {Object} options (optional) A config object with any of the UpdateManager properties you want to set - for example: {disableCaching:true, indicatorText: "Loading data..."}
12335  * @static
12336  * @deprecated
12337  * @member Roo.UpdateManager
12338  */
12339 Roo.UpdateManager.updateElement = function(el, url, params, options){
12340     var um = Roo.get(el, true).getUpdateManager();
12341     Roo.apply(um, options);
12342     um.update(url, params, options ? options.callback : null);
12343 };
12344 // alias for backwards compat
12345 Roo.UpdateManager.update = Roo.UpdateManager.updateElement;
12346 /**
12347  * @class Roo.UpdateManager.BasicRenderer
12348  * Default Content renderer. Updates the elements innerHTML with the responseText.
12349  */
12350 Roo.UpdateManager.BasicRenderer = function(){};
12351
12352 Roo.UpdateManager.BasicRenderer.prototype = {
12353     /**
12354      * This is called when the transaction is completed and it's time to update the element - The BasicRenderer
12355      * updates the elements innerHTML with the responseText - To perform a custom render (i.e. XML or JSON processing),
12356      * create an object with a "render(el, response)" method and pass it to setRenderer on the UpdateManager.
12357      * @param {Roo.Element} el The element being rendered
12358      * @param {Object} response The YUI Connect response object
12359      * @param {UpdateManager} updateManager The calling update manager
12360      * @param {Function} callback A callback that will need to be called if loadScripts is true on the UpdateManager
12361      */
12362      render : function(el, response, updateManager, callback){
12363         el.update(response.responseText, updateManager.loadScripts, callback);
12364     }
12365 };
12366 /*
12367  * Based on:
12368  * Roo JS
12369  * (c)) Alan Knowles
12370  * Licence : LGPL
12371  */
12372
12373
12374 /**
12375  * @class Roo.DomTemplate
12376  * @extends Roo.Template
12377  * An effort at a dom based template engine..
12378  *
12379  * Similar to XTemplate, except it uses dom parsing to create the template..
12380  *
12381  * Supported features:
12382  *
12383  *  Tags:
12384
12385 <pre><code>
12386       {a_variable} - output encoded.
12387       {a_variable.format:("Y-m-d")} - call a method on the variable
12388       {a_variable:raw} - unencoded output
12389       {a_variable:toFixed(1,2)} - Roo.util.Format."toFixed"
12390       {a_variable:this.method_on_template(...)} - call a method on the template object.
12391  
12392 </code></pre>
12393  *  The tpl tag:
12394 <pre><code>
12395         &lt;div roo-for="a_variable or condition.."&gt;&lt;/div&gt;
12396         &lt;div roo-if="a_variable or condition"&gt;&lt;/div&gt;
12397         &lt;div roo-exec="some javascript"&gt;&lt;/div&gt;
12398         &lt;div roo-name="named_template"&gt;&lt;/div&gt; 
12399   
12400 </code></pre>
12401  *      
12402  */
12403 Roo.DomTemplate = function()
12404 {
12405      Roo.DomTemplate.superclass.constructor.apply(this, arguments);
12406      if (this.html) {
12407         this.compile();
12408      }
12409 };
12410
12411
12412 Roo.extend(Roo.DomTemplate, Roo.Template, {
12413     /**
12414      * id counter for sub templates.
12415      */
12416     id : 0,
12417     /**
12418      * flag to indicate if dom parser is inside a pre,
12419      * it will strip whitespace if not.
12420      */
12421     inPre : false,
12422     
12423     /**
12424      * The various sub templates
12425      */
12426     tpls : false,
12427     
12428     
12429     
12430     /**
12431      *
12432      * basic tag replacing syntax
12433      * WORD:WORD()
12434      *
12435      * // you can fake an object call by doing this
12436      *  x.t:(test,tesT) 
12437      * 
12438      */
12439     re : /(\{|\%7B)([\w-\.]+)(?:\:([\w\.]*)(?:\(([^)]*?)?\))?)?(\}|\%7D)/g,
12440     //re : /\{([\w-\.]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
12441     
12442     iterChild : function (node, method) {
12443         
12444         var oldPre = this.inPre;
12445         if (node.tagName == 'PRE') {
12446             this.inPre = true;
12447         }
12448         for( var i = 0; i < node.childNodes.length; i++) {
12449             method.call(this, node.childNodes[i]);
12450         }
12451         this.inPre = oldPre;
12452     },
12453     
12454     
12455     
12456     /**
12457      * compile the template
12458      *
12459      * This is not recursive, so I'm not sure how nested templates are really going to be handled..
12460      *
12461      */
12462     compile: function()
12463     {
12464         var s = this.html;
12465         
12466         // covert the html into DOM...
12467         var doc = false;
12468         var div =false;
12469         try {
12470             doc = document.implementation.createHTMLDocument("");
12471             doc.documentElement.innerHTML =   this.html  ;
12472             div = doc.documentElement;
12473         } catch (e) {
12474             // old IE... - nasty -- it causes all sorts of issues.. with
12475             // images getting pulled from server..
12476             div = document.createElement('div');
12477             div.innerHTML = this.html;
12478         }
12479         //doc.documentElement.innerHTML = htmlBody
12480          
12481         
12482         
12483         this.tpls = [];
12484         var _t = this;
12485         this.iterChild(div, function(n) {_t.compileNode(n, true); });
12486         
12487         var tpls = this.tpls;
12488         
12489         // create a top level template from the snippet..
12490         
12491         //Roo.log(div.innerHTML);
12492         
12493         var tpl = {
12494             uid : 'master',
12495             id : this.id++,
12496             attr : false,
12497             value : false,
12498             body : div.innerHTML,
12499             
12500             forCall : false,
12501             execCall : false,
12502             dom : div,
12503             isTop : true
12504             
12505         };
12506         tpls.unshift(tpl);
12507         
12508         
12509         // compile them...
12510         this.tpls = [];
12511         Roo.each(tpls, function(tp){
12512             this.compileTpl(tp);
12513             this.tpls[tp.id] = tp;
12514         }, this);
12515         
12516         this.master = tpls[0];
12517         return this;
12518         
12519         
12520     },
12521     
12522     compileNode : function(node, istop) {
12523         // test for
12524         //Roo.log(node);
12525         
12526         
12527         // skip anything not a tag..
12528         if (node.nodeType != 1) {
12529             if (node.nodeType == 3 && !this.inPre) {
12530                 // reduce white space..
12531                 node.nodeValue = node.nodeValue.replace(/\s+/g, ' '); 
12532                 
12533             }
12534             return;
12535         }
12536         
12537         var tpl = {
12538             uid : false,
12539             id : false,
12540             attr : false,
12541             value : false,
12542             body : '',
12543             
12544             forCall : false,
12545             execCall : false,
12546             dom : false,
12547             isTop : istop
12548             
12549             
12550         };
12551         
12552         
12553         switch(true) {
12554             case (node.hasAttribute('roo-for')): tpl.attr = 'for'; break;
12555             case (node.hasAttribute('roo-if')): tpl.attr = 'if'; break;
12556             case (node.hasAttribute('roo-name')): tpl.attr = 'name'; break;
12557             case (node.hasAttribute('roo-exec')): tpl.attr = 'exec'; break;
12558             // no default..
12559         }
12560         
12561         
12562         if (!tpl.attr) {
12563             // just itterate children..
12564             this.iterChild(node,this.compileNode);
12565             return;
12566         }
12567         tpl.uid = this.id++;
12568         tpl.value = node.getAttribute('roo-' +  tpl.attr);
12569         node.removeAttribute('roo-'+ tpl.attr);
12570         if (tpl.attr != 'name') {
12571             var placeholder = document.createTextNode('{domtpl' + tpl.uid + '}');
12572             node.parentNode.replaceChild(placeholder,  node);
12573         } else {
12574             
12575             var placeholder =  document.createElement('span');
12576             placeholder.className = 'roo-tpl-' + tpl.value;
12577             node.parentNode.replaceChild(placeholder,  node);
12578         }
12579         
12580         // parent now sees '{domtplXXXX}
12581         this.iterChild(node,this.compileNode);
12582         
12583         // we should now have node body...
12584         var div = document.createElement('div');
12585         div.appendChild(node);
12586         tpl.dom = node;
12587         // this has the unfortunate side effect of converting tagged attributes
12588         // eg. href="{...}" into %7C...%7D
12589         // this has been fixed by searching for those combo's although it's a bit hacky..
12590         
12591         
12592         tpl.body = div.innerHTML;
12593         
12594         
12595          
12596         tpl.id = tpl.uid;
12597         switch(tpl.attr) {
12598             case 'for' :
12599                 switch (tpl.value) {
12600                     case '.':  tpl.forCall = new Function('values', 'parent', 'with(values){ return values; }'); break;
12601                     case '..': tpl.forCall= new Function('values', 'parent', 'with(values){ return parent; }'); break;
12602                     default:   tpl.forCall= new Function('values', 'parent', 'with(values){ return '+tpl.value+'; }');
12603                 }
12604                 break;
12605             
12606             case 'exec':
12607                 tpl.execCall = new Function('values', 'parent', 'with(values){ '+(Roo.util.Format.htmlDecode(tpl.value))+'; }');
12608                 break;
12609             
12610             case 'if':     
12611                 tpl.ifCall = new Function('values', 'parent', 'with(values){ return '+(Roo.util.Format.htmlDecode(tpl.value))+'; }');
12612                 break;
12613             
12614             case 'name':
12615                 tpl.id  = tpl.value; // replace non characters???
12616                 break;
12617             
12618         }
12619         
12620         
12621         this.tpls.push(tpl);
12622         
12623         
12624         
12625     },
12626     
12627     
12628     
12629     
12630     /**
12631      * Compile a segment of the template into a 'sub-template'
12632      *
12633      * 
12634      * 
12635      *
12636      */
12637     compileTpl : function(tpl)
12638     {
12639         var fm = Roo.util.Format;
12640         var useF = this.disableFormats !== true;
12641         
12642         var sep = Roo.isGecko ? "+\n" : ",\n";
12643         
12644         var undef = function(str) {
12645             Roo.debug && Roo.log("Property not found :"  + str);
12646             return '';
12647         };
12648           
12649         //Roo.log(tpl.body);
12650         
12651         
12652         
12653         var fn = function(m, lbrace, name, format, args)
12654         {
12655             //Roo.log("ARGS");
12656             //Roo.log(arguments);
12657             args = args ? args.replace(/\\'/g,"'") : args;
12658             //["{TEST:(a,b,c)}", "TEST", "", "a,b,c", 0, "{TEST:(a,b,c)}"]
12659             if (typeof(format) == 'undefined') {
12660                 format =  'htmlEncode'; 
12661             }
12662             if (format == 'raw' ) {
12663                 format = false;
12664             }
12665             
12666             if(name.substr(0, 6) == 'domtpl'){
12667                 return "'"+ sep +'this.applySubTemplate('+name.substr(6)+', values, parent)'+sep+"'";
12668             }
12669             
12670             // build an array of options to determine if value is undefined..
12671             
12672             // basically get 'xxxx.yyyy' then do
12673             // (typeof(xxxx) == 'undefined' || typeof(xxx.yyyy) == 'undefined') ?
12674             //    (function () { Roo.log("Property not found"); return ''; })() :
12675             //    ......
12676             
12677             var udef_ar = [];
12678             var lookfor = '';
12679             Roo.each(name.split('.'), function(st) {
12680                 lookfor += (lookfor.length ? '.': '') + st;
12681                 udef_ar.push(  "(typeof(" + lookfor + ") == 'undefined')"  );
12682             });
12683             
12684             var udef_st = '((' + udef_ar.join(" || ") +") ? undef('" + name + "') : "; // .. needs )
12685             
12686             
12687             if(format && useF){
12688                 
12689                 args = args ? ',' + args : "";
12690                  
12691                 if(format.substr(0, 5) != "this."){
12692                     format = "fm." + format + '(';
12693                 }else{
12694                     format = 'this.call("'+ format.substr(5) + '", ';
12695                     args = ", values";
12696                 }
12697                 
12698                 return "'"+ sep +   udef_st   +    format + name + args + "))"+sep+"'";
12699             }
12700              
12701             if (args && args.length) {
12702                 // called with xxyx.yuu:(test,test)
12703                 // change to ()
12704                 return "'"+ sep + udef_st  + name + '(' +  args + "))"+sep+"'";
12705             }
12706             // raw.. - :raw modifier..
12707             return "'"+ sep + udef_st  + name + ")"+sep+"'";
12708             
12709         };
12710         var body;
12711         // branched to use + in gecko and [].join() in others
12712         if(Roo.isGecko){
12713             body = "tpl.compiled = function(values, parent){  with(values) { return '" +
12714                    tpl.body.replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
12715                     "';};};";
12716         }else{
12717             body = ["tpl.compiled = function(values, parent){  with (values) { return ['"];
12718             body.push(tpl.body.replace(/(\r\n|\n)/g,
12719                             '\\n').replace(/'/g, "\\'").replace(this.re, fn));
12720             body.push("'].join('');};};");
12721             body = body.join('');
12722         }
12723         
12724         Roo.debug && Roo.log(body.replace(/\\n/,'\n'));
12725        
12726         /** eval:var:tpl eval:var:fm eval:var:useF eval:var:undef  */
12727         eval(body);
12728         
12729         return this;
12730     },
12731      
12732     /**
12733      * same as applyTemplate, except it's done to one of the subTemplates
12734      * when using named templates, you can do:
12735      *
12736      * var str = pl.applySubTemplate('your-name', values);
12737      *
12738      * 
12739      * @param {Number} id of the template
12740      * @param {Object} values to apply to template
12741      * @param {Object} parent (normaly the instance of this object)
12742      */
12743     applySubTemplate : function(id, values, parent)
12744     {
12745         
12746         
12747         var t = this.tpls[id];
12748         
12749         
12750         try { 
12751             if(t.ifCall && !t.ifCall.call(this, values, parent)){
12752                 Roo.debug && Roo.log('if call on ' + t.value + ' return false');
12753                 return '';
12754             }
12755         } catch(e) {
12756             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-if="' + t.value + '" - ' + e.toString());
12757             Roo.log(values);
12758           
12759             return '';
12760         }
12761         try { 
12762             
12763             if(t.execCall && t.execCall.call(this, values, parent)){
12764                 return '';
12765             }
12766         } catch(e) {
12767             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-for="' + t.value + '" - ' + e.toString());
12768             Roo.log(values);
12769             return '';
12770         }
12771         
12772         try {
12773             var vs = t.forCall ? t.forCall.call(this, values, parent) : values;
12774             parent = t.target ? values : parent;
12775             if(t.forCall && vs instanceof Array){
12776                 var buf = [];
12777                 for(var i = 0, len = vs.length; i < len; i++){
12778                     try {
12779                         buf[buf.length] = t.compiled.call(this, vs[i], parent);
12780                     } catch (e) {
12781                         Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on body="' + t.value + '" - ' + e.toString());
12782                         Roo.log(e.body);
12783                         //Roo.log(t.compiled);
12784                         Roo.log(vs[i]);
12785                     }   
12786                 }
12787                 return buf.join('');
12788             }
12789         } catch (e) {
12790             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-for="' + t.value + '" - ' + e.toString());
12791             Roo.log(values);
12792             return '';
12793         }
12794         try {
12795             return t.compiled.call(this, vs, parent);
12796         } catch (e) {
12797             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on body="' + t.value + '" - ' + e.toString());
12798             Roo.log(e.body);
12799             //Roo.log(t.compiled);
12800             Roo.log(values);
12801             return '';
12802         }
12803     },
12804
12805    
12806
12807     applyTemplate : function(values){
12808         return this.master.compiled.call(this, values, {});
12809         //var s = this.subs;
12810     },
12811
12812     apply : function(){
12813         return this.applyTemplate.apply(this, arguments);
12814     }
12815
12816  });
12817
12818 Roo.DomTemplate.from = function(el){
12819     el = Roo.getDom(el);
12820     return new Roo.Domtemplate(el.value || el.innerHTML);
12821 };/*
12822  * Based on:
12823  * Ext JS Library 1.1.1
12824  * Copyright(c) 2006-2007, Ext JS, LLC.
12825  *
12826  * Originally Released Under LGPL - original licence link has changed is not relivant.
12827  *
12828  * Fork - LGPL
12829  * <script type="text/javascript">
12830  */
12831
12832 /**
12833  * @class Roo.util.DelayedTask
12834  * Provides a convenient method of performing setTimeout where a new
12835  * timeout cancels the old timeout. An example would be performing validation on a keypress.
12836  * You can use this class to buffer
12837  * the keypress events for a certain number of milliseconds, and perform only if they stop
12838  * for that amount of time.
12839  * @constructor The parameters to this constructor serve as defaults and are not required.
12840  * @param {Function} fn (optional) The default function to timeout
12841  * @param {Object} scope (optional) The default scope of that timeout
12842  * @param {Array} args (optional) The default Array of arguments
12843  */
12844 Roo.util.DelayedTask = function(fn, scope, args){
12845     var id = null, d, t;
12846
12847     var call = function(){
12848         var now = new Date().getTime();
12849         if(now - t >= d){
12850             clearInterval(id);
12851             id = null;
12852             fn.apply(scope, args || []);
12853         }
12854     };
12855     /**
12856      * Cancels any pending timeout and queues a new one
12857      * @param {Number} delay The milliseconds to delay
12858      * @param {Function} newFn (optional) Overrides function passed to constructor
12859      * @param {Object} newScope (optional) Overrides scope passed to constructor
12860      * @param {Array} newArgs (optional) Overrides args passed to constructor
12861      */
12862     this.delay = function(delay, newFn, newScope, newArgs){
12863         if(id && delay != d){
12864             this.cancel();
12865         }
12866         d = delay;
12867         t = new Date().getTime();
12868         fn = newFn || fn;
12869         scope = newScope || scope;
12870         args = newArgs || args;
12871         if(!id){
12872             id = setInterval(call, d);
12873         }
12874     };
12875
12876     /**
12877      * Cancel the last queued timeout
12878      */
12879     this.cancel = function(){
12880         if(id){
12881             clearInterval(id);
12882             id = null;
12883         }
12884     };
12885 };/*
12886  * Based on:
12887  * Ext JS Library 1.1.1
12888  * Copyright(c) 2006-2007, Ext JS, LLC.
12889  *
12890  * Originally Released Under LGPL - original licence link has changed is not relivant.
12891  *
12892  * Fork - LGPL
12893  * <script type="text/javascript">
12894  */
12895  
12896  
12897 Roo.util.TaskRunner = function(interval){
12898     interval = interval || 10;
12899     var tasks = [], removeQueue = [];
12900     var id = 0;
12901     var running = false;
12902
12903     var stopThread = function(){
12904         running = false;
12905         clearInterval(id);
12906         id = 0;
12907     };
12908
12909     var startThread = function(){
12910         if(!running){
12911             running = true;
12912             id = setInterval(runTasks, interval);
12913         }
12914     };
12915
12916     var removeTask = function(task){
12917         removeQueue.push(task);
12918         if(task.onStop){
12919             task.onStop();
12920         }
12921     };
12922
12923     var runTasks = function(){
12924         if(removeQueue.length > 0){
12925             for(var i = 0, len = removeQueue.length; i < len; i++){
12926                 tasks.remove(removeQueue[i]);
12927             }
12928             removeQueue = [];
12929             if(tasks.length < 1){
12930                 stopThread();
12931                 return;
12932             }
12933         }
12934         var now = new Date().getTime();
12935         for(var i = 0, len = tasks.length; i < len; ++i){
12936             var t = tasks[i];
12937             var itime = now - t.taskRunTime;
12938             if(t.interval <= itime){
12939                 var rt = t.run.apply(t.scope || t, t.args || [++t.taskRunCount]);
12940                 t.taskRunTime = now;
12941                 if(rt === false || t.taskRunCount === t.repeat){
12942                     removeTask(t);
12943                     return;
12944                 }
12945             }
12946             if(t.duration && t.duration <= (now - t.taskStartTime)){
12947                 removeTask(t);
12948             }
12949         }
12950     };
12951
12952     /**
12953      * Queues a new task.
12954      * @param {Object} task
12955      */
12956     this.start = function(task){
12957         tasks.push(task);
12958         task.taskStartTime = new Date().getTime();
12959         task.taskRunTime = 0;
12960         task.taskRunCount = 0;
12961         startThread();
12962         return task;
12963     };
12964
12965     this.stop = function(task){
12966         removeTask(task);
12967         return task;
12968     };
12969
12970     this.stopAll = function(){
12971         stopThread();
12972         for(var i = 0, len = tasks.length; i < len; i++){
12973             if(tasks[i].onStop){
12974                 tasks[i].onStop();
12975             }
12976         }
12977         tasks = [];
12978         removeQueue = [];
12979     };
12980 };
12981
12982 Roo.TaskMgr = new Roo.util.TaskRunner();/*
12983  * Based on:
12984  * Ext JS Library 1.1.1
12985  * Copyright(c) 2006-2007, Ext JS, LLC.
12986  *
12987  * Originally Released Under LGPL - original licence link has changed is not relivant.
12988  *
12989  * Fork - LGPL
12990  * <script type="text/javascript">
12991  */
12992
12993  
12994 /**
12995  * @class Roo.util.MixedCollection
12996  * @extends Roo.util.Observable
12997  * A Collection class that maintains both numeric indexes and keys and exposes events.
12998  * @constructor
12999  * @param {Boolean} allowFunctions True if the addAll function should add function references to the
13000  * collection (defaults to false)
13001  * @param {Function} keyFn A function that can accept an item of the type(s) stored in this MixedCollection
13002  * and return the key value for that item.  This is used when available to look up the key on items that
13003  * were passed without an explicit key parameter to a MixedCollection method.  Passing this parameter is
13004  * equivalent to providing an implementation for the {@link #getKey} method.
13005  */
13006 Roo.util.MixedCollection = function(allowFunctions, keyFn){
13007     this.items = [];
13008     this.map = {};
13009     this.keys = [];
13010     this.length = 0;
13011     this.addEvents({
13012         /**
13013          * @event clear
13014          * Fires when the collection is cleared.
13015          */
13016         "clear" : true,
13017         /**
13018          * @event add
13019          * Fires when an item is added to the collection.
13020          * @param {Number} index The index at which the item was added.
13021          * @param {Object} o The item added.
13022          * @param {String} key The key associated with the added item.
13023          */
13024         "add" : true,
13025         /**
13026          * @event replace
13027          * Fires when an item is replaced in the collection.
13028          * @param {String} key he key associated with the new added.
13029          * @param {Object} old The item being replaced.
13030          * @param {Object} new The new item.
13031          */
13032         "replace" : true,
13033         /**
13034          * @event remove
13035          * Fires when an item is removed from the collection.
13036          * @param {Object} o The item being removed.
13037          * @param {String} key (optional) The key associated with the removed item.
13038          */
13039         "remove" : true,
13040         "sort" : true
13041     });
13042     this.allowFunctions = allowFunctions === true;
13043     if(keyFn){
13044         this.getKey = keyFn;
13045     }
13046     Roo.util.MixedCollection.superclass.constructor.call(this);
13047 };
13048
13049 Roo.extend(Roo.util.MixedCollection, Roo.util.Observable, {
13050     allowFunctions : false,
13051     
13052 /**
13053  * Adds an item to the collection.
13054  * @param {String} key The key to associate with the item
13055  * @param {Object} o The item to add.
13056  * @return {Object} The item added.
13057  */
13058     add : function(key, o){
13059         if(arguments.length == 1){
13060             o = arguments[0];
13061             key = this.getKey(o);
13062         }
13063         if(typeof key == "undefined" || key === null){
13064             this.length++;
13065             this.items.push(o);
13066             this.keys.push(null);
13067         }else{
13068             var old = this.map[key];
13069             if(old){
13070                 return this.replace(key, o);
13071             }
13072             this.length++;
13073             this.items.push(o);
13074             this.map[key] = o;
13075             this.keys.push(key);
13076         }
13077         this.fireEvent("add", this.length-1, o, key);
13078         return o;
13079     },
13080        
13081 /**
13082   * MixedCollection has a generic way to fetch keys if you implement getKey.
13083 <pre><code>
13084 // normal way
13085 var mc = new Roo.util.MixedCollection();
13086 mc.add(someEl.dom.id, someEl);
13087 mc.add(otherEl.dom.id, otherEl);
13088 //and so on
13089
13090 // using getKey
13091 var mc = new Roo.util.MixedCollection();
13092 mc.getKey = function(el){
13093    return el.dom.id;
13094 };
13095 mc.add(someEl);
13096 mc.add(otherEl);
13097
13098 // or via the constructor
13099 var mc = new Roo.util.MixedCollection(false, function(el){
13100    return el.dom.id;
13101 });
13102 mc.add(someEl);
13103 mc.add(otherEl);
13104 </code></pre>
13105  * @param o {Object} The item for which to find the key.
13106  * @return {Object} The key for the passed item.
13107  */
13108     getKey : function(o){
13109          return o.id; 
13110     },
13111    
13112 /**
13113  * Replaces an item in the collection.
13114  * @param {String} key The key associated with the item to replace, or the item to replace.
13115  * @param o {Object} o (optional) If the first parameter passed was a key, the item to associate with that key.
13116  * @return {Object}  The new item.
13117  */
13118     replace : function(key, o){
13119         if(arguments.length == 1){
13120             o = arguments[0];
13121             key = this.getKey(o);
13122         }
13123         var old = this.item(key);
13124         if(typeof key == "undefined" || key === null || typeof old == "undefined"){
13125              return this.add(key, o);
13126         }
13127         var index = this.indexOfKey(key);
13128         this.items[index] = o;
13129         this.map[key] = o;
13130         this.fireEvent("replace", key, old, o);
13131         return o;
13132     },
13133    
13134 /**
13135  * Adds all elements of an Array or an Object to the collection.
13136  * @param {Object/Array} objs An Object containing properties which will be added to the collection, or
13137  * an Array of values, each of which are added to the collection.
13138  */
13139     addAll : function(objs){
13140         if(arguments.length > 1 || objs instanceof Array){
13141             var args = arguments.length > 1 ? arguments : objs;
13142             for(var i = 0, len = args.length; i < len; i++){
13143                 this.add(args[i]);
13144             }
13145         }else{
13146             for(var key in objs){
13147                 if(this.allowFunctions || typeof objs[key] != "function"){
13148                     this.add(key, objs[key]);
13149                 }
13150             }
13151         }
13152     },
13153    
13154 /**
13155  * Executes the specified function once for every item in the collection, passing each
13156  * item as the first and only parameter. returning false from the function will stop the iteration.
13157  * @param {Function} fn The function to execute for each item.
13158  * @param {Object} scope (optional) The scope in which to execute the function.
13159  */
13160     each : function(fn, scope){
13161         var items = [].concat(this.items); // each safe for removal
13162         for(var i = 0, len = items.length; i < len; i++){
13163             if(fn.call(scope || items[i], items[i], i, len) === false){
13164                 break;
13165             }
13166         }
13167     },
13168    
13169 /**
13170  * Executes the specified function once for every key in the collection, passing each
13171  * key, and its associated item as the first two parameters.
13172  * @param {Function} fn The function to execute for each item.
13173  * @param {Object} scope (optional) The scope in which to execute the function.
13174  */
13175     eachKey : function(fn, scope){
13176         for(var i = 0, len = this.keys.length; i < len; i++){
13177             fn.call(scope || window, this.keys[i], this.items[i], i, len);
13178         }
13179     },
13180    
13181 /**
13182  * Returns the first item in the collection which elicits a true return value from the
13183  * passed selection function.
13184  * @param {Function} fn The selection function to execute for each item.
13185  * @param {Object} scope (optional) The scope in which to execute the function.
13186  * @return {Object} The first item in the collection which returned true from the selection function.
13187  */
13188     find : function(fn, scope){
13189         for(var i = 0, len = this.items.length; i < len; i++){
13190             if(fn.call(scope || window, this.items[i], this.keys[i])){
13191                 return this.items[i];
13192             }
13193         }
13194         return null;
13195     },
13196    
13197 /**
13198  * Inserts an item at the specified index in the collection.
13199  * @param {Number} index The index to insert the item at.
13200  * @param {String} key The key to associate with the new item, or the item itself.
13201  * @param {Object} o  (optional) If the second parameter was a key, the new item.
13202  * @return {Object} The item inserted.
13203  */
13204     insert : function(index, key, o){
13205         if(arguments.length == 2){
13206             o = arguments[1];
13207             key = this.getKey(o);
13208         }
13209         if(index >= this.length){
13210             return this.add(key, o);
13211         }
13212         this.length++;
13213         this.items.splice(index, 0, o);
13214         if(typeof key != "undefined" && key != null){
13215             this.map[key] = o;
13216         }
13217         this.keys.splice(index, 0, key);
13218         this.fireEvent("add", index, o, key);
13219         return o;
13220     },
13221    
13222 /**
13223  * Removed an item from the collection.
13224  * @param {Object} o The item to remove.
13225  * @return {Object} The item removed.
13226  */
13227     remove : function(o){
13228         return this.removeAt(this.indexOf(o));
13229     },
13230    
13231 /**
13232  * Remove an item from a specified index in the collection.
13233  * @param {Number} index The index within the collection of the item to remove.
13234  */
13235     removeAt : function(index){
13236         if(index < this.length && index >= 0){
13237             this.length--;
13238             var o = this.items[index];
13239             this.items.splice(index, 1);
13240             var key = this.keys[index];
13241             if(typeof key != "undefined"){
13242                 delete this.map[key];
13243             }
13244             this.keys.splice(index, 1);
13245             this.fireEvent("remove", o, key);
13246         }
13247     },
13248    
13249 /**
13250  * Removed an item associated with the passed key fom the collection.
13251  * @param {String} key The key of the item to remove.
13252  */
13253     removeKey : function(key){
13254         return this.removeAt(this.indexOfKey(key));
13255     },
13256    
13257 /**
13258  * Returns the number of items in the collection.
13259  * @return {Number} the number of items in the collection.
13260  */
13261     getCount : function(){
13262         return this.length; 
13263     },
13264    
13265 /**
13266  * Returns index within the collection of the passed Object.
13267  * @param {Object} o The item to find the index of.
13268  * @return {Number} index of the item.
13269  */
13270     indexOf : function(o){
13271         if(!this.items.indexOf){
13272             for(var i = 0, len = this.items.length; i < len; i++){
13273                 if(this.items[i] == o) {
13274                     return i;
13275                 }
13276             }
13277             return -1;
13278         }else{
13279             return this.items.indexOf(o);
13280         }
13281     },
13282    
13283 /**
13284  * Returns index within the collection of the passed key.
13285  * @param {String} key The key to find the index of.
13286  * @return {Number} index of the key.
13287  */
13288     indexOfKey : function(key){
13289         if(!this.keys.indexOf){
13290             for(var i = 0, len = this.keys.length; i < len; i++){
13291                 if(this.keys[i] == key) {
13292                     return i;
13293                 }
13294             }
13295             return -1;
13296         }else{
13297             return this.keys.indexOf(key);
13298         }
13299     },
13300    
13301 /**
13302  * Returns the item associated with the passed key OR index. Key has priority over index.
13303  * @param {String/Number} key The key or index of the item.
13304  * @return {Object} The item associated with the passed key.
13305  */
13306     item : function(key){
13307         var item = typeof this.map[key] != "undefined" ? this.map[key] : this.items[key];
13308         return typeof item != 'function' || this.allowFunctions ? item : null; // for prototype!
13309     },
13310     
13311 /**
13312  * Returns the item at the specified index.
13313  * @param {Number} index The index of the item.
13314  * @return {Object}
13315  */
13316     itemAt : function(index){
13317         return this.items[index];
13318     },
13319     
13320 /**
13321  * Returns the item associated with the passed key.
13322  * @param {String/Number} key The key of the item.
13323  * @return {Object} The item associated with the passed key.
13324  */
13325     key : function(key){
13326         return this.map[key];
13327     },
13328    
13329 /**
13330  * Returns true if the collection contains the passed Object as an item.
13331  * @param {Object} o  The Object to look for in the collection.
13332  * @return {Boolean} True if the collection contains the Object as an item.
13333  */
13334     contains : function(o){
13335         return this.indexOf(o) != -1;
13336     },
13337    
13338 /**
13339  * Returns true if the collection contains the passed Object as a key.
13340  * @param {String} key The key to look for in the collection.
13341  * @return {Boolean} True if the collection contains the Object as a key.
13342  */
13343     containsKey : function(key){
13344         return typeof this.map[key] != "undefined";
13345     },
13346    
13347 /**
13348  * Removes all items from the collection.
13349  */
13350     clear : function(){
13351         this.length = 0;
13352         this.items = [];
13353         this.keys = [];
13354         this.map = {};
13355         this.fireEvent("clear");
13356     },
13357    
13358 /**
13359  * Returns the first item in the collection.
13360  * @return {Object} the first item in the collection..
13361  */
13362     first : function(){
13363         return this.items[0]; 
13364     },
13365    
13366 /**
13367  * Returns the last item in the collection.
13368  * @return {Object} the last item in the collection..
13369  */
13370     last : function(){
13371         return this.items[this.length-1];   
13372     },
13373     
13374     _sort : function(property, dir, fn){
13375         var dsc = String(dir).toUpperCase() == "DESC" ? -1 : 1;
13376         fn = fn || function(a, b){
13377             return a-b;
13378         };
13379         var c = [], k = this.keys, items = this.items;
13380         for(var i = 0, len = items.length; i < len; i++){
13381             c[c.length] = {key: k[i], value: items[i], index: i};
13382         }
13383         c.sort(function(a, b){
13384             var v = fn(a[property], b[property]) * dsc;
13385             if(v == 0){
13386                 v = (a.index < b.index ? -1 : 1);
13387             }
13388             return v;
13389         });
13390         for(var i = 0, len = c.length; i < len; i++){
13391             items[i] = c[i].value;
13392             k[i] = c[i].key;
13393         }
13394         this.fireEvent("sort", this);
13395     },
13396     
13397     /**
13398      * Sorts this collection with the passed comparison function
13399      * @param {String} direction (optional) "ASC" or "DESC"
13400      * @param {Function} fn (optional) comparison function
13401      */
13402     sort : function(dir, fn){
13403         this._sort("value", dir, fn);
13404     },
13405     
13406     /**
13407      * Sorts this collection by keys
13408      * @param {String} direction (optional) "ASC" or "DESC"
13409      * @param {Function} fn (optional) a comparison function (defaults to case insensitive string)
13410      */
13411     keySort : function(dir, fn){
13412         this._sort("key", dir, fn || function(a, b){
13413             return String(a).toUpperCase()-String(b).toUpperCase();
13414         });
13415     },
13416     
13417     /**
13418      * Returns a range of items in this collection
13419      * @param {Number} startIndex (optional) defaults to 0
13420      * @param {Number} endIndex (optional) default to the last item
13421      * @return {Array} An array of items
13422      */
13423     getRange : function(start, end){
13424         var items = this.items;
13425         if(items.length < 1){
13426             return [];
13427         }
13428         start = start || 0;
13429         end = Math.min(typeof end == "undefined" ? this.length-1 : end, this.length-1);
13430         var r = [];
13431         if(start <= end){
13432             for(var i = start; i <= end; i++) {
13433                     r[r.length] = items[i];
13434             }
13435         }else{
13436             for(var i = start; i >= end; i--) {
13437                     r[r.length] = items[i];
13438             }
13439         }
13440         return r;
13441     },
13442         
13443     /**
13444      * Filter the <i>objects</i> in this collection by a specific property. 
13445      * Returns a new collection that has been filtered.
13446      * @param {String} property A property on your objects
13447      * @param {String/RegExp} value Either string that the property values 
13448      * should start with or a RegExp to test against the property
13449      * @return {MixedCollection} The new filtered collection
13450      */
13451     filter : function(property, value){
13452         if(!value.exec){ // not a regex
13453             value = String(value);
13454             if(value.length == 0){
13455                 return this.clone();
13456             }
13457             value = new RegExp("^" + Roo.escapeRe(value), "i");
13458         }
13459         return this.filterBy(function(o){
13460             return o && value.test(o[property]);
13461         });
13462         },
13463     
13464     /**
13465      * Filter by a function. * Returns a new collection that has been filtered.
13466      * The passed function will be called with each 
13467      * object in the collection. If the function returns true, the value is included 
13468      * otherwise it is filtered.
13469      * @param {Function} fn The function to be called, it will receive the args o (the object), k (the key)
13470      * @param {Object} scope (optional) The scope of the function (defaults to this) 
13471      * @return {MixedCollection} The new filtered collection
13472      */
13473     filterBy : function(fn, scope){
13474         var r = new Roo.util.MixedCollection();
13475         r.getKey = this.getKey;
13476         var k = this.keys, it = this.items;
13477         for(var i = 0, len = it.length; i < len; i++){
13478             if(fn.call(scope||this, it[i], k[i])){
13479                                 r.add(k[i], it[i]);
13480                         }
13481         }
13482         return r;
13483     },
13484     
13485     /**
13486      * Creates a duplicate of this collection
13487      * @return {MixedCollection}
13488      */
13489     clone : function(){
13490         var r = new Roo.util.MixedCollection();
13491         var k = this.keys, it = this.items;
13492         for(var i = 0, len = it.length; i < len; i++){
13493             r.add(k[i], it[i]);
13494         }
13495         r.getKey = this.getKey;
13496         return r;
13497     }
13498 });
13499 /**
13500  * Returns the item associated with the passed key or index.
13501  * @method
13502  * @param {String/Number} key The key or index of the item.
13503  * @return {Object} The item associated with the passed key.
13504  */
13505 Roo.util.MixedCollection.prototype.get = Roo.util.MixedCollection.prototype.item;/*
13506  * Based on:
13507  * Ext JS Library 1.1.1
13508  * Copyright(c) 2006-2007, Ext JS, LLC.
13509  *
13510  * Originally Released Under LGPL - original licence link has changed is not relivant.
13511  *
13512  * Fork - LGPL
13513  * <script type="text/javascript">
13514  */
13515 /**
13516  * @class Roo.util.JSON
13517  * Modified version of Douglas Crockford"s json.js that doesn"t
13518  * mess with the Object prototype 
13519  * http://www.json.org/js.html
13520  * @singleton
13521  */
13522 Roo.util.JSON = new (function(){
13523     var useHasOwn = {}.hasOwnProperty ? true : false;
13524     
13525     // crashes Safari in some instances
13526     //var validRE = /^("(\\.|[^"\\\n\r])*?"|[,:{}\[\]0-9.\-+Eaeflnr-u \n\r\t])+?$/;
13527     
13528     var pad = function(n) {
13529         return n < 10 ? "0" + n : n;
13530     };
13531     
13532     var m = {
13533         "\b": '\\b',
13534         "\t": '\\t',
13535         "\n": '\\n',
13536         "\f": '\\f',
13537         "\r": '\\r',
13538         '"' : '\\"',
13539         "\\": '\\\\'
13540     };
13541
13542     var encodeString = function(s){
13543         if (/["\\\x00-\x1f]/.test(s)) {
13544             return '"' + s.replace(/([\x00-\x1f\\"])/g, function(a, b) {
13545                 var c = m[b];
13546                 if(c){
13547                     return c;
13548                 }
13549                 c = b.charCodeAt();
13550                 return "\\u00" +
13551                     Math.floor(c / 16).toString(16) +
13552                     (c % 16).toString(16);
13553             }) + '"';
13554         }
13555         return '"' + s + '"';
13556     };
13557     
13558     var encodeArray = function(o){
13559         var a = ["["], b, i, l = o.length, v;
13560             for (i = 0; i < l; i += 1) {
13561                 v = o[i];
13562                 switch (typeof v) {
13563                     case "undefined":
13564                     case "function":
13565                     case "unknown":
13566                         break;
13567                     default:
13568                         if (b) {
13569                             a.push(',');
13570                         }
13571                         a.push(v === null ? "null" : Roo.util.JSON.encode(v));
13572                         b = true;
13573                 }
13574             }
13575             a.push("]");
13576             return a.join("");
13577     };
13578     
13579     var encodeDate = function(o){
13580         return '"' + o.getFullYear() + "-" +
13581                 pad(o.getMonth() + 1) + "-" +
13582                 pad(o.getDate()) + "T" +
13583                 pad(o.getHours()) + ":" +
13584                 pad(o.getMinutes()) + ":" +
13585                 pad(o.getSeconds()) + '"';
13586     };
13587     
13588     /**
13589      * Encodes an Object, Array or other value
13590      * @param {Mixed} o The variable to encode
13591      * @return {String} The JSON string
13592      */
13593     this.encode = function(o)
13594     {
13595         // should this be extended to fully wrap stringify..
13596         
13597         if(typeof o == "undefined" || o === null){
13598             return "null";
13599         }else if(o instanceof Array){
13600             return encodeArray(o);
13601         }else if(o instanceof Date){
13602             return encodeDate(o);
13603         }else if(typeof o == "string"){
13604             return encodeString(o);
13605         }else if(typeof o == "number"){
13606             return isFinite(o) ? String(o) : "null";
13607         }else if(typeof o == "boolean"){
13608             return String(o);
13609         }else {
13610             var a = ["{"], b, i, v;
13611             for (i in o) {
13612                 if(!useHasOwn || o.hasOwnProperty(i)) {
13613                     v = o[i];
13614                     switch (typeof v) {
13615                     case "undefined":
13616                     case "function":
13617                     case "unknown":
13618                         break;
13619                     default:
13620                         if(b){
13621                             a.push(',');
13622                         }
13623                         a.push(this.encode(i), ":",
13624                                 v === null ? "null" : this.encode(v));
13625                         b = true;
13626                     }
13627                 }
13628             }
13629             a.push("}");
13630             return a.join("");
13631         }
13632     };
13633     
13634     /**
13635      * Decodes (parses) a JSON string to an object. If the JSON is invalid, this function throws a SyntaxError.
13636      * @param {String} json The JSON string
13637      * @return {Object} The resulting object
13638      */
13639     this.decode = function(json){
13640         
13641         return  /** eval:var:json */ eval("(" + json + ')');
13642     };
13643 })();
13644 /** 
13645  * Shorthand for {@link Roo.util.JSON#encode}
13646  * @member Roo encode 
13647  * @method */
13648 Roo.encode = typeof(JSON) != 'undefined' && JSON.stringify ? JSON.stringify : Roo.util.JSON.encode;
13649 /** 
13650  * Shorthand for {@link Roo.util.JSON#decode}
13651  * @member Roo decode 
13652  * @method */
13653 Roo.decode = typeof(JSON) != 'undefined' && JSON.parse ? JSON.parse : Roo.util.JSON.decode;
13654 /*
13655  * Based on:
13656  * Ext JS Library 1.1.1
13657  * Copyright(c) 2006-2007, Ext JS, LLC.
13658  *
13659  * Originally Released Under LGPL - original licence link has changed is not relivant.
13660  *
13661  * Fork - LGPL
13662  * <script type="text/javascript">
13663  */
13664  
13665 /**
13666  * @class Roo.util.Format
13667  * Reusable data formatting functions
13668  * @singleton
13669  */
13670 Roo.util.Format = function(){
13671     var trimRe = /^\s+|\s+$/g;
13672     return {
13673         /**
13674          * Truncate a string and add an ellipsis ('...') to the end if it exceeds the specified length
13675          * @param {String} value The string to truncate
13676          * @param {Number} length The maximum length to allow before truncating
13677          * @return {String} The converted text
13678          */
13679         ellipsis : function(value, len){
13680             if(value && value.length > len){
13681                 return value.substr(0, len-3)+"...";
13682             }
13683             return value;
13684         },
13685
13686         /**
13687          * Checks a reference and converts it to empty string if it is undefined
13688          * @param {Mixed} value Reference to check
13689          * @return {Mixed} Empty string if converted, otherwise the original value
13690          */
13691         undef : function(value){
13692             return typeof value != "undefined" ? value : "";
13693         },
13694
13695         /**
13696          * Convert certain characters (&, <, >, and ') to their HTML character equivalents for literal display in web pages.
13697          * @param {String} value The string to encode
13698          * @return {String} The encoded text
13699          */
13700         htmlEncode : function(value){
13701             return !value ? value : String(value).replace(/&/g, "&amp;").replace(/>/g, "&gt;").replace(/</g, "&lt;").replace(/"/g, "&quot;");
13702         },
13703
13704         /**
13705          * Convert certain characters (&, <, >, and ') from their HTML character equivalents.
13706          * @param {String} value The string to decode
13707          * @return {String} The decoded text
13708          */
13709         htmlDecode : function(value){
13710             return !value ? value : String(value).replace(/&amp;/g, "&").replace(/&gt;/g, ">").replace(/&lt;/g, "<").replace(/&quot;/g, '"');
13711         },
13712
13713         /**
13714          * Trims any whitespace from either side of a string
13715          * @param {String} value The text to trim
13716          * @return {String} The trimmed text
13717          */
13718         trim : function(value){
13719             return String(value).replace(trimRe, "");
13720         },
13721
13722         /**
13723          * Returns a substring from within an original string
13724          * @param {String} value The original text
13725          * @param {Number} start The start index of the substring
13726          * @param {Number} length The length of the substring
13727          * @return {String} The substring
13728          */
13729         substr : function(value, start, length){
13730             return String(value).substr(start, length);
13731         },
13732
13733         /**
13734          * Converts a string to all lower case letters
13735          * @param {String} value The text to convert
13736          * @return {String} The converted text
13737          */
13738         lowercase : function(value){
13739             return String(value).toLowerCase();
13740         },
13741
13742         /**
13743          * Converts a string to all upper case letters
13744          * @param {String} value The text to convert
13745          * @return {String} The converted text
13746          */
13747         uppercase : function(value){
13748             return String(value).toUpperCase();
13749         },
13750
13751         /**
13752          * Converts the first character only of a string to upper case
13753          * @param {String} value The text to convert
13754          * @return {String} The converted text
13755          */
13756         capitalize : function(value){
13757             return !value ? value : value.charAt(0).toUpperCase() + value.substr(1).toLowerCase();
13758         },
13759
13760         // private
13761         call : function(value, fn){
13762             if(arguments.length > 2){
13763                 var args = Array.prototype.slice.call(arguments, 2);
13764                 args.unshift(value);
13765                  
13766                 return /** eval:var:value */  eval(fn).apply(window, args);
13767             }else{
13768                 /** eval:var:value */
13769                 return /** eval:var:value */ eval(fn).call(window, value);
13770             }
13771         },
13772
13773        
13774         /**
13775          * safer version of Math.toFixed..??/
13776          * @param {Number/String} value The numeric value to format
13777          * @param {Number/String} value Decimal places 
13778          * @return {String} The formatted currency string
13779          */
13780         toFixed : function(v, n)
13781         {
13782             // why not use to fixed - precision is buggered???
13783             if (!n) {
13784                 return Math.round(v-0);
13785             }
13786             var fact = Math.pow(10,n+1);
13787             v = (Math.round((v-0)*fact))/fact;
13788             var z = (''+fact).substring(2);
13789             if (v == Math.floor(v)) {
13790                 return Math.floor(v) + '.' + z;
13791             }
13792             
13793             // now just padd decimals..
13794             var ps = String(v).split('.');
13795             var fd = (ps[1] + z);
13796             var r = fd.substring(0,n); 
13797             var rm = fd.substring(n); 
13798             if (rm < 5) {
13799                 return ps[0] + '.' + r;
13800             }
13801             r*=1; // turn it into a number;
13802             r++;
13803             if (String(r).length != n) {
13804                 ps[0]*=1;
13805                 ps[0]++;
13806                 r = String(r).substring(1); // chop the end off.
13807             }
13808             
13809             return ps[0] + '.' + r;
13810              
13811         },
13812         
13813         /**
13814          * Format a number as US currency
13815          * @param {Number/String} value The numeric value to format
13816          * @return {String} The formatted currency string
13817          */
13818         usMoney : function(v){
13819             return '$' + Roo.util.Format.number(v);
13820         },
13821         
13822         /**
13823          * Format a number
13824          * eventually this should probably emulate php's number_format
13825          * @param {Number/String} value The numeric value to format
13826          * @param {Number} decimals number of decimal places
13827          * @param {String} delimiter for thousands (default comma)
13828          * @return {String} The formatted currency string
13829          */
13830         number : function(v, decimals, thousandsDelimiter)
13831         {
13832             // multiply and round.
13833             decimals = typeof(decimals) == 'undefined' ? 2 : decimals;
13834             thousandsDelimiter = typeof(thousandsDelimiter) == 'undefined' ? ',' : thousandsDelimiter;
13835             
13836             var mul = Math.pow(10, decimals);
13837             var zero = String(mul).substring(1);
13838             v = (Math.round((v-0)*mul))/mul;
13839             
13840             // if it's '0' number.. then
13841             
13842             //v = (v == Math.floor(v)) ? v + "." + zero : ((v*10 == Math.floor(v*10)) ? v + "0" : v);
13843             v = String(v);
13844             var ps = v.split('.');
13845             var whole = ps[0];
13846             
13847             var r = /(\d+)(\d{3})/;
13848             // add comma's
13849             
13850             if(thousandsDelimiter.length != 0) {
13851                 whole = whole.replace(/\B(?=(\d{3})+(?!\d))/g, thousandsDelimiter );
13852             } 
13853             
13854             var sub = ps[1] ?
13855                     // has decimals..
13856                     (decimals ?  ('.'+ ps[1] + zero.substring(ps[1].length)) : '') :
13857                     // does not have decimals
13858                     (decimals ? ('.' + zero) : '');
13859             
13860             
13861             return whole + sub ;
13862         },
13863         
13864         /**
13865          * Parse a value into a formatted date using the specified format pattern.
13866          * @param {Mixed} value The value to format
13867          * @param {String} format (optional) Any valid date format string (defaults to 'm/d/Y')
13868          * @return {String} The formatted date string
13869          */
13870         date : function(v, format){
13871             if(!v){
13872                 return "";
13873             }
13874             if(!(v instanceof Date)){
13875                 v = new Date(Date.parse(v));
13876             }
13877             return v.dateFormat(format || Roo.util.Format.defaults.date);
13878         },
13879
13880         /**
13881          * Returns a date rendering function that can be reused to apply a date format multiple times efficiently
13882          * @param {String} format Any valid date format string
13883          * @return {Function} The date formatting function
13884          */
13885         dateRenderer : function(format){
13886             return function(v){
13887                 return Roo.util.Format.date(v, format);  
13888             };
13889         },
13890
13891         // private
13892         stripTagsRE : /<\/?[^>]+>/gi,
13893         
13894         /**
13895          * Strips all HTML tags
13896          * @param {Mixed} value The text from which to strip tags
13897          * @return {String} The stripped text
13898          */
13899         stripTags : function(v){
13900             return !v ? v : String(v).replace(this.stripTagsRE, "");
13901         }
13902     };
13903 }();
13904 Roo.util.Format.defaults = {
13905     date : 'd/M/Y'
13906 };/*
13907  * Based on:
13908  * Ext JS Library 1.1.1
13909  * Copyright(c) 2006-2007, Ext JS, LLC.
13910  *
13911  * Originally Released Under LGPL - original licence link has changed is not relivant.
13912  *
13913  * Fork - LGPL
13914  * <script type="text/javascript">
13915  */
13916
13917
13918  
13919
13920 /**
13921  * @class Roo.MasterTemplate
13922  * @extends Roo.Template
13923  * Provides a template that can have child templates. The syntax is:
13924 <pre><code>
13925 var t = new Roo.MasterTemplate(
13926         '&lt;select name="{name}"&gt;',
13927                 '&lt;tpl name="options"&gt;&lt;option value="{value:trim}"&gt;{text:ellipsis(10)}&lt;/option&gt;&lt;/tpl&gt;',
13928         '&lt;/select&gt;'
13929 );
13930 t.add('options', {value: 'foo', text: 'bar'});
13931 // or you can add multiple child elements in one shot
13932 t.addAll('options', [
13933     {value: 'foo', text: 'bar'},
13934     {value: 'foo2', text: 'bar2'},
13935     {value: 'foo3', text: 'bar3'}
13936 ]);
13937 // then append, applying the master template values
13938 t.append('my-form', {name: 'my-select'});
13939 </code></pre>
13940 * A name attribute for the child template is not required if you have only one child
13941 * template or you want to refer to them by index.
13942  */
13943 Roo.MasterTemplate = function(){
13944     Roo.MasterTemplate.superclass.constructor.apply(this, arguments);
13945     this.originalHtml = this.html;
13946     var st = {};
13947     var m, re = this.subTemplateRe;
13948     re.lastIndex = 0;
13949     var subIndex = 0;
13950     while(m = re.exec(this.html)){
13951         var name = m[1], content = m[2];
13952         st[subIndex] = {
13953             name: name,
13954             index: subIndex,
13955             buffer: [],
13956             tpl : new Roo.Template(content)
13957         };
13958         if(name){
13959             st[name] = st[subIndex];
13960         }
13961         st[subIndex].tpl.compile();
13962         st[subIndex].tpl.call = this.call.createDelegate(this);
13963         subIndex++;
13964     }
13965     this.subCount = subIndex;
13966     this.subs = st;
13967 };
13968 Roo.extend(Roo.MasterTemplate, Roo.Template, {
13969     /**
13970     * The regular expression used to match sub templates
13971     * @type RegExp
13972     * @property
13973     */
13974     subTemplateRe : /<tpl(?:\sname="([\w-]+)")?>((?:.|\n)*?)<\/tpl>/gi,
13975
13976     /**
13977      * Applies the passed values to a child template.
13978      * @param {String/Number} name (optional) The name or index of the child template
13979      * @param {Array/Object} values The values to be applied to the template
13980      * @return {MasterTemplate} this
13981      */
13982      add : function(name, values){
13983         if(arguments.length == 1){
13984             values = arguments[0];
13985             name = 0;
13986         }
13987         var s = this.subs[name];
13988         s.buffer[s.buffer.length] = s.tpl.apply(values);
13989         return this;
13990     },
13991
13992     /**
13993      * Applies all the passed values to a child template.
13994      * @param {String/Number} name (optional) The name or index of the child template
13995      * @param {Array} values The values to be applied to the template, this should be an array of objects.
13996      * @param {Boolean} reset (optional) True to reset the template first
13997      * @return {MasterTemplate} this
13998      */
13999     fill : function(name, values, reset){
14000         var a = arguments;
14001         if(a.length == 1 || (a.length == 2 && typeof a[1] == "boolean")){
14002             values = a[0];
14003             name = 0;
14004             reset = a[1];
14005         }
14006         if(reset){
14007             this.reset();
14008         }
14009         for(var i = 0, len = values.length; i < len; i++){
14010             this.add(name, values[i]);
14011         }
14012         return this;
14013     },
14014
14015     /**
14016      * Resets the template for reuse
14017      * @return {MasterTemplate} this
14018      */
14019      reset : function(){
14020         var s = this.subs;
14021         for(var i = 0; i < this.subCount; i++){
14022             s[i].buffer = [];
14023         }
14024         return this;
14025     },
14026
14027     applyTemplate : function(values){
14028         var s = this.subs;
14029         var replaceIndex = -1;
14030         this.html = this.originalHtml.replace(this.subTemplateRe, function(m, name){
14031             return s[++replaceIndex].buffer.join("");
14032         });
14033         return Roo.MasterTemplate.superclass.applyTemplate.call(this, values);
14034     },
14035
14036     apply : function(){
14037         return this.applyTemplate.apply(this, arguments);
14038     },
14039
14040     compile : function(){return this;}
14041 });
14042
14043 /**
14044  * Alias for fill().
14045  * @method
14046  */
14047 Roo.MasterTemplate.prototype.addAll = Roo.MasterTemplate.prototype.fill;
14048  /**
14049  * Creates a template from the passed element's value (display:none textarea, preferred) or innerHTML. e.g.
14050  * var tpl = Roo.MasterTemplate.from('element-id');
14051  * @param {String/HTMLElement} el
14052  * @param {Object} config
14053  * @static
14054  */
14055 Roo.MasterTemplate.from = function(el, config){
14056     el = Roo.getDom(el);
14057     return new Roo.MasterTemplate(el.value || el.innerHTML, config || '');
14058 };/*
14059  * Based on:
14060  * Ext JS Library 1.1.1
14061  * Copyright(c) 2006-2007, Ext JS, LLC.
14062  *
14063  * Originally Released Under LGPL - original licence link has changed is not relivant.
14064  *
14065  * Fork - LGPL
14066  * <script type="text/javascript">
14067  */
14068
14069  
14070 /**
14071  * @class Roo.util.CSS
14072  * Utility class for manipulating CSS rules
14073  * @singleton
14074  */
14075 Roo.util.CSS = function(){
14076         var rules = null;
14077         var doc = document;
14078
14079     var camelRe = /(-[a-z])/gi;
14080     var camelFn = function(m, a){ return a.charAt(1).toUpperCase(); };
14081
14082    return {
14083    /**
14084     * Very simple dynamic creation of stylesheets from a text blob of rules.  The text will wrapped in a style
14085     * tag and appended to the HEAD of the document.
14086     * @param {String|Object} cssText The text containing the css rules
14087     * @param {String} id An id to add to the stylesheet for later removal
14088     * @return {StyleSheet}
14089     */
14090     createStyleSheet : function(cssText, id){
14091         var ss;
14092         var head = doc.getElementsByTagName("head")[0];
14093         var nrules = doc.createElement("style");
14094         nrules.setAttribute("type", "text/css");
14095         if(id){
14096             nrules.setAttribute("id", id);
14097         }
14098         if (typeof(cssText) != 'string') {
14099             // support object maps..
14100             // not sure if this a good idea.. 
14101             // perhaps it should be merged with the general css handling
14102             // and handle js style props.
14103             var cssTextNew = [];
14104             for(var n in cssText) {
14105                 var citems = [];
14106                 for(var k in cssText[n]) {
14107                     citems.push( k + ' : ' +cssText[n][k] + ';' );
14108                 }
14109                 cssTextNew.push( n + ' { ' + citems.join(' ') + '} ');
14110                 
14111             }
14112             cssText = cssTextNew.join("\n");
14113             
14114         }
14115        
14116        
14117        if(Roo.isIE){
14118            head.appendChild(nrules);
14119            ss = nrules.styleSheet;
14120            ss.cssText = cssText;
14121        }else{
14122            try{
14123                 nrules.appendChild(doc.createTextNode(cssText));
14124            }catch(e){
14125                nrules.cssText = cssText; 
14126            }
14127            head.appendChild(nrules);
14128            ss = nrules.styleSheet ? nrules.styleSheet : (nrules.sheet || doc.styleSheets[doc.styleSheets.length-1]);
14129        }
14130        this.cacheStyleSheet(ss);
14131        return ss;
14132    },
14133
14134    /**
14135     * Removes a style or link tag by id
14136     * @param {String} id The id of the tag
14137     */
14138    removeStyleSheet : function(id){
14139        var existing = doc.getElementById(id);
14140        if(existing){
14141            existing.parentNode.removeChild(existing);
14142        }
14143    },
14144
14145    /**
14146     * Dynamically swaps an existing stylesheet reference for a new one
14147     * @param {String} id The id of an existing link tag to remove
14148     * @param {String} url The href of the new stylesheet to include
14149     */
14150    swapStyleSheet : function(id, url){
14151        this.removeStyleSheet(id);
14152        var ss = doc.createElement("link");
14153        ss.setAttribute("rel", "stylesheet");
14154        ss.setAttribute("type", "text/css");
14155        ss.setAttribute("id", id);
14156        ss.setAttribute("href", url);
14157        doc.getElementsByTagName("head")[0].appendChild(ss);
14158    },
14159    
14160    /**
14161     * Refresh the rule cache if you have dynamically added stylesheets
14162     * @return {Object} An object (hash) of rules indexed by selector
14163     */
14164    refreshCache : function(){
14165        return this.getRules(true);
14166    },
14167
14168    // private
14169    cacheStyleSheet : function(stylesheet){
14170        if(!rules){
14171            rules = {};
14172        }
14173        try{// try catch for cross domain access issue
14174            var ssRules = stylesheet.cssRules || stylesheet.rules;
14175            for(var j = ssRules.length-1; j >= 0; --j){
14176                rules[ssRules[j].selectorText] = ssRules[j];
14177            }
14178        }catch(e){}
14179    },
14180    
14181    /**
14182     * Gets all css rules for the document
14183     * @param {Boolean} refreshCache true to refresh the internal cache
14184     * @return {Object} An object (hash) of rules indexed by selector
14185     */
14186    getRules : function(refreshCache){
14187                 if(rules == null || refreshCache){
14188                         rules = {};
14189                         var ds = doc.styleSheets;
14190                         for(var i =0, len = ds.length; i < len; i++){
14191                             try{
14192                         this.cacheStyleSheet(ds[i]);
14193                     }catch(e){} 
14194                 }
14195                 }
14196                 return rules;
14197         },
14198         
14199         /**
14200     * Gets an an individual CSS rule by selector(s)
14201     * @param {String/Array} selector The CSS selector or an array of selectors to try. The first selector that is found is returned.
14202     * @param {Boolean} refreshCache true to refresh the internal cache if you have recently updated any rules or added styles dynamically
14203     * @return {CSSRule} The CSS rule or null if one is not found
14204     */
14205    getRule : function(selector, refreshCache){
14206                 var rs = this.getRules(refreshCache);
14207                 if(!(selector instanceof Array)){
14208                     return rs[selector];
14209                 }
14210                 for(var i = 0; i < selector.length; i++){
14211                         if(rs[selector[i]]){
14212                                 return rs[selector[i]];
14213                         }
14214                 }
14215                 return null;
14216         },
14217         
14218         
14219         /**
14220     * Updates a rule property
14221     * @param {String/Array} selector If it's an array it tries each selector until it finds one. Stops immediately once one is found.
14222     * @param {String} property The css property
14223     * @param {String} value The new value for the property
14224     * @return {Boolean} true If a rule was found and updated
14225     */
14226    updateRule : function(selector, property, value){
14227                 if(!(selector instanceof Array)){
14228                         var rule = this.getRule(selector);
14229                         if(rule){
14230                                 rule.style[property.replace(camelRe, camelFn)] = value;
14231                                 return true;
14232                         }
14233                 }else{
14234                         for(var i = 0; i < selector.length; i++){
14235                                 if(this.updateRule(selector[i], property, value)){
14236                                         return true;
14237                                 }
14238                         }
14239                 }
14240                 return false;
14241         }
14242    };   
14243 }();/*
14244  * Based on:
14245  * Ext JS Library 1.1.1
14246  * Copyright(c) 2006-2007, Ext JS, LLC.
14247  *
14248  * Originally Released Under LGPL - original licence link has changed is not relivant.
14249  *
14250  * Fork - LGPL
14251  * <script type="text/javascript">
14252  */
14253
14254  
14255
14256 /**
14257  * @class Roo.util.ClickRepeater
14258  * @extends Roo.util.Observable
14259  * 
14260  * A wrapper class which can be applied to any element. Fires a "click" event while the
14261  * mouse is pressed. The interval between firings may be specified in the config but
14262  * defaults to 10 milliseconds.
14263  * 
14264  * Optionally, a CSS class may be applied to the element during the time it is pressed.
14265  * 
14266  * @cfg {String/HTMLElement/Element} el The element to act as a button.
14267  * @cfg {Number} delay The initial delay before the repeating event begins firing.
14268  * Similar to an autorepeat key delay.
14269  * @cfg {Number} interval The interval between firings of the "click" event. Default 10 ms.
14270  * @cfg {String} pressClass A CSS class name to be applied to the element while pressed.
14271  * @cfg {Boolean} accelerate True if autorepeating should start slowly and accelerate.
14272  *           "interval" and "delay" are ignored. "immediate" is honored.
14273  * @cfg {Boolean} preventDefault True to prevent the default click event
14274  * @cfg {Boolean} stopDefault True to stop the default click event
14275  * 
14276  * @history
14277  *     2007-02-02 jvs Original code contributed by Nige "Animal" White
14278  *     2007-02-02 jvs Renamed to ClickRepeater
14279  *   2007-02-03 jvs Modifications for FF Mac and Safari 
14280  *
14281  *  @constructor
14282  * @param {String/HTMLElement/Element} el The element to listen on
14283  * @param {Object} config
14284  **/
14285 Roo.util.ClickRepeater = function(el, config)
14286 {
14287     this.el = Roo.get(el);
14288     this.el.unselectable();
14289
14290     Roo.apply(this, config);
14291
14292     this.addEvents({
14293     /**
14294      * @event mousedown
14295      * Fires when the mouse button is depressed.
14296      * @param {Roo.util.ClickRepeater} this
14297      */
14298         "mousedown" : true,
14299     /**
14300      * @event click
14301      * Fires on a specified interval during the time the element is pressed.
14302      * @param {Roo.util.ClickRepeater} this
14303      */
14304         "click" : true,
14305     /**
14306      * @event mouseup
14307      * Fires when the mouse key is released.
14308      * @param {Roo.util.ClickRepeater} this
14309      */
14310         "mouseup" : true
14311     });
14312
14313     this.el.on("mousedown", this.handleMouseDown, this);
14314     if(this.preventDefault || this.stopDefault){
14315         this.el.on("click", function(e){
14316             if(this.preventDefault){
14317                 e.preventDefault();
14318             }
14319             if(this.stopDefault){
14320                 e.stopEvent();
14321             }
14322         }, this);
14323     }
14324
14325     // allow inline handler
14326     if(this.handler){
14327         this.on("click", this.handler,  this.scope || this);
14328     }
14329
14330     Roo.util.ClickRepeater.superclass.constructor.call(this);
14331 };
14332
14333 Roo.extend(Roo.util.ClickRepeater, Roo.util.Observable, {
14334     interval : 20,
14335     delay: 250,
14336     preventDefault : true,
14337     stopDefault : false,
14338     timer : 0,
14339
14340     // private
14341     handleMouseDown : function(){
14342         clearTimeout(this.timer);
14343         this.el.blur();
14344         if(this.pressClass){
14345             this.el.addClass(this.pressClass);
14346         }
14347         this.mousedownTime = new Date();
14348
14349         Roo.get(document).on("mouseup", this.handleMouseUp, this);
14350         this.el.on("mouseout", this.handleMouseOut, this);
14351
14352         this.fireEvent("mousedown", this);
14353         this.fireEvent("click", this);
14354         
14355         this.timer = this.click.defer(this.delay || this.interval, this);
14356     },
14357
14358     // private
14359     click : function(){
14360         this.fireEvent("click", this);
14361         this.timer = this.click.defer(this.getInterval(), this);
14362     },
14363
14364     // private
14365     getInterval: function(){
14366         if(!this.accelerate){
14367             return this.interval;
14368         }
14369         var pressTime = this.mousedownTime.getElapsed();
14370         if(pressTime < 500){
14371             return 400;
14372         }else if(pressTime < 1700){
14373             return 320;
14374         }else if(pressTime < 2600){
14375             return 250;
14376         }else if(pressTime < 3500){
14377             return 180;
14378         }else if(pressTime < 4400){
14379             return 140;
14380         }else if(pressTime < 5300){
14381             return 80;
14382         }else if(pressTime < 6200){
14383             return 50;
14384         }else{
14385             return 10;
14386         }
14387     },
14388
14389     // private
14390     handleMouseOut : function(){
14391         clearTimeout(this.timer);
14392         if(this.pressClass){
14393             this.el.removeClass(this.pressClass);
14394         }
14395         this.el.on("mouseover", this.handleMouseReturn, this);
14396     },
14397
14398     // private
14399     handleMouseReturn : function(){
14400         this.el.un("mouseover", this.handleMouseReturn);
14401         if(this.pressClass){
14402             this.el.addClass(this.pressClass);
14403         }
14404         this.click();
14405     },
14406
14407     // private
14408     handleMouseUp : function(){
14409         clearTimeout(this.timer);
14410         this.el.un("mouseover", this.handleMouseReturn);
14411         this.el.un("mouseout", this.handleMouseOut);
14412         Roo.get(document).un("mouseup", this.handleMouseUp);
14413         this.el.removeClass(this.pressClass);
14414         this.fireEvent("mouseup", this);
14415     }
14416 });/*
14417  * Based on:
14418  * Ext JS Library 1.1.1
14419  * Copyright(c) 2006-2007, Ext JS, LLC.
14420  *
14421  * Originally Released Under LGPL - original licence link has changed is not relivant.
14422  *
14423  * Fork - LGPL
14424  * <script type="text/javascript">
14425  */
14426
14427  
14428 /**
14429  * @class Roo.KeyNav
14430  * <p>Provides a convenient wrapper for normalized keyboard navigation.  KeyNav allows you to bind
14431  * navigation keys to function calls that will get called when the keys are pressed, providing an easy
14432  * way to implement custom navigation schemes for any UI component.</p>
14433  * <p>The following are all of the possible keys that can be implemented: enter, left, right, up, down, tab, esc,
14434  * pageUp, pageDown, del, home, end.  Usage:</p>
14435  <pre><code>
14436 var nav = new Roo.KeyNav("my-element", {
14437     "left" : function(e){
14438         this.moveLeft(e.ctrlKey);
14439     },
14440     "right" : function(e){
14441         this.moveRight(e.ctrlKey);
14442     },
14443     "enter" : function(e){
14444         this.save();
14445     },
14446     scope : this
14447 });
14448 </code></pre>
14449  * @constructor
14450  * @param {String/HTMLElement/Roo.Element} el The element to bind to
14451  * @param {Object} config The config
14452  */
14453 Roo.KeyNav = function(el, config){
14454     this.el = Roo.get(el);
14455     Roo.apply(this, config);
14456     if(!this.disabled){
14457         this.disabled = true;
14458         this.enable();
14459     }
14460 };
14461
14462 Roo.KeyNav.prototype = {
14463     /**
14464      * @cfg {Boolean} disabled
14465      * True to disable this KeyNav instance (defaults to false)
14466      */
14467     disabled : false,
14468     /**
14469      * @cfg {String} defaultEventAction
14470      * The method to call on the {@link Roo.EventObject} after this KeyNav intercepts a key.  Valid values are
14471      * {@link Roo.EventObject#stopEvent}, {@link Roo.EventObject#preventDefault} and
14472      * {@link Roo.EventObject#stopPropagation} (defaults to 'stopEvent')
14473      */
14474     defaultEventAction: "stopEvent",
14475     /**
14476      * @cfg {Boolean} forceKeyDown
14477      * Handle the keydown event instead of keypress (defaults to false).  KeyNav automatically does this for IE since
14478      * IE does not propagate special keys on keypress, but setting this to true will force other browsers to also
14479      * handle keydown instead of keypress.
14480      */
14481     forceKeyDown : false,
14482
14483     // private
14484     prepareEvent : function(e){
14485         var k = e.getKey();
14486         var h = this.keyToHandler[k];
14487         //if(h && this[h]){
14488         //    e.stopPropagation();
14489         //}
14490         if(Roo.isSafari && h && k >= 37 && k <= 40){
14491             e.stopEvent();
14492         }
14493     },
14494
14495     // private
14496     relay : function(e){
14497         var k = e.getKey();
14498         var h = this.keyToHandler[k];
14499         if(h && this[h]){
14500             if(this.doRelay(e, this[h], h) !== true){
14501                 e[this.defaultEventAction]();
14502             }
14503         }
14504     },
14505
14506     // private
14507     doRelay : function(e, h, hname){
14508         return h.call(this.scope || this, e);
14509     },
14510
14511     // possible handlers
14512     enter : false,
14513     left : false,
14514     right : false,
14515     up : false,
14516     down : false,
14517     tab : false,
14518     esc : false,
14519     pageUp : false,
14520     pageDown : false,
14521     del : false,
14522     home : false,
14523     end : false,
14524
14525     // quick lookup hash
14526     keyToHandler : {
14527         37 : "left",
14528         39 : "right",
14529         38 : "up",
14530         40 : "down",
14531         33 : "pageUp",
14532         34 : "pageDown",
14533         46 : "del",
14534         36 : "home",
14535         35 : "end",
14536         13 : "enter",
14537         27 : "esc",
14538         9  : "tab"
14539     },
14540
14541         /**
14542          * Enable this KeyNav
14543          */
14544         enable: function(){
14545                 if(this.disabled){
14546             // ie won't do special keys on keypress, no one else will repeat keys with keydown
14547             // the EventObject will normalize Safari automatically
14548             if(this.forceKeyDown || Roo.isIE || Roo.isAir){
14549                 this.el.on("keydown", this.relay,  this);
14550             }else{
14551                 this.el.on("keydown", this.prepareEvent,  this);
14552                 this.el.on("keypress", this.relay,  this);
14553             }
14554                     this.disabled = false;
14555                 }
14556         },
14557
14558         /**
14559          * Disable this KeyNav
14560          */
14561         disable: function(){
14562                 if(!this.disabled){
14563                     if(this.forceKeyDown || Roo.isIE || Roo.isAir){
14564                 this.el.un("keydown", this.relay);
14565             }else{
14566                 this.el.un("keydown", this.prepareEvent);
14567                 this.el.un("keypress", this.relay);
14568             }
14569                     this.disabled = true;
14570                 }
14571         }
14572 };/*
14573  * Based on:
14574  * Ext JS Library 1.1.1
14575  * Copyright(c) 2006-2007, Ext JS, LLC.
14576  *
14577  * Originally Released Under LGPL - original licence link has changed is not relivant.
14578  *
14579  * Fork - LGPL
14580  * <script type="text/javascript">
14581  */
14582
14583  
14584 /**
14585  * @class Roo.KeyMap
14586  * Handles mapping keys to actions for an element. One key map can be used for multiple actions.
14587  * The constructor accepts the same config object as defined by {@link #addBinding}.
14588  * If you bind a callback function to a KeyMap, anytime the KeyMap handles an expected key
14589  * combination it will call the function with this signature (if the match is a multi-key
14590  * combination the callback will still be called only once): (String key, Roo.EventObject e)
14591  * A KeyMap can also handle a string representation of keys.<br />
14592  * Usage:
14593  <pre><code>
14594 // map one key by key code
14595 var map = new Roo.KeyMap("my-element", {
14596     key: 13, // or Roo.EventObject.ENTER
14597     fn: myHandler,
14598     scope: myObject
14599 });
14600
14601 // map multiple keys to one action by string
14602 var map = new Roo.KeyMap("my-element", {
14603     key: "a\r\n\t",
14604     fn: myHandler,
14605     scope: myObject
14606 });
14607
14608 // map multiple keys to multiple actions by strings and array of codes
14609 var map = new Roo.KeyMap("my-element", [
14610     {
14611         key: [10,13],
14612         fn: function(){ alert("Return was pressed"); }
14613     }, {
14614         key: "abc",
14615         fn: function(){ alert('a, b or c was pressed'); }
14616     }, {
14617         key: "\t",
14618         ctrl:true,
14619         shift:true,
14620         fn: function(){ alert('Control + shift + tab was pressed.'); }
14621     }
14622 ]);
14623 </code></pre>
14624  * <b>Note: A KeyMap starts enabled</b>
14625  * @constructor
14626  * @param {String/HTMLElement/Roo.Element} el The element to bind to
14627  * @param {Object} config The config (see {@link #addBinding})
14628  * @param {String} eventName (optional) The event to bind to (defaults to "keydown")
14629  */
14630 Roo.KeyMap = function(el, config, eventName){
14631     this.el  = Roo.get(el);
14632     this.eventName = eventName || "keydown";
14633     this.bindings = [];
14634     if(config){
14635         this.addBinding(config);
14636     }
14637     this.enable();
14638 };
14639
14640 Roo.KeyMap.prototype = {
14641     /**
14642      * True to stop the event from bubbling and prevent the default browser action if the
14643      * key was handled by the KeyMap (defaults to false)
14644      * @type Boolean
14645      */
14646     stopEvent : false,
14647
14648     /**
14649      * Add a new binding to this KeyMap. The following config object properties are supported:
14650      * <pre>
14651 Property    Type             Description
14652 ----------  ---------------  ----------------------------------------------------------------------
14653 key         String/Array     A single keycode or an array of keycodes to handle
14654 shift       Boolean          True to handle key only when shift is pressed (defaults to false)
14655 ctrl        Boolean          True to handle key only when ctrl is pressed (defaults to false)
14656 alt         Boolean          True to handle key only when alt is pressed (defaults to false)
14657 fn          Function         The function to call when KeyMap finds the expected key combination
14658 scope       Object           The scope of the callback function
14659 </pre>
14660      *
14661      * Usage:
14662      * <pre><code>
14663 // Create a KeyMap
14664 var map = new Roo.KeyMap(document, {
14665     key: Roo.EventObject.ENTER,
14666     fn: handleKey,
14667     scope: this
14668 });
14669
14670 //Add a new binding to the existing KeyMap later
14671 map.addBinding({
14672     key: 'abc',
14673     shift: true,
14674     fn: handleKey,
14675     scope: this
14676 });
14677 </code></pre>
14678      * @param {Object/Array} config A single KeyMap config or an array of configs
14679      */
14680         addBinding : function(config){
14681         if(config instanceof Array){
14682             for(var i = 0, len = config.length; i < len; i++){
14683                 this.addBinding(config[i]);
14684             }
14685             return;
14686         }
14687         var keyCode = config.key,
14688             shift = config.shift, 
14689             ctrl = config.ctrl, 
14690             alt = config.alt,
14691             fn = config.fn,
14692             scope = config.scope;
14693         if(typeof keyCode == "string"){
14694             var ks = [];
14695             var keyString = keyCode.toUpperCase();
14696             for(var j = 0, len = keyString.length; j < len; j++){
14697                 ks.push(keyString.charCodeAt(j));
14698             }
14699             keyCode = ks;
14700         }
14701         var keyArray = keyCode instanceof Array;
14702         var handler = function(e){
14703             if((!shift || e.shiftKey) && (!ctrl || e.ctrlKey) &&  (!alt || e.altKey)){
14704                 var k = e.getKey();
14705                 if(keyArray){
14706                     for(var i = 0, len = keyCode.length; i < len; i++){
14707                         if(keyCode[i] == k){
14708                           if(this.stopEvent){
14709                               e.stopEvent();
14710                           }
14711                           fn.call(scope || window, k, e);
14712                           return;
14713                         }
14714                     }
14715                 }else{
14716                     if(k == keyCode){
14717                         if(this.stopEvent){
14718                            e.stopEvent();
14719                         }
14720                         fn.call(scope || window, k, e);
14721                     }
14722                 }
14723             }
14724         };
14725         this.bindings.push(handler);  
14726         },
14727
14728     /**
14729      * Shorthand for adding a single key listener
14730      * @param {Number/Array/Object} key Either the numeric key code, array of key codes or an object with the
14731      * following options:
14732      * {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
14733      * @param {Function} fn The function to call
14734      * @param {Object} scope (optional) The scope of the function
14735      */
14736     on : function(key, fn, scope){
14737         var keyCode, shift, ctrl, alt;
14738         if(typeof key == "object" && !(key instanceof Array)){
14739             keyCode = key.key;
14740             shift = key.shift;
14741             ctrl = key.ctrl;
14742             alt = key.alt;
14743         }else{
14744             keyCode = key;
14745         }
14746         this.addBinding({
14747             key: keyCode,
14748             shift: shift,
14749             ctrl: ctrl,
14750             alt: alt,
14751             fn: fn,
14752             scope: scope
14753         })
14754     },
14755
14756     // private
14757     handleKeyDown : function(e){
14758             if(this.enabled){ //just in case
14759             var b = this.bindings;
14760             for(var i = 0, len = b.length; i < len; i++){
14761                 b[i].call(this, e);
14762             }
14763             }
14764         },
14765         
14766         /**
14767          * Returns true if this KeyMap is enabled
14768          * @return {Boolean} 
14769          */
14770         isEnabled : function(){
14771             return this.enabled;  
14772         },
14773         
14774         /**
14775          * Enables this KeyMap
14776          */
14777         enable: function(){
14778                 if(!this.enabled){
14779                     this.el.on(this.eventName, this.handleKeyDown, this);
14780                     this.enabled = true;
14781                 }
14782         },
14783
14784         /**
14785          * Disable this KeyMap
14786          */
14787         disable: function(){
14788                 if(this.enabled){
14789                     this.el.removeListener(this.eventName, this.handleKeyDown, this);
14790                     this.enabled = false;
14791                 }
14792         }
14793 };/*
14794  * Based on:
14795  * Ext JS Library 1.1.1
14796  * Copyright(c) 2006-2007, Ext JS, LLC.
14797  *
14798  * Originally Released Under LGPL - original licence link has changed is not relivant.
14799  *
14800  * Fork - LGPL
14801  * <script type="text/javascript">
14802  */
14803
14804  
14805 /**
14806  * @class Roo.util.TextMetrics
14807  * Provides precise pixel measurements for blocks of text so that you can determine exactly how high and
14808  * wide, in pixels, a given block of text will be.
14809  * @singleton
14810  */
14811 Roo.util.TextMetrics = function(){
14812     var shared;
14813     return {
14814         /**
14815          * Measures the size of the specified text
14816          * @param {String/HTMLElement} el The element, dom node or id from which to copy existing CSS styles
14817          * that can affect the size of the rendered text
14818          * @param {String} text The text to measure
14819          * @param {Number} fixedWidth (optional) If the text will be multiline, you have to set a fixed width
14820          * in order to accurately measure the text height
14821          * @return {Object} An object containing the text's size {width: (width), height: (height)}
14822          */
14823         measure : function(el, text, fixedWidth){
14824             if(!shared){
14825                 shared = Roo.util.TextMetrics.Instance(el, fixedWidth);
14826             }
14827             shared.bind(el);
14828             shared.setFixedWidth(fixedWidth || 'auto');
14829             return shared.getSize(text);
14830         },
14831
14832         /**
14833          * Return a unique TextMetrics instance that can be bound directly to an element and reused.  This reduces
14834          * the overhead of multiple calls to initialize the style properties on each measurement.
14835          * @param {String/HTMLElement} el The element, dom node or id that the instance will be bound to
14836          * @param {Number} fixedWidth (optional) If the text will be multiline, you have to set a fixed width
14837          * in order to accurately measure the text height
14838          * @return {Roo.util.TextMetrics.Instance} instance The new instance
14839          */
14840         createInstance : function(el, fixedWidth){
14841             return Roo.util.TextMetrics.Instance(el, fixedWidth);
14842         }
14843     };
14844 }();
14845
14846  
14847
14848 Roo.util.TextMetrics.Instance = function(bindTo, fixedWidth){
14849     var ml = new Roo.Element(document.createElement('div'));
14850     document.body.appendChild(ml.dom);
14851     ml.position('absolute');
14852     ml.setLeftTop(-1000, -1000);
14853     ml.hide();
14854
14855     if(fixedWidth){
14856         ml.setWidth(fixedWidth);
14857     }
14858      
14859     var instance = {
14860         /**
14861          * Returns the size of the specified text based on the internal element's style and width properties
14862          * @memberOf Roo.util.TextMetrics.Instance#
14863          * @param {String} text The text to measure
14864          * @return {Object} An object containing the text's size {width: (width), height: (height)}
14865          */
14866         getSize : function(text){
14867             ml.update(text);
14868             var s = ml.getSize();
14869             ml.update('');
14870             return s;
14871         },
14872
14873         /**
14874          * Binds this TextMetrics instance to an element from which to copy existing CSS styles
14875          * that can affect the size of the rendered text
14876          * @memberOf Roo.util.TextMetrics.Instance#
14877          * @param {String/HTMLElement} el The element, dom node or id
14878          */
14879         bind : function(el){
14880             ml.setStyle(
14881                 Roo.fly(el).getStyles('font-size','font-style', 'font-weight', 'font-family','line-height')
14882             );
14883         },
14884
14885         /**
14886          * Sets a fixed width on the internal measurement element.  If the text will be multiline, you have
14887          * to set a fixed width in order to accurately measure the text height.
14888          * @memberOf Roo.util.TextMetrics.Instance#
14889          * @param {Number} width The width to set on the element
14890          */
14891         setFixedWidth : function(width){
14892             ml.setWidth(width);
14893         },
14894
14895         /**
14896          * Returns the measured width of the specified text
14897          * @memberOf Roo.util.TextMetrics.Instance#
14898          * @param {String} text The text to measure
14899          * @return {Number} width The width in pixels
14900          */
14901         getWidth : function(text){
14902             ml.dom.style.width = 'auto';
14903             return this.getSize(text).width;
14904         },
14905
14906         /**
14907          * Returns the measured height of the specified text.  For multiline text, be sure to call
14908          * {@link #setFixedWidth} if necessary.
14909          * @memberOf Roo.util.TextMetrics.Instance#
14910          * @param {String} text The text to measure
14911          * @return {Number} height The height in pixels
14912          */
14913         getHeight : function(text){
14914             return this.getSize(text).height;
14915         }
14916     };
14917
14918     instance.bind(bindTo);
14919
14920     return instance;
14921 };
14922
14923 // backwards compat
14924 Roo.Element.measureText = Roo.util.TextMetrics.measure;/*
14925  * Based on:
14926  * Ext JS Library 1.1.1
14927  * Copyright(c) 2006-2007, Ext JS, LLC.
14928  *
14929  * Originally Released Under LGPL - original licence link has changed is not relivant.
14930  *
14931  * Fork - LGPL
14932  * <script type="text/javascript">
14933  */
14934
14935 /**
14936  * @class Roo.state.Provider
14937  * Abstract base class for state provider implementations. This class provides methods
14938  * for encoding and decoding <b>typed</b> variables including dates and defines the 
14939  * Provider interface.
14940  */
14941 Roo.state.Provider = function(){
14942     /**
14943      * @event statechange
14944      * Fires when a state change occurs.
14945      * @param {Provider} this This state provider
14946      * @param {String} key The state key which was changed
14947      * @param {String} value The encoded value for the state
14948      */
14949     this.addEvents({
14950         "statechange": true
14951     });
14952     this.state = {};
14953     Roo.state.Provider.superclass.constructor.call(this);
14954 };
14955 Roo.extend(Roo.state.Provider, Roo.util.Observable, {
14956     /**
14957      * Returns the current value for a key
14958      * @param {String} name The key name
14959      * @param {Mixed} defaultValue A default value to return if the key's value is not found
14960      * @return {Mixed} The state data
14961      */
14962     get : function(name, defaultValue){
14963         return typeof this.state[name] == "undefined" ?
14964             defaultValue : this.state[name];
14965     },
14966     
14967     /**
14968      * Clears a value from the state
14969      * @param {String} name The key name
14970      */
14971     clear : function(name){
14972         delete this.state[name];
14973         this.fireEvent("statechange", this, name, null);
14974     },
14975     
14976     /**
14977      * Sets the value for a key
14978      * @param {String} name The key name
14979      * @param {Mixed} value The value to set
14980      */
14981     set : function(name, value){
14982         this.state[name] = value;
14983         this.fireEvent("statechange", this, name, value);
14984     },
14985     
14986     /**
14987      * Decodes a string previously encoded with {@link #encodeValue}.
14988      * @param {String} value The value to decode
14989      * @return {Mixed} The decoded value
14990      */
14991     decodeValue : function(cookie){
14992         var re = /^(a|n|d|b|s|o)\:(.*)$/;
14993         var matches = re.exec(unescape(cookie));
14994         if(!matches || !matches[1]) {
14995             return; // non state cookie
14996         }
14997         var type = matches[1];
14998         var v = matches[2];
14999         switch(type){
15000             case "n":
15001                 return parseFloat(v);
15002             case "d":
15003                 return new Date(Date.parse(v));
15004             case "b":
15005                 return (v == "1");
15006             case "a":
15007                 var all = [];
15008                 var values = v.split("^");
15009                 for(var i = 0, len = values.length; i < len; i++){
15010                     all.push(this.decodeValue(values[i]));
15011                 }
15012                 return all;
15013            case "o":
15014                 var all = {};
15015                 var values = v.split("^");
15016                 for(var i = 0, len = values.length; i < len; i++){
15017                     var kv = values[i].split("=");
15018                     all[kv[0]] = this.decodeValue(kv[1]);
15019                 }
15020                 return all;
15021            default:
15022                 return v;
15023         }
15024     },
15025     
15026     /**
15027      * Encodes a value including type information.  Decode with {@link #decodeValue}.
15028      * @param {Mixed} value The value to encode
15029      * @return {String} The encoded value
15030      */
15031     encodeValue : function(v){
15032         var enc;
15033         if(typeof v == "number"){
15034             enc = "n:" + v;
15035         }else if(typeof v == "boolean"){
15036             enc = "b:" + (v ? "1" : "0");
15037         }else if(v instanceof Date){
15038             enc = "d:" + v.toGMTString();
15039         }else if(v instanceof Array){
15040             var flat = "";
15041             for(var i = 0, len = v.length; i < len; i++){
15042                 flat += this.encodeValue(v[i]);
15043                 if(i != len-1) {
15044                     flat += "^";
15045                 }
15046             }
15047             enc = "a:" + flat;
15048         }else if(typeof v == "object"){
15049             var flat = "";
15050             for(var key in v){
15051                 if(typeof v[key] != "function"){
15052                     flat += key + "=" + this.encodeValue(v[key]) + "^";
15053                 }
15054             }
15055             enc = "o:" + flat.substring(0, flat.length-1);
15056         }else{
15057             enc = "s:" + v;
15058         }
15059         return escape(enc);        
15060     }
15061 });
15062
15063 /*
15064  * Based on:
15065  * Ext JS Library 1.1.1
15066  * Copyright(c) 2006-2007, Ext JS, LLC.
15067  *
15068  * Originally Released Under LGPL - original licence link has changed is not relivant.
15069  *
15070  * Fork - LGPL
15071  * <script type="text/javascript">
15072  */
15073 /**
15074  * @class Roo.state.Manager
15075  * This is the global state manager. By default all components that are "state aware" check this class
15076  * for state information if you don't pass them a custom state provider. In order for this class
15077  * to be useful, it must be initialized with a provider when your application initializes.
15078  <pre><code>
15079 // in your initialization function
15080 init : function(){
15081    Roo.state.Manager.setProvider(new Roo.state.CookieProvider());
15082    ...
15083    // supposed you have a {@link Roo.BorderLayout}
15084    var layout = new Roo.BorderLayout(...);
15085    layout.restoreState();
15086    // or a {Roo.BasicDialog}
15087    var dialog = new Roo.BasicDialog(...);
15088    dialog.restoreState();
15089  </code></pre>
15090  * @singleton
15091  */
15092 Roo.state.Manager = function(){
15093     var provider = new Roo.state.Provider();
15094     
15095     return {
15096         /**
15097          * Configures the default state provider for your application
15098          * @param {Provider} stateProvider The state provider to set
15099          */
15100         setProvider : function(stateProvider){
15101             provider = stateProvider;
15102         },
15103         
15104         /**
15105          * Returns the current value for a key
15106          * @param {String} name The key name
15107          * @param {Mixed} defaultValue The default value to return if the key lookup does not match
15108          * @return {Mixed} The state data
15109          */
15110         get : function(key, defaultValue){
15111             return provider.get(key, defaultValue);
15112         },
15113         
15114         /**
15115          * Sets the value for a key
15116          * @param {String} name The key name
15117          * @param {Mixed} value The state data
15118          */
15119          set : function(key, value){
15120             provider.set(key, value);
15121         },
15122         
15123         /**
15124          * Clears a value from the state
15125          * @param {String} name The key name
15126          */
15127         clear : function(key){
15128             provider.clear(key);
15129         },
15130         
15131         /**
15132          * Gets the currently configured state provider
15133          * @return {Provider} The state provider
15134          */
15135         getProvider : function(){
15136             return provider;
15137         }
15138     };
15139 }();
15140 /*
15141  * Based on:
15142  * Ext JS Library 1.1.1
15143  * Copyright(c) 2006-2007, Ext JS, LLC.
15144  *
15145  * Originally Released Under LGPL - original licence link has changed is not relivant.
15146  *
15147  * Fork - LGPL
15148  * <script type="text/javascript">
15149  */
15150 /**
15151  * @class Roo.state.CookieProvider
15152  * @extends Roo.state.Provider
15153  * The default Provider implementation which saves state via cookies.
15154  * <br />Usage:
15155  <pre><code>
15156    var cp = new Roo.state.CookieProvider({
15157        path: "/cgi-bin/",
15158        expires: new Date(new Date().getTime()+(1000*60*60*24*30)); //30 days
15159        domain: "roojs.com"
15160    })
15161    Roo.state.Manager.setProvider(cp);
15162  </code></pre>
15163  * @cfg {String} path The path for which the cookie is active (defaults to root '/' which makes it active for all pages in the site)
15164  * @cfg {Date} expires The cookie expiration date (defaults to 7 days from now)
15165  * @cfg {String} domain The domain to save the cookie for.  Note that you cannot specify a different domain than
15166  * your page is on, but you can specify a sub-domain, or simply the domain itself like 'roojs.com' to include
15167  * all sub-domains if you need to access cookies across different sub-domains (defaults to null which uses the same
15168  * domain the page is running on including the 'www' like 'www.roojs.com')
15169  * @cfg {Boolean} secure True if the site is using SSL (defaults to false)
15170  * @constructor
15171  * Create a new CookieProvider
15172  * @param {Object} config The configuration object
15173  */
15174 Roo.state.CookieProvider = function(config){
15175     Roo.state.CookieProvider.superclass.constructor.call(this);
15176     this.path = "/";
15177     this.expires = new Date(new Date().getTime()+(1000*60*60*24*7)); //7 days
15178     this.domain = null;
15179     this.secure = false;
15180     Roo.apply(this, config);
15181     this.state = this.readCookies();
15182 };
15183
15184 Roo.extend(Roo.state.CookieProvider, Roo.state.Provider, {
15185     // private
15186     set : function(name, value){
15187         if(typeof value == "undefined" || value === null){
15188             this.clear(name);
15189             return;
15190         }
15191         this.setCookie(name, value);
15192         Roo.state.CookieProvider.superclass.set.call(this, name, value);
15193     },
15194
15195     // private
15196     clear : function(name){
15197         this.clearCookie(name);
15198         Roo.state.CookieProvider.superclass.clear.call(this, name);
15199     },
15200
15201     // private
15202     readCookies : function(){
15203         var cookies = {};
15204         var c = document.cookie + ";";
15205         var re = /\s?(.*?)=(.*?);/g;
15206         var matches;
15207         while((matches = re.exec(c)) != null){
15208             var name = matches[1];
15209             var value = matches[2];
15210             if(name && name.substring(0,3) == "ys-"){
15211                 cookies[name.substr(3)] = this.decodeValue(value);
15212             }
15213         }
15214         return cookies;
15215     },
15216
15217     // private
15218     setCookie : function(name, value){
15219         document.cookie = "ys-"+ name + "=" + this.encodeValue(value) +
15220            ((this.expires == null) ? "" : ("; expires=" + this.expires.toGMTString())) +
15221            ((this.path == null) ? "" : ("; path=" + this.path)) +
15222            ((this.domain == null) ? "" : ("; domain=" + this.domain)) +
15223            ((this.secure == true) ? "; secure" : "");
15224     },
15225
15226     // private
15227     clearCookie : function(name){
15228         document.cookie = "ys-" + name + "=null; expires=Thu, 01-Jan-70 00:00:01 GMT" +
15229            ((this.path == null) ? "" : ("; path=" + this.path)) +
15230            ((this.domain == null) ? "" : ("; domain=" + this.domain)) +
15231            ((this.secure == true) ? "; secure" : "");
15232     }
15233 });/*
15234  * Based on:
15235  * Ext JS Library 1.1.1
15236  * Copyright(c) 2006-2007, Ext JS, LLC.
15237  *
15238  * Originally Released Under LGPL - original licence link has changed is not relivant.
15239  *
15240  * Fork - LGPL
15241  * <script type="text/javascript">
15242  */
15243  
15244
15245 /**
15246  * @class Roo.ComponentMgr
15247  * Provides a common registry of all components on a page so that they can be easily accessed by component id (see {@link Roo.getCmp}).
15248  * @singleton
15249  */
15250 Roo.ComponentMgr = function(){
15251     var all = new Roo.util.MixedCollection();
15252
15253     return {
15254         /**
15255          * Registers a component.
15256          * @param {Roo.Component} c The component
15257          */
15258         register : function(c){
15259             all.add(c);
15260         },
15261
15262         /**
15263          * Unregisters a component.
15264          * @param {Roo.Component} c The component
15265          */
15266         unregister : function(c){
15267             all.remove(c);
15268         },
15269
15270         /**
15271          * Returns a component by id
15272          * @param {String} id The component id
15273          */
15274         get : function(id){
15275             return all.get(id);
15276         },
15277
15278         /**
15279          * Registers a function that will be called when a specified component is added to ComponentMgr
15280          * @param {String} id The component id
15281          * @param {Funtction} fn The callback function
15282          * @param {Object} scope The scope of the callback
15283          */
15284         onAvailable : function(id, fn, scope){
15285             all.on("add", function(index, o){
15286                 if(o.id == id){
15287                     fn.call(scope || o, o);
15288                     all.un("add", fn, scope);
15289                 }
15290             });
15291         }
15292     };
15293 }();/*
15294  * Based on:
15295  * Ext JS Library 1.1.1
15296  * Copyright(c) 2006-2007, Ext JS, LLC.
15297  *
15298  * Originally Released Under LGPL - original licence link has changed is not relivant.
15299  *
15300  * Fork - LGPL
15301  * <script type="text/javascript">
15302  */
15303  
15304 /**
15305  * @class Roo.Component
15306  * @extends Roo.util.Observable
15307  * Base class for all major Roo components.  All subclasses of Component can automatically participate in the standard
15308  * Roo component lifecycle of creation, rendering and destruction.  They also have automatic support for basic hide/show
15309  * and enable/disable behavior.  Component allows any subclass to be lazy-rendered into any {@link Roo.Container} and
15310  * to be automatically registered with the {@link Roo.ComponentMgr} so that it can be referenced at any time via {@link Roo.getCmp}.
15311  * All visual components (widgets) that require rendering into a layout should subclass Component.
15312  * @constructor
15313  * @param {Roo.Element/String/Object} config The configuration options.  If an element is passed, it is set as the internal
15314  * element and its id used as the component id.  If a string is passed, it is assumed to be the id of an existing element
15315  * and is used as the component id.  Otherwise, it is assumed to be a standard config object and is applied to the component.
15316  */
15317 Roo.Component = function(config){
15318     config = config || {};
15319     if(config.tagName || config.dom || typeof config == "string"){ // element object
15320         config = {el: config, id: config.id || config};
15321     }
15322     this.initialConfig = config;
15323
15324     Roo.apply(this, config);
15325     this.addEvents({
15326         /**
15327          * @event disable
15328          * Fires after the component is disabled.
15329              * @param {Roo.Component} this
15330              */
15331         disable : true,
15332         /**
15333          * @event enable
15334          * Fires after the component is enabled.
15335              * @param {Roo.Component} this
15336              */
15337         enable : true,
15338         /**
15339          * @event beforeshow
15340          * Fires before the component is shown.  Return false to stop the show.
15341              * @param {Roo.Component} this
15342              */
15343         beforeshow : true,
15344         /**
15345          * @event show
15346          * Fires after the component is shown.
15347              * @param {Roo.Component} this
15348              */
15349         show : true,
15350         /**
15351          * @event beforehide
15352          * Fires before the component is hidden. Return false to stop the hide.
15353              * @param {Roo.Component} this
15354              */
15355         beforehide : true,
15356         /**
15357          * @event hide
15358          * Fires after the component is hidden.
15359              * @param {Roo.Component} this
15360              */
15361         hide : true,
15362         /**
15363          * @event beforerender
15364          * Fires before the component is rendered. Return false to stop the render.
15365              * @param {Roo.Component} this
15366              */
15367         beforerender : true,
15368         /**
15369          * @event render
15370          * Fires after the component is rendered.
15371              * @param {Roo.Component} this
15372              */
15373         render : true,
15374         /**
15375          * @event beforedestroy
15376          * Fires before the component is destroyed. Return false to stop the destroy.
15377              * @param {Roo.Component} this
15378              */
15379         beforedestroy : true,
15380         /**
15381          * @event destroy
15382          * Fires after the component is destroyed.
15383              * @param {Roo.Component} this
15384              */
15385         destroy : true
15386     });
15387     if(!this.id){
15388         this.id = "roo-comp-" + (++Roo.Component.AUTO_ID);
15389     }
15390     Roo.ComponentMgr.register(this);
15391     Roo.Component.superclass.constructor.call(this);
15392     this.initComponent();
15393     if(this.renderTo){ // not supported by all components yet. use at your own risk!
15394         this.render(this.renderTo);
15395         delete this.renderTo;
15396     }
15397 };
15398
15399 /** @private */
15400 Roo.Component.AUTO_ID = 1000;
15401
15402 Roo.extend(Roo.Component, Roo.util.Observable, {
15403     /**
15404      * @scope Roo.Component.prototype
15405      * @type {Boolean}
15406      * true if this component is hidden. Read-only.
15407      */
15408     hidden : false,
15409     /**
15410      * @type {Boolean}
15411      * true if this component is disabled. Read-only.
15412      */
15413     disabled : false,
15414     /**
15415      * @type {Boolean}
15416      * true if this component has been rendered. Read-only.
15417      */
15418     rendered : false,
15419     
15420     /** @cfg {String} disableClass
15421      * CSS class added to the component when it is disabled (defaults to "x-item-disabled").
15422      */
15423     disabledClass : "x-item-disabled",
15424         /** @cfg {Boolean} allowDomMove
15425          * Whether the component can move the Dom node when rendering (defaults to true).
15426          */
15427     allowDomMove : true,
15428     /** @cfg {String} hideMode (display|visibility)
15429      * How this component should hidden. Supported values are
15430      * "visibility" (css visibility), "offsets" (negative offset position) and
15431      * "display" (css display) - defaults to "display".
15432      */
15433     hideMode: 'display',
15434
15435     /** @private */
15436     ctype : "Roo.Component",
15437
15438     /**
15439      * @cfg {String} actionMode 
15440      * which property holds the element that used for  hide() / show() / disable() / enable()
15441      * default is 'el' 
15442      */
15443     actionMode : "el",
15444
15445     /** @private */
15446     getActionEl : function(){
15447         return this[this.actionMode];
15448     },
15449
15450     initComponent : Roo.emptyFn,
15451     /**
15452      * If this is a lazy rendering component, render it to its container element.
15453      * @param {String/HTMLElement/Element} container (optional) The element this component should be rendered into. If it is being applied to existing markup, this should be left off.
15454      */
15455     render : function(container, position){
15456         
15457         if(this.rendered){
15458             return this;
15459         }
15460         
15461         if(this.fireEvent("beforerender", this) === false){
15462             return false;
15463         }
15464         
15465         if(!container && this.el){
15466             this.el = Roo.get(this.el);
15467             container = this.el.dom.parentNode;
15468             this.allowDomMove = false;
15469         }
15470         this.container = Roo.get(container);
15471         this.rendered = true;
15472         if(position !== undefined){
15473             if(typeof position == 'number'){
15474                 position = this.container.dom.childNodes[position];
15475             }else{
15476                 position = Roo.getDom(position);
15477             }
15478         }
15479         this.onRender(this.container, position || null);
15480         if(this.cls){
15481             this.el.addClass(this.cls);
15482             delete this.cls;
15483         }
15484         if(this.style){
15485             this.el.applyStyles(this.style);
15486             delete this.style;
15487         }
15488         this.fireEvent("render", this);
15489         this.afterRender(this.container);
15490         if(this.hidden){
15491             this.hide();
15492         }
15493         if(this.disabled){
15494             this.disable();
15495         }
15496
15497         return this;
15498         
15499     },
15500
15501     /** @private */
15502     // default function is not really useful
15503     onRender : function(ct, position){
15504         if(this.el){
15505             this.el = Roo.get(this.el);
15506             if(this.allowDomMove !== false){
15507                 ct.dom.insertBefore(this.el.dom, position);
15508             }
15509         }
15510     },
15511
15512     /** @private */
15513     getAutoCreate : function(){
15514         var cfg = typeof this.autoCreate == "object" ?
15515                       this.autoCreate : Roo.apply({}, this.defaultAutoCreate);
15516         if(this.id && !cfg.id){
15517             cfg.id = this.id;
15518         }
15519         return cfg;
15520     },
15521
15522     /** @private */
15523     afterRender : Roo.emptyFn,
15524
15525     /**
15526      * Destroys this component by purging any event listeners, removing the component's element from the DOM,
15527      * removing the component from its {@link Roo.Container} (if applicable) and unregistering it from {@link Roo.ComponentMgr}.
15528      */
15529     destroy : function(){
15530         if(this.fireEvent("beforedestroy", this) !== false){
15531             this.purgeListeners();
15532             this.beforeDestroy();
15533             if(this.rendered){
15534                 this.el.removeAllListeners();
15535                 this.el.remove();
15536                 if(this.actionMode == "container"){
15537                     this.container.remove();
15538                 }
15539             }
15540             this.onDestroy();
15541             Roo.ComponentMgr.unregister(this);
15542             this.fireEvent("destroy", this);
15543         }
15544     },
15545
15546         /** @private */
15547     beforeDestroy : function(){
15548
15549     },
15550
15551         /** @private */
15552         onDestroy : function(){
15553
15554     },
15555
15556     /**
15557      * Returns the underlying {@link Roo.Element}.
15558      * @return {Roo.Element} The element
15559      */
15560     getEl : function(){
15561         return this.el;
15562     },
15563
15564     /**
15565      * Returns the id of this component.
15566      * @return {String}
15567      */
15568     getId : function(){
15569         return this.id;
15570     },
15571
15572     /**
15573      * Try to focus this component.
15574      * @param {Boolean} selectText True to also select the text in this component (if applicable)
15575      * @return {Roo.Component} this
15576      */
15577     focus : function(selectText){
15578         if(this.rendered){
15579             this.el.focus();
15580             if(selectText === true){
15581                 this.el.dom.select();
15582             }
15583         }
15584         return this;
15585     },
15586
15587     /** @private */
15588     blur : function(){
15589         if(this.rendered){
15590             this.el.blur();
15591         }
15592         return this;
15593     },
15594
15595     /**
15596      * Disable this component.
15597      * @return {Roo.Component} this
15598      */
15599     disable : function(){
15600         if(this.rendered){
15601             this.onDisable();
15602         }
15603         this.disabled = true;
15604         this.fireEvent("disable", this);
15605         return this;
15606     },
15607
15608         // private
15609     onDisable : function(){
15610         this.getActionEl().addClass(this.disabledClass);
15611         this.el.dom.disabled = true;
15612     },
15613
15614     /**
15615      * Enable this component.
15616      * @return {Roo.Component} this
15617      */
15618     enable : function(){
15619         if(this.rendered){
15620             this.onEnable();
15621         }
15622         this.disabled = false;
15623         this.fireEvent("enable", this);
15624         return this;
15625     },
15626
15627         // private
15628     onEnable : function(){
15629         this.getActionEl().removeClass(this.disabledClass);
15630         this.el.dom.disabled = false;
15631     },
15632
15633     /**
15634      * Convenience function for setting disabled/enabled by boolean.
15635      * @param {Boolean} disabled
15636      */
15637     setDisabled : function(disabled){
15638         this[disabled ? "disable" : "enable"]();
15639     },
15640
15641     /**
15642      * Show this component.
15643      * @return {Roo.Component} this
15644      */
15645     show: function(){
15646         if(this.fireEvent("beforeshow", this) !== false){
15647             this.hidden = false;
15648             if(this.rendered){
15649                 this.onShow();
15650             }
15651             this.fireEvent("show", this);
15652         }
15653         return this;
15654     },
15655
15656     // private
15657     onShow : function(){
15658         var ae = this.getActionEl();
15659         if(this.hideMode == 'visibility'){
15660             ae.dom.style.visibility = "visible";
15661         }else if(this.hideMode == 'offsets'){
15662             ae.removeClass('x-hidden');
15663         }else{
15664             ae.dom.style.display = "";
15665         }
15666     },
15667
15668     /**
15669      * Hide this component.
15670      * @return {Roo.Component} this
15671      */
15672     hide: function(){
15673         if(this.fireEvent("beforehide", this) !== false){
15674             this.hidden = true;
15675             if(this.rendered){
15676                 this.onHide();
15677             }
15678             this.fireEvent("hide", this);
15679         }
15680         return this;
15681     },
15682
15683     // private
15684     onHide : function(){
15685         var ae = this.getActionEl();
15686         if(this.hideMode == 'visibility'){
15687             ae.dom.style.visibility = "hidden";
15688         }else if(this.hideMode == 'offsets'){
15689             ae.addClass('x-hidden');
15690         }else{
15691             ae.dom.style.display = "none";
15692         }
15693     },
15694
15695     /**
15696      * Convenience function to hide or show this component by boolean.
15697      * @param {Boolean} visible True to show, false to hide
15698      * @return {Roo.Component} this
15699      */
15700     setVisible: function(visible){
15701         if(visible) {
15702             this.show();
15703         }else{
15704             this.hide();
15705         }
15706         return this;
15707     },
15708
15709     /**
15710      * Returns true if this component is visible.
15711      */
15712     isVisible : function(){
15713         return this.getActionEl().isVisible();
15714     },
15715
15716     cloneConfig : function(overrides){
15717         overrides = overrides || {};
15718         var id = overrides.id || Roo.id();
15719         var cfg = Roo.applyIf(overrides, this.initialConfig);
15720         cfg.id = id; // prevent dup id
15721         return new this.constructor(cfg);
15722     }
15723 });/*
15724  * Based on:
15725  * Ext JS Library 1.1.1
15726  * Copyright(c) 2006-2007, Ext JS, LLC.
15727  *
15728  * Originally Released Under LGPL - original licence link has changed is not relivant.
15729  *
15730  * Fork - LGPL
15731  * <script type="text/javascript">
15732  */
15733
15734 /**
15735  * @class Roo.BoxComponent
15736  * @extends Roo.Component
15737  * Base class for any visual {@link Roo.Component} that uses a box container.  BoxComponent provides automatic box
15738  * model adjustments for sizing and positioning and will work correctly withnin the Component rendering model.  All
15739  * container classes should subclass BoxComponent so that they will work consistently when nested within other Ext
15740  * layout containers.
15741  * @constructor
15742  * @param {Roo.Element/String/Object} config The configuration options.
15743  */
15744 Roo.BoxComponent = function(config){
15745     Roo.Component.call(this, config);
15746     this.addEvents({
15747         /**
15748          * @event resize
15749          * Fires after the component is resized.
15750              * @param {Roo.Component} this
15751              * @param {Number} adjWidth The box-adjusted width that was set
15752              * @param {Number} adjHeight The box-adjusted height that was set
15753              * @param {Number} rawWidth The width that was originally specified
15754              * @param {Number} rawHeight The height that was originally specified
15755              */
15756         resize : true,
15757         /**
15758          * @event move
15759          * Fires after the component is moved.
15760              * @param {Roo.Component} this
15761              * @param {Number} x The new x position
15762              * @param {Number} y The new y position
15763              */
15764         move : true
15765     });
15766 };
15767
15768 Roo.extend(Roo.BoxComponent, Roo.Component, {
15769     // private, set in afterRender to signify that the component has been rendered
15770     boxReady : false,
15771     // private, used to defer height settings to subclasses
15772     deferHeight: false,
15773     /** @cfg {Number} width
15774      * width (optional) size of component
15775      */
15776      /** @cfg {Number} height
15777      * height (optional) size of component
15778      */
15779      
15780     /**
15781      * Sets the width and height of the component.  This method fires the resize event.  This method can accept
15782      * either width and height as separate numeric arguments, or you can pass a size object like {width:10, height:20}.
15783      * @param {Number/Object} width The new width to set, or a size object in the format {width, height}
15784      * @param {Number} height The new height to set (not required if a size object is passed as the first arg)
15785      * @return {Roo.BoxComponent} this
15786      */
15787     setSize : function(w, h){
15788         // support for standard size objects
15789         if(typeof w == 'object'){
15790             h = w.height;
15791             w = w.width;
15792         }
15793         // not rendered
15794         if(!this.boxReady){
15795             this.width = w;
15796             this.height = h;
15797             return this;
15798         }
15799
15800         // prevent recalcs when not needed
15801         if(this.lastSize && this.lastSize.width == w && this.lastSize.height == h){
15802             return this;
15803         }
15804         this.lastSize = {width: w, height: h};
15805
15806         var adj = this.adjustSize(w, h);
15807         var aw = adj.width, ah = adj.height;
15808         if(aw !== undefined || ah !== undefined){ // this code is nasty but performs better with floaters
15809             var rz = this.getResizeEl();
15810             if(!this.deferHeight && aw !== undefined && ah !== undefined){
15811                 rz.setSize(aw, ah);
15812             }else if(!this.deferHeight && ah !== undefined){
15813                 rz.setHeight(ah);
15814             }else if(aw !== undefined){
15815                 rz.setWidth(aw);
15816             }
15817             this.onResize(aw, ah, w, h);
15818             this.fireEvent('resize', this, aw, ah, w, h);
15819         }
15820         return this;
15821     },
15822
15823     /**
15824      * Gets the current size of the component's underlying element.
15825      * @return {Object} An object containing the element's size {width: (element width), height: (element height)}
15826      */
15827     getSize : function(){
15828         return this.el.getSize();
15829     },
15830
15831     /**
15832      * Gets the current XY position of the component's underlying element.
15833      * @param {Boolean} local (optional) If true the element's left and top are returned instead of page XY (defaults to false)
15834      * @return {Array} The XY position of the element (e.g., [100, 200])
15835      */
15836     getPosition : function(local){
15837         if(local === true){
15838             return [this.el.getLeft(true), this.el.getTop(true)];
15839         }
15840         return this.xy || this.el.getXY();
15841     },
15842
15843     /**
15844      * Gets the current box measurements of the component's underlying element.
15845      * @param {Boolean} local (optional) If true the element's left and top are returned instead of page XY (defaults to false)
15846      * @returns {Object} box An object in the format {x, y, width, height}
15847      */
15848     getBox : function(local){
15849         var s = this.el.getSize();
15850         if(local){
15851             s.x = this.el.getLeft(true);
15852             s.y = this.el.getTop(true);
15853         }else{
15854             var xy = this.xy || this.el.getXY();
15855             s.x = xy[0];
15856             s.y = xy[1];
15857         }
15858         return s;
15859     },
15860
15861     /**
15862      * Sets the current box measurements of the component's underlying element.
15863      * @param {Object} box An object in the format {x, y, width, height}
15864      * @returns {Roo.BoxComponent} this
15865      */
15866     updateBox : function(box){
15867         this.setSize(box.width, box.height);
15868         this.setPagePosition(box.x, box.y);
15869         return this;
15870     },
15871
15872     // protected
15873     getResizeEl : function(){
15874         return this.resizeEl || this.el;
15875     },
15876
15877     // protected
15878     getPositionEl : function(){
15879         return this.positionEl || this.el;
15880     },
15881
15882     /**
15883      * Sets the left and top of the component.  To set the page XY position instead, use {@link #setPagePosition}.
15884      * This method fires the move event.
15885      * @param {Number} left The new left
15886      * @param {Number} top The new top
15887      * @returns {Roo.BoxComponent} this
15888      */
15889     setPosition : function(x, y){
15890         this.x = x;
15891         this.y = y;
15892         if(!this.boxReady){
15893             return this;
15894         }
15895         var adj = this.adjustPosition(x, y);
15896         var ax = adj.x, ay = adj.y;
15897
15898         var el = this.getPositionEl();
15899         if(ax !== undefined || ay !== undefined){
15900             if(ax !== undefined && ay !== undefined){
15901                 el.setLeftTop(ax, ay);
15902             }else if(ax !== undefined){
15903                 el.setLeft(ax);
15904             }else if(ay !== undefined){
15905                 el.setTop(ay);
15906             }
15907             this.onPosition(ax, ay);
15908             this.fireEvent('move', this, ax, ay);
15909         }
15910         return this;
15911     },
15912
15913     /**
15914      * Sets the page XY position of the component.  To set the left and top instead, use {@link #setPosition}.
15915      * This method fires the move event.
15916      * @param {Number} x The new x position
15917      * @param {Number} y The new y position
15918      * @returns {Roo.BoxComponent} this
15919      */
15920     setPagePosition : function(x, y){
15921         this.pageX = x;
15922         this.pageY = y;
15923         if(!this.boxReady){
15924             return;
15925         }
15926         if(x === undefined || y === undefined){ // cannot translate undefined points
15927             return;
15928         }
15929         var p = this.el.translatePoints(x, y);
15930         this.setPosition(p.left, p.top);
15931         return this;
15932     },
15933
15934     // private
15935     onRender : function(ct, position){
15936         Roo.BoxComponent.superclass.onRender.call(this, ct, position);
15937         if(this.resizeEl){
15938             this.resizeEl = Roo.get(this.resizeEl);
15939         }
15940         if(this.positionEl){
15941             this.positionEl = Roo.get(this.positionEl);
15942         }
15943     },
15944
15945     // private
15946     afterRender : function(){
15947         Roo.BoxComponent.superclass.afterRender.call(this);
15948         this.boxReady = true;
15949         this.setSize(this.width, this.height);
15950         if(this.x || this.y){
15951             this.setPosition(this.x, this.y);
15952         }
15953         if(this.pageX || this.pageY){
15954             this.setPagePosition(this.pageX, this.pageY);
15955         }
15956     },
15957
15958     /**
15959      * Force the component's size to recalculate based on the underlying element's current height and width.
15960      * @returns {Roo.BoxComponent} this
15961      */
15962     syncSize : function(){
15963         delete this.lastSize;
15964         this.setSize(this.el.getWidth(), this.el.getHeight());
15965         return this;
15966     },
15967
15968     /**
15969      * Called after the component is resized, this method is empty by default but can be implemented by any
15970      * subclass that needs to perform custom logic after a resize occurs.
15971      * @param {Number} adjWidth The box-adjusted width that was set
15972      * @param {Number} adjHeight The box-adjusted height that was set
15973      * @param {Number} rawWidth The width that was originally specified
15974      * @param {Number} rawHeight The height that was originally specified
15975      */
15976     onResize : function(adjWidth, adjHeight, rawWidth, rawHeight){
15977
15978     },
15979
15980     /**
15981      * Called after the component is moved, this method is empty by default but can be implemented by any
15982      * subclass that needs to perform custom logic after a move occurs.
15983      * @param {Number} x The new x position
15984      * @param {Number} y The new y position
15985      */
15986     onPosition : function(x, y){
15987
15988     },
15989
15990     // private
15991     adjustSize : function(w, h){
15992         if(this.autoWidth){
15993             w = 'auto';
15994         }
15995         if(this.autoHeight){
15996             h = 'auto';
15997         }
15998         return {width : w, height: h};
15999     },
16000
16001     // private
16002     adjustPosition : function(x, y){
16003         return {x : x, y: y};
16004     }
16005 });/*
16006  * Original code for Roojs - LGPL
16007  * <script type="text/javascript">
16008  */
16009  
16010 /**
16011  * @class Roo.XComponent
16012  * A delayed Element creator...
16013  * Or a way to group chunks of interface together.
16014  * technically this is a wrapper around a tree of Roo elements (which defines a 'module'),
16015  *  used in conjunction with XComponent.build() it will create an instance of each element,
16016  *  then call addxtype() to build the User interface.
16017  * 
16018  * Mypart.xyx = new Roo.XComponent({
16019
16020     parent : 'Mypart.xyz', // empty == document.element.!!
16021     order : '001',
16022     name : 'xxxx'
16023     region : 'xxxx'
16024     disabled : function() {} 
16025      
16026     tree : function() { // return an tree of xtype declared components
16027         var MODULE = this;
16028         return 
16029         {
16030             xtype : 'NestedLayoutPanel',
16031             // technicall
16032         }
16033      ]
16034  *})
16035  *
16036  *
16037  * It can be used to build a big heiracy, with parent etc.
16038  * or you can just use this to render a single compoent to a dom element
16039  * MYPART.render(Roo.Element | String(id) | dom_element )
16040  *
16041  *
16042  * Usage patterns.
16043  *
16044  * Classic Roo
16045  *
16046  * Roo is designed primarily as a single page application, so the UI build for a standard interface will
16047  * expect a single 'TOP' level module normally indicated by the 'parent' of the XComponent definition being defined as false.
16048  *
16049  * Each sub module is expected to have a parent pointing to the class name of it's parent module.
16050  *
16051  * When the top level is false, a 'Roo.BorderLayout' is created and the element is flagged as 'topModule'
16052  * - if mulitple topModules exist, the last one is defined as the top module.
16053  *
16054  * Embeded Roo
16055  * 
16056  * When the top level or multiple modules are to embedded into a existing HTML page,
16057  * the parent element can container '#id' of the element where the module will be drawn.
16058  *
16059  * Bootstrap Roo
16060  *
16061  * Unlike classic Roo, the bootstrap tends not to be used as a single page.
16062  * it relies more on a include mechanism, where sub modules are included into an outer page.
16063  * This is normally managed by the builder tools using Roo.apply( options, Included.Sub.Module )
16064  * 
16065  * Bootstrap Roo Included elements
16066  *
16067  * Our builder application needs the ability to preview these sub compoennts. They will normally have parent=false set,
16068  * hence confusing the component builder as it thinks there are multiple top level elements. 
16069  *
16070  * String Over-ride & Translations
16071  *
16072  * Our builder application writes all the strings as _strings and _named_strings. This is to enable the translation of elements,
16073  * and also the 'overlaying of string values - needed when different versions of the same application with different text content
16074  * are needed. @see Roo.XComponent.overlayString  
16075  * 
16076  * 
16077  * 
16078  * @extends Roo.util.Observable
16079  * @constructor
16080  * @param cfg {Object} configuration of component
16081  * 
16082  */
16083 Roo.XComponent = function(cfg) {
16084     Roo.apply(this, cfg);
16085     this.addEvents({ 
16086         /**
16087              * @event built
16088              * Fires when this the componnt is built
16089              * @param {Roo.XComponent} c the component
16090              */
16091         'built' : true
16092         
16093     });
16094     this.region = this.region || 'center'; // default..
16095     Roo.XComponent.register(this);
16096     this.modules = false;
16097     this.el = false; // where the layout goes..
16098     
16099     
16100 }
16101 Roo.extend(Roo.XComponent, Roo.util.Observable, {
16102     /**
16103      * @property el
16104      * The created element (with Roo.factory())
16105      * @type {Roo.Layout}
16106      */
16107     el  : false,
16108     
16109     /**
16110      * @property el
16111      * for BC  - use el in new code
16112      * @type {Roo.Layout}
16113      */
16114     panel : false,
16115     
16116     /**
16117      * @property layout
16118      * for BC  - use el in new code
16119      * @type {Roo.Layout}
16120      */
16121     layout : false,
16122     
16123      /**
16124      * @cfg {Function|boolean} disabled
16125      * If this module is disabled by some rule, return true from the funtion
16126      */
16127     disabled : false,
16128     
16129     /**
16130      * @cfg {String} parent 
16131      * Name of parent element which it get xtype added to..
16132      */
16133     parent: false,
16134     
16135     /**
16136      * @cfg {String} order
16137      * Used to set the order in which elements are created (usefull for multiple tabs)
16138      */
16139     
16140     order : false,
16141     /**
16142      * @cfg {String} name
16143      * String to display while loading.
16144      */
16145     name : false,
16146     /**
16147      * @cfg {String} region
16148      * Region to render component to (defaults to center)
16149      */
16150     region : 'center',
16151     
16152     /**
16153      * @cfg {Array} items
16154      * A single item array - the first element is the root of the tree..
16155      * It's done this way to stay compatible with the Xtype system...
16156      */
16157     items : false,
16158     
16159     /**
16160      * @property _tree
16161      * The method that retuns the tree of parts that make up this compoennt 
16162      * @type {function}
16163      */
16164     _tree  : false,
16165     
16166      /**
16167      * render
16168      * render element to dom or tree
16169      * @param {Roo.Element|String|DomElement} optional render to if parent is not set.
16170      */
16171     
16172     render : function(el)
16173     {
16174         
16175         el = el || false;
16176         var hp = this.parent ? 1 : 0;
16177         Roo.debug &&  Roo.log(this);
16178         
16179         var tree = this._tree ? this._tree() : this.tree();
16180
16181         
16182         if (!el && typeof(this.parent) == 'string' && this.parent.substring(0,1) == '#') {
16183             // if parent is a '#.....' string, then let's use that..
16184             var ename = this.parent.substr(1);
16185             this.parent = false;
16186             Roo.debug && Roo.log(ename);
16187             switch (ename) {
16188                 case 'bootstrap-body':
16189                     if (typeof(tree.el) != 'undefined' && tree.el == document.body)  {
16190                         // this is the BorderLayout standard?
16191                        this.parent = { el : true };
16192                        break;
16193                     }
16194                     if (["Nest", "Content", "Grid", "Tree"].indexOf(tree.xtype)  > -1)  {
16195                         // need to insert stuff...
16196                         this.parent =  {
16197                              el : new Roo.bootstrap.layout.Border({
16198                                  el : document.body, 
16199                      
16200                                  center: {
16201                                     titlebar: false,
16202                                     autoScroll:false,
16203                                     closeOnTab: true,
16204                                     tabPosition: 'top',
16205                                       //resizeTabs: true,
16206                                     alwaysShowTabs: true,
16207                                     hideTabs: false
16208                                      //minTabWidth: 140
16209                                  }
16210                              })
16211                         
16212                          };
16213                          break;
16214                     }
16215                          
16216                     if (typeof(Roo.bootstrap.Body) != 'undefined' ) {
16217                         this.parent = { el :  new  Roo.bootstrap.Body() };
16218                         Roo.debug && Roo.log("setting el to doc body");
16219                          
16220                     } else {
16221                         throw "Container is bootstrap body, but Roo.bootstrap.Body is not defined";
16222                     }
16223                     break;
16224                 case 'bootstrap':
16225                     this.parent = { el : true};
16226                     // fall through
16227                 default:
16228                     el = Roo.get(ename);
16229                     if (typeof(Roo.bootstrap) != 'undefined' && tree['|xns'] == 'Roo.bootstrap') {
16230                         this.parent = { el : true};
16231                     }
16232                     
16233                     break;
16234             }
16235                 
16236             
16237             if (!el && !this.parent) {
16238                 Roo.debug && Roo.log("Warning - element can not be found :#" + ename );
16239                 return;
16240             }
16241         }
16242         
16243         Roo.debug && Roo.log("EL:");
16244         Roo.debug && Roo.log(el);
16245         Roo.debug && Roo.log("this.parent.el:");
16246         Roo.debug && Roo.log(this.parent.el);
16247         
16248
16249         // altertive root elements ??? - we need a better way to indicate these.
16250         var is_alt = Roo.XComponent.is_alt ||
16251                     (typeof(tree.el) != 'undefined' && tree.el == document.body) ||
16252                     (typeof(Roo.bootstrap) != 'undefined' && tree.xns == Roo.bootstrap) ||
16253                     (typeof(Roo.mailer) != 'undefined' && tree.xns == Roo.mailer) ;
16254         
16255         
16256         
16257         if (!this.parent && is_alt) {
16258             //el = Roo.get(document.body);
16259             this.parent = { el : true };
16260         }
16261             
16262             
16263         
16264         if (!this.parent) {
16265             
16266             Roo.debug && Roo.log("no parent - creating one");
16267             
16268             el = el ? Roo.get(el) : false;      
16269             
16270             if (typeof(Roo.BorderLayout) == 'undefined' ) {
16271                 
16272                 this.parent =  {
16273                     el : new Roo.bootstrap.layout.Border({
16274                         el: el || document.body,
16275                     
16276                         center: {
16277                             titlebar: false,
16278                             autoScroll:false,
16279                             closeOnTab: true,
16280                             tabPosition: 'top',
16281                              //resizeTabs: true,
16282                             alwaysShowTabs: false,
16283                             hideTabs: true,
16284                             minTabWidth: 140,
16285                             overflow: 'visible'
16286                          }
16287                      })
16288                 };
16289             } else {
16290             
16291                 // it's a top level one..
16292                 this.parent =  {
16293                     el : new Roo.BorderLayout(el || document.body, {
16294                         center: {
16295                             titlebar: false,
16296                             autoScroll:false,
16297                             closeOnTab: true,
16298                             tabPosition: 'top',
16299                              //resizeTabs: true,
16300                             alwaysShowTabs: el && hp? false :  true,
16301                             hideTabs: el || !hp ? true :  false,
16302                             minTabWidth: 140
16303                          }
16304                     })
16305                 };
16306             }
16307         }
16308         
16309         if (!this.parent.el) {
16310                 // probably an old style ctor, which has been disabled.
16311                 return;
16312
16313         }
16314                 // The 'tree' method is  '_tree now' 
16315             
16316         tree.region = tree.region || this.region;
16317         var is_body = false;
16318         if (this.parent.el === true) {
16319             // bootstrap... - body..
16320             if (el) {
16321                 tree.el = el;
16322             }
16323             this.parent.el = Roo.factory(tree);
16324             is_body = true;
16325         }
16326         
16327         this.el = this.parent.el.addxtype(tree, undefined, is_body);
16328         this.fireEvent('built', this);
16329         
16330         this.panel = this.el;
16331         this.layout = this.panel.layout;
16332         this.parentLayout = this.parent.layout  || false;  
16333          
16334     }
16335     
16336 });
16337
16338 Roo.apply(Roo.XComponent, {
16339     /**
16340      * @property  hideProgress
16341      * true to disable the building progress bar.. usefull on single page renders.
16342      * @type Boolean
16343      */
16344     hideProgress : false,
16345     /**
16346      * @property  buildCompleted
16347      * True when the builder has completed building the interface.
16348      * @type Boolean
16349      */
16350     buildCompleted : false,
16351      
16352     /**
16353      * @property  topModule
16354      * the upper most module - uses document.element as it's constructor.
16355      * @type Object
16356      */
16357      
16358     topModule  : false,
16359       
16360     /**
16361      * @property  modules
16362      * array of modules to be created by registration system.
16363      * @type {Array} of Roo.XComponent
16364      */
16365     
16366     modules : [],
16367     /**
16368      * @property  elmodules
16369      * array of modules to be created by which use #ID 
16370      * @type {Array} of Roo.XComponent
16371      */
16372      
16373     elmodules : [],
16374
16375      /**
16376      * @property  is_alt
16377      * Is an alternative Root - normally used by bootstrap or other systems,
16378      *    where the top element in the tree can wrap 'body' 
16379      * @type {boolean}  (default false)
16380      */
16381      
16382     is_alt : false,
16383     /**
16384      * @property  build_from_html
16385      * Build elements from html - used by bootstrap HTML stuff 
16386      *    - this is cleared after build is completed
16387      * @type {boolean}    (default false)
16388      */
16389      
16390     build_from_html : false,
16391     /**
16392      * Register components to be built later.
16393      *
16394      * This solves the following issues
16395      * - Building is not done on page load, but after an authentication process has occured.
16396      * - Interface elements are registered on page load
16397      * - Parent Interface elements may not be loaded before child, so this handles that..
16398      * 
16399      *
16400      * example:
16401      * 
16402      * MyApp.register({
16403           order : '000001',
16404           module : 'Pman.Tab.projectMgr',
16405           region : 'center',
16406           parent : 'Pman.layout',
16407           disabled : false,  // or use a function..
16408         })
16409      
16410      * * @param {Object} details about module
16411      */
16412     register : function(obj) {
16413                 
16414         Roo.XComponent.event.fireEvent('register', obj);
16415         switch(typeof(obj.disabled) ) {
16416                 
16417             case 'undefined':
16418                 break;
16419             
16420             case 'function':
16421                 if ( obj.disabled() ) {
16422                         return;
16423                 }
16424                 break;
16425             
16426             default:
16427                 if (obj.disabled || obj.region == '#disabled') {
16428                         return;
16429                 }
16430                 break;
16431         }
16432                 
16433         this.modules.push(obj);
16434          
16435     },
16436     /**
16437      * convert a string to an object..
16438      * eg. 'AAA.BBB' -> finds AAA.BBB
16439
16440      */
16441     
16442     toObject : function(str)
16443     {
16444         if (!str || typeof(str) == 'object') {
16445             return str;
16446         }
16447         if (str.substring(0,1) == '#') {
16448             return str;
16449         }
16450
16451         var ar = str.split('.');
16452         var rt, o;
16453         rt = ar.shift();
16454             /** eval:var:o */
16455         try {
16456             eval('if (typeof ' + rt + ' == "undefined"){ o = false;} o = ' + rt + ';');
16457         } catch (e) {
16458             throw "Module not found : " + str;
16459         }
16460         
16461         if (o === false) {
16462             throw "Module not found : " + str;
16463         }
16464         Roo.each(ar, function(e) {
16465             if (typeof(o[e]) == 'undefined') {
16466                 throw "Module not found : " + str;
16467             }
16468             o = o[e];
16469         });
16470         
16471         return o;
16472         
16473     },
16474     
16475     
16476     /**
16477      * move modules into their correct place in the tree..
16478      * 
16479      */
16480     preBuild : function ()
16481     {
16482         var _t = this;
16483         Roo.each(this.modules , function (obj)
16484         {
16485             Roo.XComponent.event.fireEvent('beforebuild', obj);
16486             
16487             var opar = obj.parent;
16488             try { 
16489                 obj.parent = this.toObject(opar);
16490             } catch(e) {
16491                 Roo.debug && Roo.log("parent:toObject failed: " + e.toString());
16492                 return;
16493             }
16494             
16495             if (!obj.parent) {
16496                 Roo.debug && Roo.log("GOT top level module");
16497                 Roo.debug && Roo.log(obj);
16498                 obj.modules = new Roo.util.MixedCollection(false, 
16499                     function(o) { return o.order + '' }
16500                 );
16501                 this.topModule = obj;
16502                 return;
16503             }
16504                         // parent is a string (usually a dom element name..)
16505             if (typeof(obj.parent) == 'string') {
16506                 this.elmodules.push(obj);
16507                 return;
16508             }
16509             if (obj.parent.constructor != Roo.XComponent) {
16510                 Roo.debug && Roo.log("Warning : Object Parent is not instance of XComponent:" + obj.name)
16511             }
16512             if (!obj.parent.modules) {
16513                 obj.parent.modules = new Roo.util.MixedCollection(false, 
16514                     function(o) { return o.order + '' }
16515                 );
16516             }
16517             if (obj.parent.disabled) {
16518                 obj.disabled = true;
16519             }
16520             obj.parent.modules.add(obj);
16521         }, this);
16522     },
16523     
16524      /**
16525      * make a list of modules to build.
16526      * @return {Array} list of modules. 
16527      */ 
16528     
16529     buildOrder : function()
16530     {
16531         var _this = this;
16532         var cmp = function(a,b) {   
16533             return String(a).toUpperCase() > String(b).toUpperCase() ? 1 : -1;
16534         };
16535         if ((!this.topModule || !this.topModule.modules) && !this.elmodules.length) {
16536             throw "No top level modules to build";
16537         }
16538         
16539         // make a flat list in order of modules to build.
16540         var mods = this.topModule ? [ this.topModule ] : [];
16541                 
16542         
16543         // elmodules (is a list of DOM based modules )
16544         Roo.each(this.elmodules, function(e) {
16545             mods.push(e);
16546             if (!this.topModule &&
16547                 typeof(e.parent) == 'string' &&
16548                 e.parent.substring(0,1) == '#' &&
16549                 Roo.get(e.parent.substr(1))
16550                ) {
16551                 
16552                 _this.topModule = e;
16553             }
16554             
16555         });
16556
16557         
16558         // add modules to their parents..
16559         var addMod = function(m) {
16560             Roo.debug && Roo.log("build Order: add: " + m.name);
16561                 
16562             mods.push(m);
16563             if (m.modules && !m.disabled) {
16564                 Roo.debug && Roo.log("build Order: " + m.modules.length + " child modules");
16565                 m.modules.keySort('ASC',  cmp );
16566                 Roo.debug && Roo.log("build Order: " + m.modules.length + " child modules (after sort)");
16567     
16568                 m.modules.each(addMod);
16569             } else {
16570                 Roo.debug && Roo.log("build Order: no child modules");
16571             }
16572             // not sure if this is used any more..
16573             if (m.finalize) {
16574                 m.finalize.name = m.name + " (clean up) ";
16575                 mods.push(m.finalize);
16576             }
16577             
16578         }
16579         if (this.topModule && this.topModule.modules) { 
16580             this.topModule.modules.keySort('ASC',  cmp );
16581             this.topModule.modules.each(addMod);
16582         } 
16583         return mods;
16584     },
16585     
16586      /**
16587      * Build the registered modules.
16588      * @param {Object} parent element.
16589      * @param {Function} optional method to call after module has been added.
16590      * 
16591      */ 
16592    
16593     build : function(opts) 
16594     {
16595         
16596         if (typeof(opts) != 'undefined') {
16597             Roo.apply(this,opts);
16598         }
16599         
16600         this.preBuild();
16601         var mods = this.buildOrder();
16602       
16603         //this.allmods = mods;
16604         //Roo.debug && Roo.log(mods);
16605         //return;
16606         if (!mods.length) { // should not happen
16607             throw "NO modules!!!";
16608         }
16609         
16610         
16611         var msg = "Building Interface...";
16612         // flash it up as modal - so we store the mask!?
16613         if (!this.hideProgress && Roo.MessageBox) {
16614             Roo.MessageBox.show({ title: 'loading' });
16615             Roo.MessageBox.show({
16616                title: "Please wait...",
16617                msg: msg,
16618                width:450,
16619                progress:true,
16620                buttons : false,
16621                closable:false,
16622                modal: false
16623               
16624             });
16625         }
16626         var total = mods.length;
16627         
16628         var _this = this;
16629         var progressRun = function() {
16630             if (!mods.length) {
16631                 Roo.debug && Roo.log('hide?');
16632                 if (!this.hideProgress && Roo.MessageBox) {
16633                     Roo.MessageBox.hide();
16634                 }
16635                 Roo.XComponent.build_from_html = false; // reset, so dialogs will be build from javascript
16636                 
16637                 Roo.XComponent.event.fireEvent('buildcomplete', _this.topModule);
16638                 
16639                 // THE END...
16640                 return false;   
16641             }
16642             
16643             var m = mods.shift();
16644             
16645             
16646             Roo.debug && Roo.log(m);
16647             // not sure if this is supported any more.. - modules that are are just function
16648             if (typeof(m) == 'function') { 
16649                 m.call(this);
16650                 return progressRun.defer(10, _this);
16651             } 
16652             
16653             
16654             msg = "Building Interface " + (total  - mods.length) + 
16655                     " of " + total + 
16656                     (m.name ? (' - ' + m.name) : '');
16657                         Roo.debug && Roo.log(msg);
16658             if (!_this.hideProgress &&  Roo.MessageBox) { 
16659                 Roo.MessageBox.updateProgress(  (total  - mods.length)/total, msg  );
16660             }
16661             
16662          
16663             // is the module disabled?
16664             var disabled = (typeof(m.disabled) == 'function') ?
16665                 m.disabled.call(m.module.disabled) : m.disabled;    
16666             
16667             
16668             if (disabled) {
16669                 return progressRun(); // we do not update the display!
16670             }
16671             
16672             // now build 
16673             
16674                         
16675                         
16676             m.render();
16677             // it's 10 on top level, and 1 on others??? why...
16678             return progressRun.defer(10, _this);
16679              
16680         }
16681         progressRun.defer(1, _this);
16682      
16683         
16684         
16685     },
16686     /**
16687      * Overlay a set of modified strings onto a component
16688      * This is dependant on our builder exporting the strings and 'named strings' elements.
16689      * 
16690      * @param {Object} element to overlay on - eg. Pman.Dialog.Login
16691      * @param {Object} associative array of 'named' string and it's new value.
16692      * 
16693      */
16694         overlayStrings : function( component, strings )
16695     {
16696         if (typeof(component['_named_strings']) == 'undefined') {
16697             throw "ERROR: component does not have _named_strings";
16698         }
16699         for ( var k in strings ) {
16700             var md = typeof(component['_named_strings'][k]) == 'undefined' ? false : component['_named_strings'][k];
16701             if (md !== false) {
16702                 component['_strings'][md] = strings[k];
16703             } else {
16704                 Roo.log('could not find named string: ' + k + ' in');
16705                 Roo.log(component);
16706             }
16707             
16708         }
16709         
16710     },
16711     
16712         
16713         /**
16714          * Event Object.
16715          *
16716          *
16717          */
16718         event: false, 
16719     /**
16720          * wrapper for event.on - aliased later..  
16721          * Typically use to register a event handler for register:
16722          *
16723          * eg. Roo.XComponent.on('register', function(comp) { comp.disable = true } );
16724          *
16725          */
16726     on : false
16727    
16728     
16729     
16730 });
16731
16732 Roo.XComponent.event = new Roo.util.Observable({
16733                 events : { 
16734                         /**
16735                          * @event register
16736                          * Fires when an Component is registered,
16737                          * set the disable property on the Component to stop registration.
16738                          * @param {Roo.XComponent} c the component being registerd.
16739                          * 
16740                          */
16741                         'register' : true,
16742             /**
16743                          * @event beforebuild
16744                          * Fires before each Component is built
16745                          * can be used to apply permissions.
16746                          * @param {Roo.XComponent} c the component being registerd.
16747                          * 
16748                          */
16749                         'beforebuild' : true,
16750                         /**
16751                          * @event buildcomplete
16752                          * Fires on the top level element when all elements have been built
16753                          * @param {Roo.XComponent} the top level component.
16754                          */
16755                         'buildcomplete' : true
16756                         
16757                 }
16758 });
16759
16760 Roo.XComponent.on = Roo.XComponent.event.on.createDelegate(Roo.XComponent.event); 
16761  //
16762  /**
16763  * marked - a markdown parser
16764  * Copyright (c) 2011-2014, Christopher Jeffrey. (MIT Licensed)
16765  * https://github.com/chjj/marked
16766  */
16767
16768
16769 /**
16770  *
16771  * Roo.Markdown - is a very crude wrapper around marked..
16772  *
16773  * usage:
16774  * 
16775  * alert( Roo.Markdown.toHtml("Markdown *rocks*.") );
16776  * 
16777  * Note: move the sample code to the bottom of this
16778  * file before uncommenting it.
16779  *
16780  */
16781
16782 Roo.Markdown = {};
16783 Roo.Markdown.toHtml = function(text) {
16784     
16785     var c = new Roo.Markdown.marked.setOptions({
16786             renderer: new Roo.Markdown.marked.Renderer(),
16787             gfm: true,
16788             tables: true,
16789             breaks: false,
16790             pedantic: false,
16791             sanitize: false,
16792             smartLists: true,
16793             smartypants: false
16794           });
16795     // A FEW HACKS!!?
16796     
16797     text = text.replace(/\\\n/g,' ');
16798     return Roo.Markdown.marked(text);
16799 };
16800 //
16801 // converter
16802 //
16803 // Wraps all "globals" so that the only thing
16804 // exposed is makeHtml().
16805 //
16806 (function() {
16807     
16808      /**
16809          * eval:var:escape
16810          * eval:var:unescape
16811          * eval:var:replace
16812          */
16813       
16814     /**
16815      * Helpers
16816      */
16817     
16818     var escape = function (html, encode) {
16819       return html
16820         .replace(!encode ? /&(?!#?\w+;)/g : /&/g, '&amp;')
16821         .replace(/</g, '&lt;')
16822         .replace(/>/g, '&gt;')
16823         .replace(/"/g, '&quot;')
16824         .replace(/'/g, '&#39;');
16825     }
16826     
16827     var unescape = function (html) {
16828         // explicitly match decimal, hex, and named HTML entities 
16829       return html.replace(/&(#(?:\d+)|(?:#x[0-9A-Fa-f]+)|(?:\w+));?/g, function(_, n) {
16830         n = n.toLowerCase();
16831         if (n === 'colon') { return ':'; }
16832         if (n.charAt(0) === '#') {
16833           return n.charAt(1) === 'x'
16834             ? String.fromCharCode(parseInt(n.substring(2), 16))
16835             : String.fromCharCode(+n.substring(1));
16836         }
16837         return '';
16838       });
16839     }
16840     
16841     var replace = function (regex, opt) {
16842       regex = regex.source;
16843       opt = opt || '';
16844       return function self(name, val) {
16845         if (!name) { return new RegExp(regex, opt); }
16846         val = val.source || val;
16847         val = val.replace(/(^|[^\[])\^/g, '$1');
16848         regex = regex.replace(name, val);
16849         return self;
16850       };
16851     }
16852
16853
16854          /**
16855          * eval:var:noop
16856     */
16857     var noop = function () {}
16858     noop.exec = noop;
16859     
16860          /**
16861          * eval:var:merge
16862     */
16863     var merge = function (obj) {
16864       var i = 1
16865         , target
16866         , key;
16867     
16868       for (; i < arguments.length; i++) {
16869         target = arguments[i];
16870         for (key in target) {
16871           if (Object.prototype.hasOwnProperty.call(target, key)) {
16872             obj[key] = target[key];
16873           }
16874         }
16875       }
16876     
16877       return obj;
16878     }
16879     
16880     
16881     /**
16882      * Block-Level Grammar
16883      */
16884     
16885     
16886     
16887     
16888     var block = {
16889       newline: /^\n+/,
16890       code: /^( {4}[^\n]+\n*)+/,
16891       fences: noop,
16892       hr: /^( *[-*_]){3,} *(?:\n+|$)/,
16893       heading: /^ *(#{1,6}) *([^\n]+?) *#* *(?:\n+|$)/,
16894       nptable: noop,
16895       lheading: /^([^\n]+)\n *(=|-){2,} *(?:\n+|$)/,
16896       blockquote: /^( *>[^\n]+(\n(?!def)[^\n]+)*\n*)+/,
16897       list: /^( *)(bull) [\s\S]+?(?:hr|def|\n{2,}(?! )(?!\1bull )\n*|\s*$)/,
16898       html: /^ *(?:comment *(?:\n|\s*$)|closed *(?:\n{2,}|\s*$)|closing *(?:\n{2,}|\s*$))/,
16899       def: /^ *\[([^\]]+)\]: *<?([^\s>]+)>?(?: +["(]([^\n]+)[")])? *(?:\n+|$)/,
16900       table: noop,
16901       paragraph: /^((?:[^\n]+\n?(?!hr|heading|lheading|blockquote|tag|def))+)\n*/,
16902       text: /^[^\n]+/
16903     };
16904     
16905     block.bullet = /(?:[*+-]|\d+\.)/;
16906     block.item = /^( *)(bull) [^\n]*(?:\n(?!\1bull )[^\n]*)*/;
16907     block.item = replace(block.item, 'gm')
16908       (/bull/g, block.bullet)
16909       ();
16910     
16911     block.list = replace(block.list)
16912       (/bull/g, block.bullet)
16913       ('hr', '\\n+(?=\\1?(?:[-*_] *){3,}(?:\\n+|$))')
16914       ('def', '\\n+(?=' + block.def.source + ')')
16915       ();
16916     
16917     block.blockquote = replace(block.blockquote)
16918       ('def', block.def)
16919       ();
16920     
16921     block._tag = '(?!(?:'
16922       + 'a|em|strong|small|s|cite|q|dfn|abbr|data|time|code'
16923       + '|var|samp|kbd|sub|sup|i|b|u|mark|ruby|rt|rp|bdi|bdo'
16924       + '|span|br|wbr|ins|del|img)\\b)\\w+(?!:/|[^\\w\\s@]*@)\\b';
16925     
16926     block.html = replace(block.html)
16927       ('comment', /<!--[\s\S]*?-->/)
16928       ('closed', /<(tag)[\s\S]+?<\/\1>/)
16929       ('closing', /<tag(?:"[^"]*"|'[^']*'|[^'">])*?>/)
16930       (/tag/g, block._tag)
16931       ();
16932     
16933     block.paragraph = replace(block.paragraph)
16934       ('hr', block.hr)
16935       ('heading', block.heading)
16936       ('lheading', block.lheading)
16937       ('blockquote', block.blockquote)
16938       ('tag', '<' + block._tag)
16939       ('def', block.def)
16940       ();
16941     
16942     /**
16943      * Normal Block Grammar
16944      */
16945     
16946     block.normal = merge({}, block);
16947     
16948     /**
16949      * GFM Block Grammar
16950      */
16951     
16952     block.gfm = merge({}, block.normal, {
16953       fences: /^ *(`{3,}|~{3,})[ \.]*(\S+)? *\n([\s\S]*?)\s*\1 *(?:\n+|$)/,
16954       paragraph: /^/,
16955       heading: /^ *(#{1,6}) +([^\n]+?) *#* *(?:\n+|$)/
16956     });
16957     
16958     block.gfm.paragraph = replace(block.paragraph)
16959       ('(?!', '(?!'
16960         + block.gfm.fences.source.replace('\\1', '\\2') + '|'
16961         + block.list.source.replace('\\1', '\\3') + '|')
16962       ();
16963     
16964     /**
16965      * GFM + Tables Block Grammar
16966      */
16967     
16968     block.tables = merge({}, block.gfm, {
16969       nptable: /^ *(\S.*\|.*)\n *([-:]+ *\|[-| :]*)\n((?:.*\|.*(?:\n|$))*)\n*/,
16970       table: /^ *\|(.+)\n *\|( *[-:]+[-| :]*)\n((?: *\|.*(?:\n|$))*)\n*/
16971     });
16972     
16973     /**
16974      * Block Lexer
16975      */
16976     
16977     var Lexer = function (options) {
16978       this.tokens = [];
16979       this.tokens.links = {};
16980       this.options = options || marked.defaults;
16981       this.rules = block.normal;
16982     
16983       if (this.options.gfm) {
16984         if (this.options.tables) {
16985           this.rules = block.tables;
16986         } else {
16987           this.rules = block.gfm;
16988         }
16989       }
16990     }
16991     
16992     /**
16993      * Expose Block Rules
16994      */
16995     
16996     Lexer.rules = block;
16997     
16998     /**
16999      * Static Lex Method
17000      */
17001     
17002     Lexer.lex = function(src, options) {
17003       var lexer = new Lexer(options);
17004       return lexer.lex(src);
17005     };
17006     
17007     /**
17008      * Preprocessing
17009      */
17010     
17011     Lexer.prototype.lex = function(src) {
17012       src = src
17013         .replace(/\r\n|\r/g, '\n')
17014         .replace(/\t/g, '    ')
17015         .replace(/\u00a0/g, ' ')
17016         .replace(/\u2424/g, '\n');
17017     
17018       return this.token(src, true);
17019     };
17020     
17021     /**
17022      * Lexing
17023      */
17024     
17025     Lexer.prototype.token = function(src, top, bq) {
17026       var src = src.replace(/^ +$/gm, '')
17027         , next
17028         , loose
17029         , cap
17030         , bull
17031         , b
17032         , item
17033         , space
17034         , i
17035         , l;
17036     
17037       while (src) {
17038         // newline
17039         if (cap = this.rules.newline.exec(src)) {
17040           src = src.substring(cap[0].length);
17041           if (cap[0].length > 1) {
17042             this.tokens.push({
17043               type: 'space'
17044             });
17045           }
17046         }
17047     
17048         // code
17049         if (cap = this.rules.code.exec(src)) {
17050           src = src.substring(cap[0].length);
17051           cap = cap[0].replace(/^ {4}/gm, '');
17052           this.tokens.push({
17053             type: 'code',
17054             text: !this.options.pedantic
17055               ? cap.replace(/\n+$/, '')
17056               : cap
17057           });
17058           continue;
17059         }
17060     
17061         // fences (gfm)
17062         if (cap = this.rules.fences.exec(src)) {
17063           src = src.substring(cap[0].length);
17064           this.tokens.push({
17065             type: 'code',
17066             lang: cap[2],
17067             text: cap[3] || ''
17068           });
17069           continue;
17070         }
17071     
17072         // heading
17073         if (cap = this.rules.heading.exec(src)) {
17074           src = src.substring(cap[0].length);
17075           this.tokens.push({
17076             type: 'heading',
17077             depth: cap[1].length,
17078             text: cap[2]
17079           });
17080           continue;
17081         }
17082     
17083         // table no leading pipe (gfm)
17084         if (top && (cap = this.rules.nptable.exec(src))) {
17085           src = src.substring(cap[0].length);
17086     
17087           item = {
17088             type: 'table',
17089             header: cap[1].replace(/^ *| *\| *$/g, '').split(/ *\| */),
17090             align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
17091             cells: cap[3].replace(/\n$/, '').split('\n')
17092           };
17093     
17094           for (i = 0; i < item.align.length; i++) {
17095             if (/^ *-+: *$/.test(item.align[i])) {
17096               item.align[i] = 'right';
17097             } else if (/^ *:-+: *$/.test(item.align[i])) {
17098               item.align[i] = 'center';
17099             } else if (/^ *:-+ *$/.test(item.align[i])) {
17100               item.align[i] = 'left';
17101             } else {
17102               item.align[i] = null;
17103             }
17104           }
17105     
17106           for (i = 0; i < item.cells.length; i++) {
17107             item.cells[i] = item.cells[i].split(/ *\| */);
17108           }
17109     
17110           this.tokens.push(item);
17111     
17112           continue;
17113         }
17114     
17115         // lheading
17116         if (cap = this.rules.lheading.exec(src)) {
17117           src = src.substring(cap[0].length);
17118           this.tokens.push({
17119             type: 'heading',
17120             depth: cap[2] === '=' ? 1 : 2,
17121             text: cap[1]
17122           });
17123           continue;
17124         }
17125     
17126         // hr
17127         if (cap = this.rules.hr.exec(src)) {
17128           src = src.substring(cap[0].length);
17129           this.tokens.push({
17130             type: 'hr'
17131           });
17132           continue;
17133         }
17134     
17135         // blockquote
17136         if (cap = this.rules.blockquote.exec(src)) {
17137           src = src.substring(cap[0].length);
17138     
17139           this.tokens.push({
17140             type: 'blockquote_start'
17141           });
17142     
17143           cap = cap[0].replace(/^ *> ?/gm, '');
17144     
17145           // Pass `top` to keep the current
17146           // "toplevel" state. This is exactly
17147           // how markdown.pl works.
17148           this.token(cap, top, true);
17149     
17150           this.tokens.push({
17151             type: 'blockquote_end'
17152           });
17153     
17154           continue;
17155         }
17156     
17157         // list
17158         if (cap = this.rules.list.exec(src)) {
17159           src = src.substring(cap[0].length);
17160           bull = cap[2];
17161     
17162           this.tokens.push({
17163             type: 'list_start',
17164             ordered: bull.length > 1
17165           });
17166     
17167           // Get each top-level item.
17168           cap = cap[0].match(this.rules.item);
17169     
17170           next = false;
17171           l = cap.length;
17172           i = 0;
17173     
17174           for (; i < l; i++) {
17175             item = cap[i];
17176     
17177             // Remove the list item's bullet
17178             // so it is seen as the next token.
17179             space = item.length;
17180             item = item.replace(/^ *([*+-]|\d+\.) +/, '');
17181     
17182             // Outdent whatever the
17183             // list item contains. Hacky.
17184             if (~item.indexOf('\n ')) {
17185               space -= item.length;
17186               item = !this.options.pedantic
17187                 ? item.replace(new RegExp('^ {1,' + space + '}', 'gm'), '')
17188                 : item.replace(/^ {1,4}/gm, '');
17189             }
17190     
17191             // Determine whether the next list item belongs here.
17192             // Backpedal if it does not belong in this list.
17193             if (this.options.smartLists && i !== l - 1) {
17194               b = block.bullet.exec(cap[i + 1])[0];
17195               if (bull !== b && !(bull.length > 1 && b.length > 1)) {
17196                 src = cap.slice(i + 1).join('\n') + src;
17197                 i = l - 1;
17198               }
17199             }
17200     
17201             // Determine whether item is loose or not.
17202             // Use: /(^|\n)(?! )[^\n]+\n\n(?!\s*$)/
17203             // for discount behavior.
17204             loose = next || /\n\n(?!\s*$)/.test(item);
17205             if (i !== l - 1) {
17206               next = item.charAt(item.length - 1) === '\n';
17207               if (!loose) { loose = next; }
17208             }
17209     
17210             this.tokens.push({
17211               type: loose
17212                 ? 'loose_item_start'
17213                 : 'list_item_start'
17214             });
17215     
17216             // Recurse.
17217             this.token(item, false, bq);
17218     
17219             this.tokens.push({
17220               type: 'list_item_end'
17221             });
17222           }
17223     
17224           this.tokens.push({
17225             type: 'list_end'
17226           });
17227     
17228           continue;
17229         }
17230     
17231         // html
17232         if (cap = this.rules.html.exec(src)) {
17233           src = src.substring(cap[0].length);
17234           this.tokens.push({
17235             type: this.options.sanitize
17236               ? 'paragraph'
17237               : 'html',
17238             pre: !this.options.sanitizer
17239               && (cap[1] === 'pre' || cap[1] === 'script' || cap[1] === 'style'),
17240             text: cap[0]
17241           });
17242           continue;
17243         }
17244     
17245         // def
17246         if ((!bq && top) && (cap = this.rules.def.exec(src))) {
17247           src = src.substring(cap[0].length);
17248           this.tokens.links[cap[1].toLowerCase()] = {
17249             href: cap[2],
17250             title: cap[3]
17251           };
17252           continue;
17253         }
17254     
17255         // table (gfm)
17256         if (top && (cap = this.rules.table.exec(src))) {
17257           src = src.substring(cap[0].length);
17258     
17259           item = {
17260             type: 'table',
17261             header: cap[1].replace(/^ *| *\| *$/g, '').split(/ *\| */),
17262             align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
17263             cells: cap[3].replace(/(?: *\| *)?\n$/, '').split('\n')
17264           };
17265     
17266           for (i = 0; i < item.align.length; i++) {
17267             if (/^ *-+: *$/.test(item.align[i])) {
17268               item.align[i] = 'right';
17269             } else if (/^ *:-+: *$/.test(item.align[i])) {
17270               item.align[i] = 'center';
17271             } else if (/^ *:-+ *$/.test(item.align[i])) {
17272               item.align[i] = 'left';
17273             } else {
17274               item.align[i] = null;
17275             }
17276           }
17277     
17278           for (i = 0; i < item.cells.length; i++) {
17279             item.cells[i] = item.cells[i]
17280               .replace(/^ *\| *| *\| *$/g, '')
17281               .split(/ *\| */);
17282           }
17283     
17284           this.tokens.push(item);
17285     
17286           continue;
17287         }
17288     
17289         // top-level paragraph
17290         if (top && (cap = this.rules.paragraph.exec(src))) {
17291           src = src.substring(cap[0].length);
17292           this.tokens.push({
17293             type: 'paragraph',
17294             text: cap[1].charAt(cap[1].length - 1) === '\n'
17295               ? cap[1].slice(0, -1)
17296               : cap[1]
17297           });
17298           continue;
17299         }
17300     
17301         // text
17302         if (cap = this.rules.text.exec(src)) {
17303           // Top-level should never reach here.
17304           src = src.substring(cap[0].length);
17305           this.tokens.push({
17306             type: 'text',
17307             text: cap[0]
17308           });
17309           continue;
17310         }
17311     
17312         if (src) {
17313           throw new
17314             Error('Infinite loop on byte: ' + src.charCodeAt(0));
17315         }
17316       }
17317     
17318       return this.tokens;
17319     };
17320     
17321     /**
17322      * Inline-Level Grammar
17323      */
17324     
17325     var inline = {
17326       escape: /^\\([\\`*{}\[\]()#+\-.!_>])/,
17327       autolink: /^<([^ >]+(@|:\/)[^ >]+)>/,
17328       url: noop,
17329       tag: /^<!--[\s\S]*?-->|^<\/?\w+(?:"[^"]*"|'[^']*'|[^'">])*?>/,
17330       link: /^!?\[(inside)\]\(href\)/,
17331       reflink: /^!?\[(inside)\]\s*\[([^\]]*)\]/,
17332       nolink: /^!?\[((?:\[[^\]]*\]|[^\[\]])*)\]/,
17333       strong: /^__([\s\S]+?)__(?!_)|^\*\*([\s\S]+?)\*\*(?!\*)/,
17334       em: /^\b_((?:[^_]|__)+?)_\b|^\*((?:\*\*|[\s\S])+?)\*(?!\*)/,
17335       code: /^(`+)\s*([\s\S]*?[^`])\s*\1(?!`)/,
17336       br: /^ {2,}\n(?!\s*$)/,
17337       del: noop,
17338       text: /^[\s\S]+?(?=[\\<!\[_*`]| {2,}\n|$)/
17339     };
17340     
17341     inline._inside = /(?:\[[^\]]*\]|[^\[\]]|\](?=[^\[]*\]))*/;
17342     inline._href = /\s*<?([\s\S]*?)>?(?:\s+['"]([\s\S]*?)['"])?\s*/;
17343     
17344     inline.link = replace(inline.link)
17345       ('inside', inline._inside)
17346       ('href', inline._href)
17347       ();
17348     
17349     inline.reflink = replace(inline.reflink)
17350       ('inside', inline._inside)
17351       ();
17352     
17353     /**
17354      * Normal Inline Grammar
17355      */
17356     
17357     inline.normal = merge({}, inline);
17358     
17359     /**
17360      * Pedantic Inline Grammar
17361      */
17362     
17363     inline.pedantic = merge({}, inline.normal, {
17364       strong: /^__(?=\S)([\s\S]*?\S)__(?!_)|^\*\*(?=\S)([\s\S]*?\S)\*\*(?!\*)/,
17365       em: /^_(?=\S)([\s\S]*?\S)_(?!_)|^\*(?=\S)([\s\S]*?\S)\*(?!\*)/
17366     });
17367     
17368     /**
17369      * GFM Inline Grammar
17370      */
17371     
17372     inline.gfm = merge({}, inline.normal, {
17373       escape: replace(inline.escape)('])', '~|])')(),
17374       url: /^(https?:\/\/[^\s<]+[^<.,:;"')\]\s])/,
17375       del: /^~~(?=\S)([\s\S]*?\S)~~/,
17376       text: replace(inline.text)
17377         (']|', '~]|')
17378         ('|', '|https?://|')
17379         ()
17380     });
17381     
17382     /**
17383      * GFM + Line Breaks Inline Grammar
17384      */
17385     
17386     inline.breaks = merge({}, inline.gfm, {
17387       br: replace(inline.br)('{2,}', '*')(),
17388       text: replace(inline.gfm.text)('{2,}', '*')()
17389     });
17390     
17391     /**
17392      * Inline Lexer & Compiler
17393      */
17394     
17395     var InlineLexer  = function (links, options) {
17396       this.options = options || marked.defaults;
17397       this.links = links;
17398       this.rules = inline.normal;
17399       this.renderer = this.options.renderer || new Renderer;
17400       this.renderer.options = this.options;
17401     
17402       if (!this.links) {
17403         throw new
17404           Error('Tokens array requires a `links` property.');
17405       }
17406     
17407       if (this.options.gfm) {
17408         if (this.options.breaks) {
17409           this.rules = inline.breaks;
17410         } else {
17411           this.rules = inline.gfm;
17412         }
17413       } else if (this.options.pedantic) {
17414         this.rules = inline.pedantic;
17415       }
17416     }
17417     
17418     /**
17419      * Expose Inline Rules
17420      */
17421     
17422     InlineLexer.rules = inline;
17423     
17424     /**
17425      * Static Lexing/Compiling Method
17426      */
17427     
17428     InlineLexer.output = function(src, links, options) {
17429       var inline = new InlineLexer(links, options);
17430       return inline.output(src);
17431     };
17432     
17433     /**
17434      * Lexing/Compiling
17435      */
17436     
17437     InlineLexer.prototype.output = function(src) {
17438       var out = ''
17439         , link
17440         , text
17441         , href
17442         , cap;
17443     
17444       while (src) {
17445         // escape
17446         if (cap = this.rules.escape.exec(src)) {
17447           src = src.substring(cap[0].length);
17448           out += cap[1];
17449           continue;
17450         }
17451     
17452         // autolink
17453         if (cap = this.rules.autolink.exec(src)) {
17454           src = src.substring(cap[0].length);
17455           if (cap[2] === '@') {
17456             text = cap[1].charAt(6) === ':'
17457               ? this.mangle(cap[1].substring(7))
17458               : this.mangle(cap[1]);
17459             href = this.mangle('mailto:') + text;
17460           } else {
17461             text = escape(cap[1]);
17462             href = text;
17463           }
17464           out += this.renderer.link(href, null, text);
17465           continue;
17466         }
17467     
17468         // url (gfm)
17469         if (!this.inLink && (cap = this.rules.url.exec(src))) {
17470           src = src.substring(cap[0].length);
17471           text = escape(cap[1]);
17472           href = text;
17473           out += this.renderer.link(href, null, text);
17474           continue;
17475         }
17476     
17477         // tag
17478         if (cap = this.rules.tag.exec(src)) {
17479           if (!this.inLink && /^<a /i.test(cap[0])) {
17480             this.inLink = true;
17481           } else if (this.inLink && /^<\/a>/i.test(cap[0])) {
17482             this.inLink = false;
17483           }
17484           src = src.substring(cap[0].length);
17485           out += this.options.sanitize
17486             ? this.options.sanitizer
17487               ? this.options.sanitizer(cap[0])
17488               : escape(cap[0])
17489             : cap[0];
17490           continue;
17491         }
17492     
17493         // link
17494         if (cap = this.rules.link.exec(src)) {
17495           src = src.substring(cap[0].length);
17496           this.inLink = true;
17497           out += this.outputLink(cap, {
17498             href: cap[2],
17499             title: cap[3]
17500           });
17501           this.inLink = false;
17502           continue;
17503         }
17504     
17505         // reflink, nolink
17506         if ((cap = this.rules.reflink.exec(src))
17507             || (cap = this.rules.nolink.exec(src))) {
17508           src = src.substring(cap[0].length);
17509           link = (cap[2] || cap[1]).replace(/\s+/g, ' ');
17510           link = this.links[link.toLowerCase()];
17511           if (!link || !link.href) {
17512             out += cap[0].charAt(0);
17513             src = cap[0].substring(1) + src;
17514             continue;
17515           }
17516           this.inLink = true;
17517           out += this.outputLink(cap, link);
17518           this.inLink = false;
17519           continue;
17520         }
17521     
17522         // strong
17523         if (cap = this.rules.strong.exec(src)) {
17524           src = src.substring(cap[0].length);
17525           out += this.renderer.strong(this.output(cap[2] || cap[1]));
17526           continue;
17527         }
17528     
17529         // em
17530         if (cap = this.rules.em.exec(src)) {
17531           src = src.substring(cap[0].length);
17532           out += this.renderer.em(this.output(cap[2] || cap[1]));
17533           continue;
17534         }
17535     
17536         // code
17537         if (cap = this.rules.code.exec(src)) {
17538           src = src.substring(cap[0].length);
17539           out += this.renderer.codespan(escape(cap[2], true));
17540           continue;
17541         }
17542     
17543         // br
17544         if (cap = this.rules.br.exec(src)) {
17545           src = src.substring(cap[0].length);
17546           out += this.renderer.br();
17547           continue;
17548         }
17549     
17550         // del (gfm)
17551         if (cap = this.rules.del.exec(src)) {
17552           src = src.substring(cap[0].length);
17553           out += this.renderer.del(this.output(cap[1]));
17554           continue;
17555         }
17556     
17557         // text
17558         if (cap = this.rules.text.exec(src)) {
17559           src = src.substring(cap[0].length);
17560           out += this.renderer.text(escape(this.smartypants(cap[0])));
17561           continue;
17562         }
17563     
17564         if (src) {
17565           throw new
17566             Error('Infinite loop on byte: ' + src.charCodeAt(0));
17567         }
17568       }
17569     
17570       return out;
17571     };
17572     
17573     /**
17574      * Compile Link
17575      */
17576     
17577     InlineLexer.prototype.outputLink = function(cap, link) {
17578       var href = escape(link.href)
17579         , title = link.title ? escape(link.title) : null;
17580     
17581       return cap[0].charAt(0) !== '!'
17582         ? this.renderer.link(href, title, this.output(cap[1]))
17583         : this.renderer.image(href, title, escape(cap[1]));
17584     };
17585     
17586     /**
17587      * Smartypants Transformations
17588      */
17589     
17590     InlineLexer.prototype.smartypants = function(text) {
17591       if (!this.options.smartypants)  { return text; }
17592       return text
17593         // em-dashes
17594         .replace(/---/g, '\u2014')
17595         // en-dashes
17596         .replace(/--/g, '\u2013')
17597         // opening singles
17598         .replace(/(^|[-\u2014/(\[{"\s])'/g, '$1\u2018')
17599         // closing singles & apostrophes
17600         .replace(/'/g, '\u2019')
17601         // opening doubles
17602         .replace(/(^|[-\u2014/(\[{\u2018\s])"/g, '$1\u201c')
17603         // closing doubles
17604         .replace(/"/g, '\u201d')
17605         // ellipses
17606         .replace(/\.{3}/g, '\u2026');
17607     };
17608     
17609     /**
17610      * Mangle Links
17611      */
17612     
17613     InlineLexer.prototype.mangle = function(text) {
17614       if (!this.options.mangle) { return text; }
17615       var out = ''
17616         , l = text.length
17617         , i = 0
17618         , ch;
17619     
17620       for (; i < l; i++) {
17621         ch = text.charCodeAt(i);
17622         if (Math.random() > 0.5) {
17623           ch = 'x' + ch.toString(16);
17624         }
17625         out += '&#' + ch + ';';
17626       }
17627     
17628       return out;
17629     };
17630     
17631     /**
17632      * Renderer
17633      */
17634     
17635      /**
17636          * eval:var:Renderer
17637     */
17638     
17639     var Renderer   = function (options) {
17640       this.options = options || {};
17641     }
17642     
17643     Renderer.prototype.code = function(code, lang, escaped) {
17644       if (this.options.highlight) {
17645         var out = this.options.highlight(code, lang);
17646         if (out != null && out !== code) {
17647           escaped = true;
17648           code = out;
17649         }
17650       } else {
17651             // hack!!! - it's already escapeD?
17652             escaped = true;
17653       }
17654     
17655       if (!lang) {
17656         return '<pre><code>'
17657           + (escaped ? code : escape(code, true))
17658           + '\n</code></pre>';
17659       }
17660     
17661       return '<pre><code class="'
17662         + this.options.langPrefix
17663         + escape(lang, true)
17664         + '">'
17665         + (escaped ? code : escape(code, true))
17666         + '\n</code></pre>\n';
17667     };
17668     
17669     Renderer.prototype.blockquote = function(quote) {
17670       return '<blockquote>\n' + quote + '</blockquote>\n';
17671     };
17672     
17673     Renderer.prototype.html = function(html) {
17674       return html;
17675     };
17676     
17677     Renderer.prototype.heading = function(text, level, raw) {
17678       return '<h'
17679         + level
17680         + ' id="'
17681         + this.options.headerPrefix
17682         + raw.toLowerCase().replace(/[^\w]+/g, '-')
17683         + '">'
17684         + text
17685         + '</h'
17686         + level
17687         + '>\n';
17688     };
17689     
17690     Renderer.prototype.hr = function() {
17691       return this.options.xhtml ? '<hr/>\n' : '<hr>\n';
17692     };
17693     
17694     Renderer.prototype.list = function(body, ordered) {
17695       var type = ordered ? 'ol' : 'ul';
17696       return '<' + type + '>\n' + body + '</' + type + '>\n';
17697     };
17698     
17699     Renderer.prototype.listitem = function(text) {
17700       return '<li>' + text + '</li>\n';
17701     };
17702     
17703     Renderer.prototype.paragraph = function(text) {
17704       return '<p>' + text + '</p>\n';
17705     };
17706     
17707     Renderer.prototype.table = function(header, body) {
17708       return '<table class="table table-striped">\n'
17709         + '<thead>\n'
17710         + header
17711         + '</thead>\n'
17712         + '<tbody>\n'
17713         + body
17714         + '</tbody>\n'
17715         + '</table>\n';
17716     };
17717     
17718     Renderer.prototype.tablerow = function(content) {
17719       return '<tr>\n' + content + '</tr>\n';
17720     };
17721     
17722     Renderer.prototype.tablecell = function(content, flags) {
17723       var type = flags.header ? 'th' : 'td';
17724       var tag = flags.align
17725         ? '<' + type + ' style="text-align:' + flags.align + '">'
17726         : '<' + type + '>';
17727       return tag + content + '</' + type + '>\n';
17728     };
17729     
17730     // span level renderer
17731     Renderer.prototype.strong = function(text) {
17732       return '<strong>' + text + '</strong>';
17733     };
17734     
17735     Renderer.prototype.em = function(text) {
17736       return '<em>' + text + '</em>';
17737     };
17738     
17739     Renderer.prototype.codespan = function(text) {
17740       return '<code>' + text + '</code>';
17741     };
17742     
17743     Renderer.prototype.br = function() {
17744       return this.options.xhtml ? '<br/>' : '<br>';
17745     };
17746     
17747     Renderer.prototype.del = function(text) {
17748       return '<del>' + text + '</del>';
17749     };
17750     
17751     Renderer.prototype.link = function(href, title, text) {
17752       if (this.options.sanitize) {
17753         try {
17754           var prot = decodeURIComponent(unescape(href))
17755             .replace(/[^\w:]/g, '')
17756             .toLowerCase();
17757         } catch (e) {
17758           return '';
17759         }
17760         if (prot.indexOf('javascript:') === 0 || prot.indexOf('vbscript:') === 0) {
17761           return '';
17762         }
17763       }
17764       var out = '<a href="' + href + '"';
17765       if (title) {
17766         out += ' title="' + title + '"';
17767       }
17768       out += '>' + text + '</a>';
17769       return out;
17770     };
17771     
17772     Renderer.prototype.image = function(href, title, text) {
17773       var out = '<img src="' + href + '" alt="' + text + '"';
17774       if (title) {
17775         out += ' title="' + title + '"';
17776       }
17777       out += this.options.xhtml ? '/>' : '>';
17778       return out;
17779     };
17780     
17781     Renderer.prototype.text = function(text) {
17782       return text;
17783     };
17784     
17785     /**
17786      * Parsing & Compiling
17787      */
17788          /**
17789          * eval:var:Parser
17790     */
17791     
17792     var Parser= function (options) {
17793       this.tokens = [];
17794       this.token = null;
17795       this.options = options || marked.defaults;
17796       this.options.renderer = this.options.renderer || new Renderer;
17797       this.renderer = this.options.renderer;
17798       this.renderer.options = this.options;
17799     }
17800     
17801     /**
17802      * Static Parse Method
17803      */
17804     
17805     Parser.parse = function(src, options, renderer) {
17806       var parser = new Parser(options, renderer);
17807       return parser.parse(src);
17808     };
17809     
17810     /**
17811      * Parse Loop
17812      */
17813     
17814     Parser.prototype.parse = function(src) {
17815       this.inline = new InlineLexer(src.links, this.options, this.renderer);
17816       this.tokens = src.reverse();
17817     
17818       var out = '';
17819       while (this.next()) {
17820         out += this.tok();
17821       }
17822     
17823       return out;
17824     };
17825     
17826     /**
17827      * Next Token
17828      */
17829     
17830     Parser.prototype.next = function() {
17831       return this.token = this.tokens.pop();
17832     };
17833     
17834     /**
17835      * Preview Next Token
17836      */
17837     
17838     Parser.prototype.peek = function() {
17839       return this.tokens[this.tokens.length - 1] || 0;
17840     };
17841     
17842     /**
17843      * Parse Text Tokens
17844      */
17845     
17846     Parser.prototype.parseText = function() {
17847       var body = this.token.text;
17848     
17849       while (this.peek().type === 'text') {
17850         body += '\n' + this.next().text;
17851       }
17852     
17853       return this.inline.output(body);
17854     };
17855     
17856     /**
17857      * Parse Current Token
17858      */
17859     
17860     Parser.prototype.tok = function() {
17861       switch (this.token.type) {
17862         case 'space': {
17863           return '';
17864         }
17865         case 'hr': {
17866           return this.renderer.hr();
17867         }
17868         case 'heading': {
17869           return this.renderer.heading(
17870             this.inline.output(this.token.text),
17871             this.token.depth,
17872             this.token.text);
17873         }
17874         case 'code': {
17875           return this.renderer.code(this.token.text,
17876             this.token.lang,
17877             this.token.escaped);
17878         }
17879         case 'table': {
17880           var header = ''
17881             , body = ''
17882             , i
17883             , row
17884             , cell
17885             , flags
17886             , j;
17887     
17888           // header
17889           cell = '';
17890           for (i = 0; i < this.token.header.length; i++) {
17891             flags = { header: true, align: this.token.align[i] };
17892             cell += this.renderer.tablecell(
17893               this.inline.output(this.token.header[i]),
17894               { header: true, align: this.token.align[i] }
17895             );
17896           }
17897           header += this.renderer.tablerow(cell);
17898     
17899           for (i = 0; i < this.token.cells.length; i++) {
17900             row = this.token.cells[i];
17901     
17902             cell = '';
17903             for (j = 0; j < row.length; j++) {
17904               cell += this.renderer.tablecell(
17905                 this.inline.output(row[j]),
17906                 { header: false, align: this.token.align[j] }
17907               );
17908             }
17909     
17910             body += this.renderer.tablerow(cell);
17911           }
17912           return this.renderer.table(header, body);
17913         }
17914         case 'blockquote_start': {
17915           var body = '';
17916     
17917           while (this.next().type !== 'blockquote_end') {
17918             body += this.tok();
17919           }
17920     
17921           return this.renderer.blockquote(body);
17922         }
17923         case 'list_start': {
17924           var body = ''
17925             , ordered = this.token.ordered;
17926     
17927           while (this.next().type !== 'list_end') {
17928             body += this.tok();
17929           }
17930     
17931           return this.renderer.list(body, ordered);
17932         }
17933         case 'list_item_start': {
17934           var body = '';
17935     
17936           while (this.next().type !== 'list_item_end') {
17937             body += this.token.type === 'text'
17938               ? this.parseText()
17939               : this.tok();
17940           }
17941     
17942           return this.renderer.listitem(body);
17943         }
17944         case 'loose_item_start': {
17945           var body = '';
17946     
17947           while (this.next().type !== 'list_item_end') {
17948             body += this.tok();
17949           }
17950     
17951           return this.renderer.listitem(body);
17952         }
17953         case 'html': {
17954           var html = !this.token.pre && !this.options.pedantic
17955             ? this.inline.output(this.token.text)
17956             : this.token.text;
17957           return this.renderer.html(html);
17958         }
17959         case 'paragraph': {
17960           return this.renderer.paragraph(this.inline.output(this.token.text));
17961         }
17962         case 'text': {
17963           return this.renderer.paragraph(this.parseText());
17964         }
17965       }
17966     };
17967   
17968     
17969     /**
17970      * Marked
17971      */
17972          /**
17973          * eval:var:marked
17974     */
17975     var marked = function (src, opt, callback) {
17976       if (callback || typeof opt === 'function') {
17977         if (!callback) {
17978           callback = opt;
17979           opt = null;
17980         }
17981     
17982         opt = merge({}, marked.defaults, opt || {});
17983     
17984         var highlight = opt.highlight
17985           , tokens
17986           , pending
17987           , i = 0;
17988     
17989         try {
17990           tokens = Lexer.lex(src, opt)
17991         } catch (e) {
17992           return callback(e);
17993         }
17994     
17995         pending = tokens.length;
17996          /**
17997          * eval:var:done
17998     */
17999         var done = function(err) {
18000           if (err) {
18001             opt.highlight = highlight;
18002             return callback(err);
18003           }
18004     
18005           var out;
18006     
18007           try {
18008             out = Parser.parse(tokens, opt);
18009           } catch (e) {
18010             err = e;
18011           }
18012     
18013           opt.highlight = highlight;
18014     
18015           return err
18016             ? callback(err)
18017             : callback(null, out);
18018         };
18019     
18020         if (!highlight || highlight.length < 3) {
18021           return done();
18022         }
18023     
18024         delete opt.highlight;
18025     
18026         if (!pending) { return done(); }
18027     
18028         for (; i < tokens.length; i++) {
18029           (function(token) {
18030             if (token.type !== 'code') {
18031               return --pending || done();
18032             }
18033             return highlight(token.text, token.lang, function(err, code) {
18034               if (err) { return done(err); }
18035               if (code == null || code === token.text) {
18036                 return --pending || done();
18037               }
18038               token.text = code;
18039               token.escaped = true;
18040               --pending || done();
18041             });
18042           })(tokens[i]);
18043         }
18044     
18045         return;
18046       }
18047       try {
18048         if (opt) { opt = merge({}, marked.defaults, opt); }
18049         return Parser.parse(Lexer.lex(src, opt), opt);
18050       } catch (e) {
18051         e.message += '\nPlease report this to https://github.com/chjj/marked.';
18052         if ((opt || marked.defaults).silent) {
18053           return '<p>An error occured:</p><pre>'
18054             + escape(e.message + '', true)
18055             + '</pre>';
18056         }
18057         throw e;
18058       }
18059     }
18060     
18061     /**
18062      * Options
18063      */
18064     
18065     marked.options =
18066     marked.setOptions = function(opt) {
18067       merge(marked.defaults, opt);
18068       return marked;
18069     };
18070     
18071     marked.defaults = {
18072       gfm: true,
18073       tables: true,
18074       breaks: false,
18075       pedantic: false,
18076       sanitize: false,
18077       sanitizer: null,
18078       mangle: true,
18079       smartLists: false,
18080       silent: false,
18081       highlight: null,
18082       langPrefix: 'lang-',
18083       smartypants: false,
18084       headerPrefix: '',
18085       renderer: new Renderer,
18086       xhtml: false
18087     };
18088     
18089     /**
18090      * Expose
18091      */
18092     
18093     marked.Parser = Parser;
18094     marked.parser = Parser.parse;
18095     
18096     marked.Renderer = Renderer;
18097     
18098     marked.Lexer = Lexer;
18099     marked.lexer = Lexer.lex;
18100     
18101     marked.InlineLexer = InlineLexer;
18102     marked.inlineLexer = InlineLexer.output;
18103     
18104     marked.parse = marked;
18105     
18106     Roo.Markdown.marked = marked;
18107
18108 })();/*
18109  * Based on:
18110  * Ext JS Library 1.1.1
18111  * Copyright(c) 2006-2007, Ext JS, LLC.
18112  *
18113  * Originally Released Under LGPL - original licence link has changed is not relivant.
18114  *
18115  * Fork - LGPL
18116  * <script type="text/javascript">
18117  */
18118
18119
18120
18121 /*
18122  * These classes are derivatives of the similarly named classes in the YUI Library.
18123  * The original license:
18124  * Copyright (c) 2006, Yahoo! Inc. All rights reserved.
18125  * Code licensed under the BSD License:
18126  * http://developer.yahoo.net/yui/license.txt
18127  */
18128
18129 (function() {
18130
18131 var Event=Roo.EventManager;
18132 var Dom=Roo.lib.Dom;
18133
18134 /**
18135  * @class Roo.dd.DragDrop
18136  * @extends Roo.util.Observable
18137  * Defines the interface and base operation of items that that can be
18138  * dragged or can be drop targets.  It was designed to be extended, overriding
18139  * the event handlers for startDrag, onDrag, onDragOver and onDragOut.
18140  * Up to three html elements can be associated with a DragDrop instance:
18141  * <ul>
18142  * <li>linked element: the element that is passed into the constructor.
18143  * This is the element which defines the boundaries for interaction with
18144  * other DragDrop objects.</li>
18145  * <li>handle element(s): The drag operation only occurs if the element that
18146  * was clicked matches a handle element.  By default this is the linked
18147  * element, but there are times that you will want only a portion of the
18148  * linked element to initiate the drag operation, and the setHandleElId()
18149  * method provides a way to define this.</li>
18150  * <li>drag element: this represents the element that would be moved along
18151  * with the cursor during a drag operation.  By default, this is the linked
18152  * element itself as in {@link Roo.dd.DD}.  setDragElId() lets you define
18153  * a separate element that would be moved, as in {@link Roo.dd.DDProxy}.
18154  * </li>
18155  * </ul>
18156  * This class should not be instantiated until the onload event to ensure that
18157  * the associated elements are available.
18158  * The following would define a DragDrop obj that would interact with any
18159  * other DragDrop obj in the "group1" group:
18160  * <pre>
18161  *  dd = new Roo.dd.DragDrop("div1", "group1");
18162  * </pre>
18163  * Since none of the event handlers have been implemented, nothing would
18164  * actually happen if you were to run the code above.  Normally you would
18165  * override this class or one of the default implementations, but you can
18166  * also override the methods you want on an instance of the class...
18167  * <pre>
18168  *  dd.onDragDrop = function(e, id) {
18169  *  &nbsp;&nbsp;alert("dd was dropped on " + id);
18170  *  }
18171  * </pre>
18172  * @constructor
18173  * @param {String} id of the element that is linked to this instance
18174  * @param {String} sGroup the group of related DragDrop objects
18175  * @param {object} config an object containing configurable attributes
18176  *                Valid properties for DragDrop:
18177  *                    padding, isTarget, maintainOffset, primaryButtonOnly
18178  */
18179 Roo.dd.DragDrop = function(id, sGroup, config) {
18180     if (id) {
18181         this.init(id, sGroup, config);
18182     }
18183     
18184 };
18185
18186 Roo.extend(Roo.dd.DragDrop, Roo.util.Observable , {
18187
18188     /**
18189      * The id of the element associated with this object.  This is what we
18190      * refer to as the "linked element" because the size and position of
18191      * this element is used to determine when the drag and drop objects have
18192      * interacted.
18193      * @property id
18194      * @type String
18195      */
18196     id: null,
18197
18198     /**
18199      * Configuration attributes passed into the constructor
18200      * @property config
18201      * @type object
18202      */
18203     config: null,
18204
18205     /**
18206      * The id of the element that will be dragged.  By default this is same
18207      * as the linked element , but could be changed to another element. Ex:
18208      * Roo.dd.DDProxy
18209      * @property dragElId
18210      * @type String
18211      * @private
18212      */
18213     dragElId: null,
18214
18215     /**
18216      * the id of the element that initiates the drag operation.  By default
18217      * this is the linked element, but could be changed to be a child of this
18218      * element.  This lets us do things like only starting the drag when the
18219      * header element within the linked html element is clicked.
18220      * @property handleElId
18221      * @type String
18222      * @private
18223      */
18224     handleElId: null,
18225
18226     /**
18227      * An associative array of HTML tags that will be ignored if clicked.
18228      * @property invalidHandleTypes
18229      * @type {string: string}
18230      */
18231     invalidHandleTypes: null,
18232
18233     /**
18234      * An associative array of ids for elements that will be ignored if clicked
18235      * @property invalidHandleIds
18236      * @type {string: string}
18237      */
18238     invalidHandleIds: null,
18239
18240     /**
18241      * An indexted array of css class names for elements that will be ignored
18242      * if clicked.
18243      * @property invalidHandleClasses
18244      * @type string[]
18245      */
18246     invalidHandleClasses: null,
18247
18248     /**
18249      * The linked element's absolute X position at the time the drag was
18250      * started
18251      * @property startPageX
18252      * @type int
18253      * @private
18254      */
18255     startPageX: 0,
18256
18257     /**
18258      * The linked element's absolute X position at the time the drag was
18259      * started
18260      * @property startPageY
18261      * @type int
18262      * @private
18263      */
18264     startPageY: 0,
18265
18266     /**
18267      * The group defines a logical collection of DragDrop objects that are
18268      * related.  Instances only get events when interacting with other
18269      * DragDrop object in the same group.  This lets us define multiple
18270      * groups using a single DragDrop subclass if we want.
18271      * @property groups
18272      * @type {string: string}
18273      */
18274     groups: null,
18275
18276     /**
18277      * Individual drag/drop instances can be locked.  This will prevent
18278      * onmousedown start drag.
18279      * @property locked
18280      * @type boolean
18281      * @private
18282      */
18283     locked: false,
18284
18285     /**
18286      * Lock this instance
18287      * @method lock
18288      */
18289     lock: function() { this.locked = true; },
18290
18291     /**
18292      * Unlock this instace
18293      * @method unlock
18294      */
18295     unlock: function() { this.locked = false; },
18296
18297     /**
18298      * By default, all insances can be a drop target.  This can be disabled by
18299      * setting isTarget to false.
18300      * @method isTarget
18301      * @type boolean
18302      */
18303     isTarget: true,
18304
18305     /**
18306      * The padding configured for this drag and drop object for calculating
18307      * the drop zone intersection with this object.
18308      * @method padding
18309      * @type int[]
18310      */
18311     padding: null,
18312
18313     /**
18314      * Cached reference to the linked element
18315      * @property _domRef
18316      * @private
18317      */
18318     _domRef: null,
18319
18320     /**
18321      * Internal typeof flag
18322      * @property __ygDragDrop
18323      * @private
18324      */
18325     __ygDragDrop: true,
18326
18327     /**
18328      * Set to true when horizontal contraints are applied
18329      * @property constrainX
18330      * @type boolean
18331      * @private
18332      */
18333     constrainX: false,
18334
18335     /**
18336      * Set to true when vertical contraints are applied
18337      * @property constrainY
18338      * @type boolean
18339      * @private
18340      */
18341     constrainY: false,
18342
18343     /**
18344      * The left constraint
18345      * @property minX
18346      * @type int
18347      * @private
18348      */
18349     minX: 0,
18350
18351     /**
18352      * The right constraint
18353      * @property maxX
18354      * @type int
18355      * @private
18356      */
18357     maxX: 0,
18358
18359     /**
18360      * The up constraint
18361      * @property minY
18362      * @type int
18363      * @type int
18364      * @private
18365      */
18366     minY: 0,
18367
18368     /**
18369      * The down constraint
18370      * @property maxY
18371      * @type int
18372      * @private
18373      */
18374     maxY: 0,
18375
18376     /**
18377      * Maintain offsets when we resetconstraints.  Set to true when you want
18378      * the position of the element relative to its parent to stay the same
18379      * when the page changes
18380      *
18381      * @property maintainOffset
18382      * @type boolean
18383      */
18384     maintainOffset: false,
18385
18386     /**
18387      * Array of pixel locations the element will snap to if we specified a
18388      * horizontal graduation/interval.  This array is generated automatically
18389      * when you define a tick interval.
18390      * @property xTicks
18391      * @type int[]
18392      */
18393     xTicks: null,
18394
18395     /**
18396      * Array of pixel locations the element will snap to if we specified a
18397      * vertical graduation/interval.  This array is generated automatically
18398      * when you define a tick interval.
18399      * @property yTicks
18400      * @type int[]
18401      */
18402     yTicks: null,
18403
18404     /**
18405      * By default the drag and drop instance will only respond to the primary
18406      * button click (left button for a right-handed mouse).  Set to true to
18407      * allow drag and drop to start with any mouse click that is propogated
18408      * by the browser
18409      * @property primaryButtonOnly
18410      * @type boolean
18411      */
18412     primaryButtonOnly: true,
18413
18414     /**
18415      * The availabe property is false until the linked dom element is accessible.
18416      * @property available
18417      * @type boolean
18418      */
18419     available: false,
18420
18421     /**
18422      * By default, drags can only be initiated if the mousedown occurs in the
18423      * region the linked element is.  This is done in part to work around a
18424      * bug in some browsers that mis-report the mousedown if the previous
18425      * mouseup happened outside of the window.  This property is set to true
18426      * if outer handles are defined.
18427      *
18428      * @property hasOuterHandles
18429      * @type boolean
18430      * @default false
18431      */
18432     hasOuterHandles: false,
18433
18434     /**
18435      * Code that executes immediately before the startDrag event
18436      * @method b4StartDrag
18437      * @private
18438      */
18439     b4StartDrag: function(x, y) { },
18440
18441     /**
18442      * Abstract method called after a drag/drop object is clicked
18443      * and the drag or mousedown time thresholds have beeen met.
18444      * @method startDrag
18445      * @param {int} X click location
18446      * @param {int} Y click location
18447      */
18448     startDrag: function(x, y) { /* override this */ },
18449
18450     /**
18451      * Code that executes immediately before the onDrag event
18452      * @method b4Drag
18453      * @private
18454      */
18455     b4Drag: function(e) { },
18456
18457     /**
18458      * Abstract method called during the onMouseMove event while dragging an
18459      * object.
18460      * @method onDrag
18461      * @param {Event} e the mousemove event
18462      */
18463     onDrag: function(e) { /* override this */ },
18464
18465     /**
18466      * Abstract method called when this element fist begins hovering over
18467      * another DragDrop obj
18468      * @method onDragEnter
18469      * @param {Event} e the mousemove event
18470      * @param {String|DragDrop[]} id In POINT mode, the element
18471      * id this is hovering over.  In INTERSECT mode, an array of one or more
18472      * dragdrop items being hovered over.
18473      */
18474     onDragEnter: function(e, id) { /* override this */ },
18475
18476     /**
18477      * Code that executes immediately before the onDragOver event
18478      * @method b4DragOver
18479      * @private
18480      */
18481     b4DragOver: function(e) { },
18482
18483     /**
18484      * Abstract method called when this element is hovering over another
18485      * DragDrop obj
18486      * @method onDragOver
18487      * @param {Event} e the mousemove event
18488      * @param {String|DragDrop[]} id In POINT mode, the element
18489      * id this is hovering over.  In INTERSECT mode, an array of dd items
18490      * being hovered over.
18491      */
18492     onDragOver: function(e, id) { /* override this */ },
18493
18494     /**
18495      * Code that executes immediately before the onDragOut event
18496      * @method b4DragOut
18497      * @private
18498      */
18499     b4DragOut: function(e) { },
18500
18501     /**
18502      * Abstract method called when we are no longer hovering over an element
18503      * @method onDragOut
18504      * @param {Event} e the mousemove event
18505      * @param {String|DragDrop[]} id In POINT mode, the element
18506      * id this was hovering over.  In INTERSECT mode, an array of dd items
18507      * that the mouse is no longer over.
18508      */
18509     onDragOut: function(e, id) { /* override this */ },
18510
18511     /**
18512      * Code that executes immediately before the onDragDrop event
18513      * @method b4DragDrop
18514      * @private
18515      */
18516     b4DragDrop: function(e) { },
18517
18518     /**
18519      * Abstract method called when this item is dropped on another DragDrop
18520      * obj
18521      * @method onDragDrop
18522      * @param {Event} e the mouseup event
18523      * @param {String|DragDrop[]} id In POINT mode, the element
18524      * id this was dropped on.  In INTERSECT mode, an array of dd items this
18525      * was dropped on.
18526      */
18527     onDragDrop: function(e, id) { /* override this */ },
18528
18529     /**
18530      * Abstract method called when this item is dropped on an area with no
18531      * drop target
18532      * @method onInvalidDrop
18533      * @param {Event} e the mouseup event
18534      */
18535     onInvalidDrop: function(e) { /* override this */ },
18536
18537     /**
18538      * Code that executes immediately before the endDrag event
18539      * @method b4EndDrag
18540      * @private
18541      */
18542     b4EndDrag: function(e) { },
18543
18544     /**
18545      * Fired when we are done dragging the object
18546      * @method endDrag
18547      * @param {Event} e the mouseup event
18548      */
18549     endDrag: function(e) { /* override this */ },
18550
18551     /**
18552      * Code executed immediately before the onMouseDown event
18553      * @method b4MouseDown
18554      * @param {Event} e the mousedown event
18555      * @private
18556      */
18557     b4MouseDown: function(e) {  },
18558
18559     /**
18560      * Event handler that fires when a drag/drop obj gets a mousedown
18561      * @method onMouseDown
18562      * @param {Event} e the mousedown event
18563      */
18564     onMouseDown: function(e) { /* override this */ },
18565
18566     /**
18567      * Event handler that fires when a drag/drop obj gets a mouseup
18568      * @method onMouseUp
18569      * @param {Event} e the mouseup event
18570      */
18571     onMouseUp: function(e) { /* override this */ },
18572
18573     /**
18574      * Override the onAvailable method to do what is needed after the initial
18575      * position was determined.
18576      * @method onAvailable
18577      */
18578     onAvailable: function () {
18579     },
18580
18581     /*
18582      * Provides default constraint padding to "constrainTo" elements (defaults to {left: 0, right:0, top:0, bottom:0}).
18583      * @type Object
18584      */
18585     defaultPadding : {left:0, right:0, top:0, bottom:0},
18586
18587     /*
18588      * Initializes the drag drop object's constraints to restrict movement to a certain element.
18589  *
18590  * Usage:
18591  <pre><code>
18592  var dd = new Roo.dd.DDProxy("dragDiv1", "proxytest",
18593                 { dragElId: "existingProxyDiv" });
18594  dd.startDrag = function(){
18595      this.constrainTo("parent-id");
18596  };
18597  </code></pre>
18598  * Or you can initalize it using the {@link Roo.Element} object:
18599  <pre><code>
18600  Roo.get("dragDiv1").initDDProxy("proxytest", {dragElId: "existingProxyDiv"}, {
18601      startDrag : function(){
18602          this.constrainTo("parent-id");
18603      }
18604  });
18605  </code></pre>
18606      * @param {String/HTMLElement/Element} constrainTo The element to constrain to.
18607      * @param {Object/Number} pad (optional) Pad provides a way to specify "padding" of the constraints,
18608      * and can be either a number for symmetrical padding (4 would be equal to {left:4, right:4, top:4, bottom:4}) or
18609      * an object containing the sides to pad. For example: {right:10, bottom:10}
18610      * @param {Boolean} inContent (optional) Constrain the draggable in the content box of the element (inside padding and borders)
18611      */
18612     constrainTo : function(constrainTo, pad, inContent){
18613         if(typeof pad == "number"){
18614             pad = {left: pad, right:pad, top:pad, bottom:pad};
18615         }
18616         pad = pad || this.defaultPadding;
18617         var b = Roo.get(this.getEl()).getBox();
18618         var ce = Roo.get(constrainTo);
18619         var s = ce.getScroll();
18620         var c, cd = ce.dom;
18621         if(cd == document.body){
18622             c = { x: s.left, y: s.top, width: Roo.lib.Dom.getViewWidth(), height: Roo.lib.Dom.getViewHeight()};
18623         }else{
18624             xy = ce.getXY();
18625             c = {x : xy[0]+s.left, y: xy[1]+s.top, width: cd.clientWidth, height: cd.clientHeight};
18626         }
18627
18628
18629         var topSpace = b.y - c.y;
18630         var leftSpace = b.x - c.x;
18631
18632         this.resetConstraints();
18633         this.setXConstraint(leftSpace - (pad.left||0), // left
18634                 c.width - leftSpace - b.width - (pad.right||0) //right
18635         );
18636         this.setYConstraint(topSpace - (pad.top||0), //top
18637                 c.height - topSpace - b.height - (pad.bottom||0) //bottom
18638         );
18639     },
18640
18641     /**
18642      * Returns a reference to the linked element
18643      * @method getEl
18644      * @return {HTMLElement} the html element
18645      */
18646     getEl: function() {
18647         if (!this._domRef) {
18648             this._domRef = Roo.getDom(this.id);
18649         }
18650
18651         return this._domRef;
18652     },
18653
18654     /**
18655      * Returns a reference to the actual element to drag.  By default this is
18656      * the same as the html element, but it can be assigned to another
18657      * element. An example of this can be found in Roo.dd.DDProxy
18658      * @method getDragEl
18659      * @return {HTMLElement} the html element
18660      */
18661     getDragEl: function() {
18662         return Roo.getDom(this.dragElId);
18663     },
18664
18665     /**
18666      * Sets up the DragDrop object.  Must be called in the constructor of any
18667      * Roo.dd.DragDrop subclass
18668      * @method init
18669      * @param id the id of the linked element
18670      * @param {String} sGroup the group of related items
18671      * @param {object} config configuration attributes
18672      */
18673     init: function(id, sGroup, config) {
18674         this.initTarget(id, sGroup, config);
18675         if (!Roo.isTouch) {
18676             Event.on(this.id, "mousedown", this.handleMouseDown, this);
18677         }
18678         Event.on(this.id, "touchstart", this.handleMouseDown, this);
18679         // Event.on(this.id, "selectstart", Event.preventDefault);
18680     },
18681
18682     /**
18683      * Initializes Targeting functionality only... the object does not
18684      * get a mousedown handler.
18685      * @method initTarget
18686      * @param id the id of the linked element
18687      * @param {String} sGroup the group of related items
18688      * @param {object} config configuration attributes
18689      */
18690     initTarget: function(id, sGroup, config) {
18691
18692         // configuration attributes
18693         this.config = config || {};
18694
18695         // create a local reference to the drag and drop manager
18696         this.DDM = Roo.dd.DDM;
18697         // initialize the groups array
18698         this.groups = {};
18699
18700         // assume that we have an element reference instead of an id if the
18701         // parameter is not a string
18702         if (typeof id !== "string") {
18703             id = Roo.id(id);
18704         }
18705
18706         // set the id
18707         this.id = id;
18708
18709         // add to an interaction group
18710         this.addToGroup((sGroup) ? sGroup : "default");
18711
18712         // We don't want to register this as the handle with the manager
18713         // so we just set the id rather than calling the setter.
18714         this.handleElId = id;
18715
18716         // the linked element is the element that gets dragged by default
18717         this.setDragElId(id);
18718
18719         // by default, clicked anchors will not start drag operations.
18720         this.invalidHandleTypes = { A: "A" };
18721         this.invalidHandleIds = {};
18722         this.invalidHandleClasses = [];
18723
18724         this.applyConfig();
18725
18726         this.handleOnAvailable();
18727     },
18728
18729     /**
18730      * Applies the configuration parameters that were passed into the constructor.
18731      * This is supposed to happen at each level through the inheritance chain.  So
18732      * a DDProxy implentation will execute apply config on DDProxy, DD, and
18733      * DragDrop in order to get all of the parameters that are available in
18734      * each object.
18735      * @method applyConfig
18736      */
18737     applyConfig: function() {
18738
18739         // configurable properties:
18740         //    padding, isTarget, maintainOffset, primaryButtonOnly
18741         this.padding           = this.config.padding || [0, 0, 0, 0];
18742         this.isTarget          = (this.config.isTarget !== false);
18743         this.maintainOffset    = (this.config.maintainOffset);
18744         this.primaryButtonOnly = (this.config.primaryButtonOnly !== false);
18745
18746     },
18747
18748     /**
18749      * Executed when the linked element is available
18750      * @method handleOnAvailable
18751      * @private
18752      */
18753     handleOnAvailable: function() {
18754         this.available = true;
18755         this.resetConstraints();
18756         this.onAvailable();
18757     },
18758
18759      /**
18760      * Configures the padding for the target zone in px.  Effectively expands
18761      * (or reduces) the virtual object size for targeting calculations.
18762      * Supports css-style shorthand; if only one parameter is passed, all sides
18763      * will have that padding, and if only two are passed, the top and bottom
18764      * will have the first param, the left and right the second.
18765      * @method setPadding
18766      * @param {int} iTop    Top pad
18767      * @param {int} iRight  Right pad
18768      * @param {int} iBot    Bot pad
18769      * @param {int} iLeft   Left pad
18770      */
18771     setPadding: function(iTop, iRight, iBot, iLeft) {
18772         // this.padding = [iLeft, iRight, iTop, iBot];
18773         if (!iRight && 0 !== iRight) {
18774             this.padding = [iTop, iTop, iTop, iTop];
18775         } else if (!iBot && 0 !== iBot) {
18776             this.padding = [iTop, iRight, iTop, iRight];
18777         } else {
18778             this.padding = [iTop, iRight, iBot, iLeft];
18779         }
18780     },
18781
18782     /**
18783      * Stores the initial placement of the linked element.
18784      * @method setInitialPosition
18785      * @param {int} diffX   the X offset, default 0
18786      * @param {int} diffY   the Y offset, default 0
18787      */
18788     setInitPosition: function(diffX, diffY) {
18789         var el = this.getEl();
18790
18791         if (!this.DDM.verifyEl(el)) {
18792             return;
18793         }
18794
18795         var dx = diffX || 0;
18796         var dy = diffY || 0;
18797
18798         var p = Dom.getXY( el );
18799
18800         this.initPageX = p[0] - dx;
18801         this.initPageY = p[1] - dy;
18802
18803         this.lastPageX = p[0];
18804         this.lastPageY = p[1];
18805
18806
18807         this.setStartPosition(p);
18808     },
18809
18810     /**
18811      * Sets the start position of the element.  This is set when the obj
18812      * is initialized, the reset when a drag is started.
18813      * @method setStartPosition
18814      * @param pos current position (from previous lookup)
18815      * @private
18816      */
18817     setStartPosition: function(pos) {
18818         var p = pos || Dom.getXY( this.getEl() );
18819         this.deltaSetXY = null;
18820
18821         this.startPageX = p[0];
18822         this.startPageY = p[1];
18823     },
18824
18825     /**
18826      * Add this instance to a group of related drag/drop objects.  All
18827      * instances belong to at least one group, and can belong to as many
18828      * groups as needed.
18829      * @method addToGroup
18830      * @param sGroup {string} the name of the group
18831      */
18832     addToGroup: function(sGroup) {
18833         this.groups[sGroup] = true;
18834         this.DDM.regDragDrop(this, sGroup);
18835     },
18836
18837     /**
18838      * Remove's this instance from the supplied interaction group
18839      * @method removeFromGroup
18840      * @param {string}  sGroup  The group to drop
18841      */
18842     removeFromGroup: function(sGroup) {
18843         if (this.groups[sGroup]) {
18844             delete this.groups[sGroup];
18845         }
18846
18847         this.DDM.removeDDFromGroup(this, sGroup);
18848     },
18849
18850     /**
18851      * Allows you to specify that an element other than the linked element
18852      * will be moved with the cursor during a drag
18853      * @method setDragElId
18854      * @param id {string} the id of the element that will be used to initiate the drag
18855      */
18856     setDragElId: function(id) {
18857         this.dragElId = id;
18858     },
18859
18860     /**
18861      * Allows you to specify a child of the linked element that should be
18862      * used to initiate the drag operation.  An example of this would be if
18863      * you have a content div with text and links.  Clicking anywhere in the
18864      * content area would normally start the drag operation.  Use this method
18865      * to specify that an element inside of the content div is the element
18866      * that starts the drag operation.
18867      * @method setHandleElId
18868      * @param id {string} the id of the element that will be used to
18869      * initiate the drag.
18870      */
18871     setHandleElId: function(id) {
18872         if (typeof id !== "string") {
18873             id = Roo.id(id);
18874         }
18875         this.handleElId = id;
18876         this.DDM.regHandle(this.id, id);
18877     },
18878
18879     /**
18880      * Allows you to set an element outside of the linked element as a drag
18881      * handle
18882      * @method setOuterHandleElId
18883      * @param id the id of the element that will be used to initiate the drag
18884      */
18885     setOuterHandleElId: function(id) {
18886         if (typeof id !== "string") {
18887             id = Roo.id(id);
18888         }
18889         Event.on(id, "mousedown",
18890                 this.handleMouseDown, this);
18891         this.setHandleElId(id);
18892
18893         this.hasOuterHandles = true;
18894     },
18895
18896     /**
18897      * Remove all drag and drop hooks for this element
18898      * @method unreg
18899      */
18900     unreg: function() {
18901         Event.un(this.id, "mousedown",
18902                 this.handleMouseDown);
18903         Event.un(this.id, "touchstart",
18904                 this.handleMouseDown);
18905         this._domRef = null;
18906         this.DDM._remove(this);
18907     },
18908
18909     destroy : function(){
18910         this.unreg();
18911     },
18912
18913     /**
18914      * Returns true if this instance is locked, or the drag drop mgr is locked
18915      * (meaning that all drag/drop is disabled on the page.)
18916      * @method isLocked
18917      * @return {boolean} true if this obj or all drag/drop is locked, else
18918      * false
18919      */
18920     isLocked: function() {
18921         return (this.DDM.isLocked() || this.locked);
18922     },
18923
18924     /**
18925      * Fired when this object is clicked
18926      * @method handleMouseDown
18927      * @param {Event} e
18928      * @param {Roo.dd.DragDrop} oDD the clicked dd object (this dd obj)
18929      * @private
18930      */
18931     handleMouseDown: function(e, oDD){
18932      
18933         if (!Roo.isTouch && this.primaryButtonOnly && e.button != 0) {
18934             //Roo.log('not touch/ button !=0');
18935             return;
18936         }
18937         if (e.browserEvent.touches && e.browserEvent.touches.length != 1) {
18938             return; // double touch..
18939         }
18940         
18941
18942         if (this.isLocked()) {
18943             //Roo.log('locked');
18944             return;
18945         }
18946
18947         this.DDM.refreshCache(this.groups);
18948 //        Roo.log([Roo.lib.Event.getPageX(e), Roo.lib.Event.getPageY(e)]);
18949         var pt = new Roo.lib.Point(Roo.lib.Event.getPageX(e), Roo.lib.Event.getPageY(e));
18950         if (!this.hasOuterHandles && !this.DDM.isOverTarget(pt, this) )  {
18951             //Roo.log('no outer handes or not over target');
18952                 // do nothing.
18953         } else {
18954 //            Roo.log('check validator');
18955             if (this.clickValidator(e)) {
18956 //                Roo.log('validate success');
18957                 // set the initial element position
18958                 this.setStartPosition();
18959
18960
18961                 this.b4MouseDown(e);
18962                 this.onMouseDown(e);
18963
18964                 this.DDM.handleMouseDown(e, this);
18965
18966                 this.DDM.stopEvent(e);
18967             } else {
18968
18969
18970             }
18971         }
18972     },
18973
18974     clickValidator: function(e) {
18975         var target = e.getTarget();
18976         return ( this.isValidHandleChild(target) &&
18977                     (this.id == this.handleElId ||
18978                         this.DDM.handleWasClicked(target, this.id)) );
18979     },
18980
18981     /**
18982      * Allows you to specify a tag name that should not start a drag operation
18983      * when clicked.  This is designed to facilitate embedding links within a
18984      * drag handle that do something other than start the drag.
18985      * @method addInvalidHandleType
18986      * @param {string} tagName the type of element to exclude
18987      */
18988     addInvalidHandleType: function(tagName) {
18989         var type = tagName.toUpperCase();
18990         this.invalidHandleTypes[type] = type;
18991     },
18992
18993     /**
18994      * Lets you to specify an element id for a child of a drag handle
18995      * that should not initiate a drag
18996      * @method addInvalidHandleId
18997      * @param {string} id the element id of the element you wish to ignore
18998      */
18999     addInvalidHandleId: function(id) {
19000         if (typeof id !== "string") {
19001             id = Roo.id(id);
19002         }
19003         this.invalidHandleIds[id] = id;
19004     },
19005
19006     /**
19007      * Lets you specify a css class of elements that will not initiate a drag
19008      * @method addInvalidHandleClass
19009      * @param {string} cssClass the class of the elements you wish to ignore
19010      */
19011     addInvalidHandleClass: function(cssClass) {
19012         this.invalidHandleClasses.push(cssClass);
19013     },
19014
19015     /**
19016      * Unsets an excluded tag name set by addInvalidHandleType
19017      * @method removeInvalidHandleType
19018      * @param {string} tagName the type of element to unexclude
19019      */
19020     removeInvalidHandleType: function(tagName) {
19021         var type = tagName.toUpperCase();
19022         // this.invalidHandleTypes[type] = null;
19023         delete this.invalidHandleTypes[type];
19024     },
19025
19026     /**
19027      * Unsets an invalid handle id
19028      * @method removeInvalidHandleId
19029      * @param {string} id the id of the element to re-enable
19030      */
19031     removeInvalidHandleId: function(id) {
19032         if (typeof id !== "string") {
19033             id = Roo.id(id);
19034         }
19035         delete this.invalidHandleIds[id];
19036     },
19037
19038     /**
19039      * Unsets an invalid css class
19040      * @method removeInvalidHandleClass
19041      * @param {string} cssClass the class of the element(s) you wish to
19042      * re-enable
19043      */
19044     removeInvalidHandleClass: function(cssClass) {
19045         for (var i=0, len=this.invalidHandleClasses.length; i<len; ++i) {
19046             if (this.invalidHandleClasses[i] == cssClass) {
19047                 delete this.invalidHandleClasses[i];
19048             }
19049         }
19050     },
19051
19052     /**
19053      * Checks the tag exclusion list to see if this click should be ignored
19054      * @method isValidHandleChild
19055      * @param {HTMLElement} node the HTMLElement to evaluate
19056      * @return {boolean} true if this is a valid tag type, false if not
19057      */
19058     isValidHandleChild: function(node) {
19059
19060         var valid = true;
19061         // var n = (node.nodeName == "#text") ? node.parentNode : node;
19062         var nodeName;
19063         try {
19064             nodeName = node.nodeName.toUpperCase();
19065         } catch(e) {
19066             nodeName = node.nodeName;
19067         }
19068         valid = valid && !this.invalidHandleTypes[nodeName];
19069         valid = valid && !this.invalidHandleIds[node.id];
19070
19071         for (var i=0, len=this.invalidHandleClasses.length; valid && i<len; ++i) {
19072             valid = !Dom.hasClass(node, this.invalidHandleClasses[i]);
19073         }
19074
19075
19076         return valid;
19077
19078     },
19079
19080     /**
19081      * Create the array of horizontal tick marks if an interval was specified
19082      * in setXConstraint().
19083      * @method setXTicks
19084      * @private
19085      */
19086     setXTicks: function(iStartX, iTickSize) {
19087         this.xTicks = [];
19088         this.xTickSize = iTickSize;
19089
19090         var tickMap = {};
19091
19092         for (var i = this.initPageX; i >= this.minX; i = i - iTickSize) {
19093             if (!tickMap[i]) {
19094                 this.xTicks[this.xTicks.length] = i;
19095                 tickMap[i] = true;
19096             }
19097         }
19098
19099         for (i = this.initPageX; i <= this.maxX; i = i + iTickSize) {
19100             if (!tickMap[i]) {
19101                 this.xTicks[this.xTicks.length] = i;
19102                 tickMap[i] = true;
19103             }
19104         }
19105
19106         this.xTicks.sort(this.DDM.numericSort) ;
19107     },
19108
19109     /**
19110      * Create the array of vertical tick marks if an interval was specified in
19111      * setYConstraint().
19112      * @method setYTicks
19113      * @private
19114      */
19115     setYTicks: function(iStartY, iTickSize) {
19116         this.yTicks = [];
19117         this.yTickSize = iTickSize;
19118
19119         var tickMap = {};
19120
19121         for (var i = this.initPageY; i >= this.minY; i = i - iTickSize) {
19122             if (!tickMap[i]) {
19123                 this.yTicks[this.yTicks.length] = i;
19124                 tickMap[i] = true;
19125             }
19126         }
19127
19128         for (i = this.initPageY; i <= this.maxY; i = i + iTickSize) {
19129             if (!tickMap[i]) {
19130                 this.yTicks[this.yTicks.length] = i;
19131                 tickMap[i] = true;
19132             }
19133         }
19134
19135         this.yTicks.sort(this.DDM.numericSort) ;
19136     },
19137
19138     /**
19139      * By default, the element can be dragged any place on the screen.  Use
19140      * this method to limit the horizontal travel of the element.  Pass in
19141      * 0,0 for the parameters if you want to lock the drag to the y axis.
19142      * @method setXConstraint
19143      * @param {int} iLeft the number of pixels the element can move to the left
19144      * @param {int} iRight the number of pixels the element can move to the
19145      * right
19146      * @param {int} iTickSize optional parameter for specifying that the
19147      * element
19148      * should move iTickSize pixels at a time.
19149      */
19150     setXConstraint: function(iLeft, iRight, iTickSize) {
19151         this.leftConstraint = iLeft;
19152         this.rightConstraint = iRight;
19153
19154         this.minX = this.initPageX - iLeft;
19155         this.maxX = this.initPageX + iRight;
19156         if (iTickSize) { this.setXTicks(this.initPageX, iTickSize); }
19157
19158         this.constrainX = true;
19159     },
19160
19161     /**
19162      * Clears any constraints applied to this instance.  Also clears ticks
19163      * since they can't exist independent of a constraint at this time.
19164      * @method clearConstraints
19165      */
19166     clearConstraints: function() {
19167         this.constrainX = false;
19168         this.constrainY = false;
19169         this.clearTicks();
19170     },
19171
19172     /**
19173      * Clears any tick interval defined for this instance
19174      * @method clearTicks
19175      */
19176     clearTicks: function() {
19177         this.xTicks = null;
19178         this.yTicks = null;
19179         this.xTickSize = 0;
19180         this.yTickSize = 0;
19181     },
19182
19183     /**
19184      * By default, the element can be dragged any place on the screen.  Set
19185      * this to limit the vertical travel of the element.  Pass in 0,0 for the
19186      * parameters if you want to lock the drag to the x axis.
19187      * @method setYConstraint
19188      * @param {int} iUp the number of pixels the element can move up
19189      * @param {int} iDown the number of pixels the element can move down
19190      * @param {int} iTickSize optional parameter for specifying that the
19191      * element should move iTickSize pixels at a time.
19192      */
19193     setYConstraint: function(iUp, iDown, iTickSize) {
19194         this.topConstraint = iUp;
19195         this.bottomConstraint = iDown;
19196
19197         this.minY = this.initPageY - iUp;
19198         this.maxY = this.initPageY + iDown;
19199         if (iTickSize) { this.setYTicks(this.initPageY, iTickSize); }
19200
19201         this.constrainY = true;
19202
19203     },
19204
19205     /**
19206      * resetConstraints must be called if you manually reposition a dd element.
19207      * @method resetConstraints
19208      * @param {boolean} maintainOffset
19209      */
19210     resetConstraints: function() {
19211
19212
19213         // Maintain offsets if necessary
19214         if (this.initPageX || this.initPageX === 0) {
19215             // figure out how much this thing has moved
19216             var dx = (this.maintainOffset) ? this.lastPageX - this.initPageX : 0;
19217             var dy = (this.maintainOffset) ? this.lastPageY - this.initPageY : 0;
19218
19219             this.setInitPosition(dx, dy);
19220
19221         // This is the first time we have detected the element's position
19222         } else {
19223             this.setInitPosition();
19224         }
19225
19226         if (this.constrainX) {
19227             this.setXConstraint( this.leftConstraint,
19228                                  this.rightConstraint,
19229                                  this.xTickSize        );
19230         }
19231
19232         if (this.constrainY) {
19233             this.setYConstraint( this.topConstraint,
19234                                  this.bottomConstraint,
19235                                  this.yTickSize         );
19236         }
19237     },
19238
19239     /**
19240      * Normally the drag element is moved pixel by pixel, but we can specify
19241      * that it move a number of pixels at a time.  This method resolves the
19242      * location when we have it set up like this.
19243      * @method getTick
19244      * @param {int} val where we want to place the object
19245      * @param {int[]} tickArray sorted array of valid points
19246      * @return {int} the closest tick
19247      * @private
19248      */
19249     getTick: function(val, tickArray) {
19250
19251         if (!tickArray) {
19252             // If tick interval is not defined, it is effectively 1 pixel,
19253             // so we return the value passed to us.
19254             return val;
19255         } else if (tickArray[0] >= val) {
19256             // The value is lower than the first tick, so we return the first
19257             // tick.
19258             return tickArray[0];
19259         } else {
19260             for (var i=0, len=tickArray.length; i<len; ++i) {
19261                 var next = i + 1;
19262                 if (tickArray[next] && tickArray[next] >= val) {
19263                     var diff1 = val - tickArray[i];
19264                     var diff2 = tickArray[next] - val;
19265                     return (diff2 > diff1) ? tickArray[i] : tickArray[next];
19266                 }
19267             }
19268
19269             // The value is larger than the last tick, so we return the last
19270             // tick.
19271             return tickArray[tickArray.length - 1];
19272         }
19273     },
19274
19275     /**
19276      * toString method
19277      * @method toString
19278      * @return {string} string representation of the dd obj
19279      */
19280     toString: function() {
19281         return ("DragDrop " + this.id);
19282     }
19283
19284 });
19285
19286 })();
19287 /*
19288  * Based on:
19289  * Ext JS Library 1.1.1
19290  * Copyright(c) 2006-2007, Ext JS, LLC.
19291  *
19292  * Originally Released Under LGPL - original licence link has changed is not relivant.
19293  *
19294  * Fork - LGPL
19295  * <script type="text/javascript">
19296  */
19297
19298
19299 /**
19300  * The drag and drop utility provides a framework for building drag and drop
19301  * applications.  In addition to enabling drag and drop for specific elements,
19302  * the drag and drop elements are tracked by the manager class, and the
19303  * interactions between the various elements are tracked during the drag and
19304  * the implementing code is notified about these important moments.
19305  */
19306
19307 // Only load the library once.  Rewriting the manager class would orphan
19308 // existing drag and drop instances.
19309 if (!Roo.dd.DragDropMgr) {
19310
19311 /**
19312  * @class Roo.dd.DragDropMgr
19313  * DragDropMgr is a singleton that tracks the element interaction for
19314  * all DragDrop items in the window.  Generally, you will not call
19315  * this class directly, but it does have helper methods that could
19316  * be useful in your DragDrop implementations.
19317  * @singleton
19318  */
19319 Roo.dd.DragDropMgr = function() {
19320
19321     var Event = Roo.EventManager;
19322
19323     return {
19324
19325         /**
19326          * Two dimensional Array of registered DragDrop objects.  The first
19327          * dimension is the DragDrop item group, the second the DragDrop
19328          * object.
19329          * @property ids
19330          * @type {string: string}
19331          * @private
19332          * @static
19333          */
19334         ids: {},
19335
19336         /**
19337          * Array of element ids defined as drag handles.  Used to determine
19338          * if the element that generated the mousedown event is actually the
19339          * handle and not the html element itself.
19340          * @property handleIds
19341          * @type {string: string}
19342          * @private
19343          * @static
19344          */
19345         handleIds: {},
19346
19347         /**
19348          * the DragDrop object that is currently being dragged
19349          * @property dragCurrent
19350          * @type DragDrop
19351          * @private
19352          * @static
19353          **/
19354         dragCurrent: null,
19355
19356         /**
19357          * the DragDrop object(s) that are being hovered over
19358          * @property dragOvers
19359          * @type Array
19360          * @private
19361          * @static
19362          */
19363         dragOvers: {},
19364
19365         /**
19366          * the X distance between the cursor and the object being dragged
19367          * @property deltaX
19368          * @type int
19369          * @private
19370          * @static
19371          */
19372         deltaX: 0,
19373
19374         /**
19375          * the Y distance between the cursor and the object being dragged
19376          * @property deltaY
19377          * @type int
19378          * @private
19379          * @static
19380          */
19381         deltaY: 0,
19382
19383         /**
19384          * Flag to determine if we should prevent the default behavior of the
19385          * events we define. By default this is true, but this can be set to
19386          * false if you need the default behavior (not recommended)
19387          * @property preventDefault
19388          * @type boolean
19389          * @static
19390          */
19391         preventDefault: true,
19392
19393         /**
19394          * Flag to determine if we should stop the propagation of the events
19395          * we generate. This is true by default but you may want to set it to
19396          * false if the html element contains other features that require the
19397          * mouse click.
19398          * @property stopPropagation
19399          * @type boolean
19400          * @static
19401          */
19402         stopPropagation: true,
19403
19404         /**
19405          * Internal flag that is set to true when drag and drop has been
19406          * intialized
19407          * @property initialized
19408          * @private
19409          * @static
19410          */
19411         initalized: false,
19412
19413         /**
19414          * All drag and drop can be disabled.
19415          * @property locked
19416          * @private
19417          * @static
19418          */
19419         locked: false,
19420
19421         /**
19422          * Called the first time an element is registered.
19423          * @method init
19424          * @private
19425          * @static
19426          */
19427         init: function() {
19428             this.initialized = true;
19429         },
19430
19431         /**
19432          * In point mode, drag and drop interaction is defined by the
19433          * location of the cursor during the drag/drop
19434          * @property POINT
19435          * @type int
19436          * @static
19437          */
19438         POINT: 0,
19439
19440         /**
19441          * In intersect mode, drag and drop interactio nis defined by the
19442          * overlap of two or more drag and drop objects.
19443          * @property INTERSECT
19444          * @type int
19445          * @static
19446          */
19447         INTERSECT: 1,
19448
19449         /**
19450          * The current drag and drop mode.  Default: POINT
19451          * @property mode
19452          * @type int
19453          * @static
19454          */
19455         mode: 0,
19456
19457         /**
19458          * Runs method on all drag and drop objects
19459          * @method _execOnAll
19460          * @private
19461          * @static
19462          */
19463         _execOnAll: function(sMethod, args) {
19464             for (var i in this.ids) {
19465                 for (var j in this.ids[i]) {
19466                     var oDD = this.ids[i][j];
19467                     if (! this.isTypeOfDD(oDD)) {
19468                         continue;
19469                     }
19470                     oDD[sMethod].apply(oDD, args);
19471                 }
19472             }
19473         },
19474
19475         /**
19476          * Drag and drop initialization.  Sets up the global event handlers
19477          * @method _onLoad
19478          * @private
19479          * @static
19480          */
19481         _onLoad: function() {
19482
19483             this.init();
19484
19485             if (!Roo.isTouch) {
19486                 Event.on(document, "mouseup",   this.handleMouseUp, this, true);
19487                 Event.on(document, "mousemove", this.handleMouseMove, this, true);
19488             }
19489             Event.on(document, "touchend",   this.handleMouseUp, this, true);
19490             Event.on(document, "touchmove", this.handleMouseMove, this, true);
19491             
19492             Event.on(window,   "unload",    this._onUnload, this, true);
19493             Event.on(window,   "resize",    this._onResize, this, true);
19494             // Event.on(window,   "mouseout",    this._test);
19495
19496         },
19497
19498         /**
19499          * Reset constraints on all drag and drop objs
19500          * @method _onResize
19501          * @private
19502          * @static
19503          */
19504         _onResize: function(e) {
19505             this._execOnAll("resetConstraints", []);
19506         },
19507
19508         /**
19509          * Lock all drag and drop functionality
19510          * @method lock
19511          * @static
19512          */
19513         lock: function() { this.locked = true; },
19514
19515         /**
19516          * Unlock all drag and drop functionality
19517          * @method unlock
19518          * @static
19519          */
19520         unlock: function() { this.locked = false; },
19521
19522         /**
19523          * Is drag and drop locked?
19524          * @method isLocked
19525          * @return {boolean} True if drag and drop is locked, false otherwise.
19526          * @static
19527          */
19528         isLocked: function() { return this.locked; },
19529
19530         /**
19531          * Location cache that is set for all drag drop objects when a drag is
19532          * initiated, cleared when the drag is finished.
19533          * @property locationCache
19534          * @private
19535          * @static
19536          */
19537         locationCache: {},
19538
19539         /**
19540          * Set useCache to false if you want to force object the lookup of each
19541          * drag and drop linked element constantly during a drag.
19542          * @property useCache
19543          * @type boolean
19544          * @static
19545          */
19546         useCache: true,
19547
19548         /**
19549          * The number of pixels that the mouse needs to move after the
19550          * mousedown before the drag is initiated.  Default=3;
19551          * @property clickPixelThresh
19552          * @type int
19553          * @static
19554          */
19555         clickPixelThresh: 3,
19556
19557         /**
19558          * The number of milliseconds after the mousedown event to initiate the
19559          * drag if we don't get a mouseup event. Default=1000
19560          * @property clickTimeThresh
19561          * @type int
19562          * @static
19563          */
19564         clickTimeThresh: 350,
19565
19566         /**
19567          * Flag that indicates that either the drag pixel threshold or the
19568          * mousdown time threshold has been met
19569          * @property dragThreshMet
19570          * @type boolean
19571          * @private
19572          * @static
19573          */
19574         dragThreshMet: false,
19575
19576         /**
19577          * Timeout used for the click time threshold
19578          * @property clickTimeout
19579          * @type Object
19580          * @private
19581          * @static
19582          */
19583         clickTimeout: null,
19584
19585         /**
19586          * The X position of the mousedown event stored for later use when a
19587          * drag threshold is met.
19588          * @property startX
19589          * @type int
19590          * @private
19591          * @static
19592          */
19593         startX: 0,
19594
19595         /**
19596          * The Y position of the mousedown event stored for later use when a
19597          * drag threshold is met.
19598          * @property startY
19599          * @type int
19600          * @private
19601          * @static
19602          */
19603         startY: 0,
19604
19605         /**
19606          * Each DragDrop instance must be registered with the DragDropMgr.
19607          * This is executed in DragDrop.init()
19608          * @method regDragDrop
19609          * @param {DragDrop} oDD the DragDrop object to register
19610          * @param {String} sGroup the name of the group this element belongs to
19611          * @static
19612          */
19613         regDragDrop: function(oDD, sGroup) {
19614             if (!this.initialized) { this.init(); }
19615
19616             if (!this.ids[sGroup]) {
19617                 this.ids[sGroup] = {};
19618             }
19619             this.ids[sGroup][oDD.id] = oDD;
19620         },
19621
19622         /**
19623          * Removes the supplied dd instance from the supplied group. Executed
19624          * by DragDrop.removeFromGroup, so don't call this function directly.
19625          * @method removeDDFromGroup
19626          * @private
19627          * @static
19628          */
19629         removeDDFromGroup: function(oDD, sGroup) {
19630             if (!this.ids[sGroup]) {
19631                 this.ids[sGroup] = {};
19632             }
19633
19634             var obj = this.ids[sGroup];
19635             if (obj && obj[oDD.id]) {
19636                 delete obj[oDD.id];
19637             }
19638         },
19639
19640         /**
19641          * Unregisters a drag and drop item.  This is executed in
19642          * DragDrop.unreg, use that method instead of calling this directly.
19643          * @method _remove
19644          * @private
19645          * @static
19646          */
19647         _remove: function(oDD) {
19648             for (var g in oDD.groups) {
19649                 if (g && this.ids[g][oDD.id]) {
19650                     delete this.ids[g][oDD.id];
19651                 }
19652             }
19653             delete this.handleIds[oDD.id];
19654         },
19655
19656         /**
19657          * Each DragDrop handle element must be registered.  This is done
19658          * automatically when executing DragDrop.setHandleElId()
19659          * @method regHandle
19660          * @param {String} sDDId the DragDrop id this element is a handle for
19661          * @param {String} sHandleId the id of the element that is the drag
19662          * handle
19663          * @static
19664          */
19665         regHandle: function(sDDId, sHandleId) {
19666             if (!this.handleIds[sDDId]) {
19667                 this.handleIds[sDDId] = {};
19668             }
19669             this.handleIds[sDDId][sHandleId] = sHandleId;
19670         },
19671
19672         /**
19673          * Utility function to determine if a given element has been
19674          * registered as a drag drop item.
19675          * @method isDragDrop
19676          * @param {String} id the element id to check
19677          * @return {boolean} true if this element is a DragDrop item,
19678          * false otherwise
19679          * @static
19680          */
19681         isDragDrop: function(id) {
19682             return ( this.getDDById(id) ) ? true : false;
19683         },
19684
19685         /**
19686          * Returns the drag and drop instances that are in all groups the
19687          * passed in instance belongs to.
19688          * @method getRelated
19689          * @param {DragDrop} p_oDD the obj to get related data for
19690          * @param {boolean} bTargetsOnly if true, only return targetable objs
19691          * @return {DragDrop[]} the related instances
19692          * @static
19693          */
19694         getRelated: function(p_oDD, bTargetsOnly) {
19695             var oDDs = [];
19696             for (var i in p_oDD.groups) {
19697                 for (j in this.ids[i]) {
19698                     var dd = this.ids[i][j];
19699                     if (! this.isTypeOfDD(dd)) {
19700                         continue;
19701                     }
19702                     if (!bTargetsOnly || dd.isTarget) {
19703                         oDDs[oDDs.length] = dd;
19704                     }
19705                 }
19706             }
19707
19708             return oDDs;
19709         },
19710
19711         /**
19712          * Returns true if the specified dd target is a legal target for
19713          * the specifice drag obj
19714          * @method isLegalTarget
19715          * @param {DragDrop} the drag obj
19716          * @param {DragDrop} the target
19717          * @return {boolean} true if the target is a legal target for the
19718          * dd obj
19719          * @static
19720          */
19721         isLegalTarget: function (oDD, oTargetDD) {
19722             var targets = this.getRelated(oDD, true);
19723             for (var i=0, len=targets.length;i<len;++i) {
19724                 if (targets[i].id == oTargetDD.id) {
19725                     return true;
19726                 }
19727             }
19728
19729             return false;
19730         },
19731
19732         /**
19733          * My goal is to be able to transparently determine if an object is
19734          * typeof DragDrop, and the exact subclass of DragDrop.  typeof
19735          * returns "object", oDD.constructor.toString() always returns
19736          * "DragDrop" and not the name of the subclass.  So for now it just
19737          * evaluates a well-known variable in DragDrop.
19738          * @method isTypeOfDD
19739          * @param {Object} the object to evaluate
19740          * @return {boolean} true if typeof oDD = DragDrop
19741          * @static
19742          */
19743         isTypeOfDD: function (oDD) {
19744             return (oDD && oDD.__ygDragDrop);
19745         },
19746
19747         /**
19748          * Utility function to determine if a given element has been
19749          * registered as a drag drop handle for the given Drag Drop object.
19750          * @method isHandle
19751          * @param {String} id the element id to check
19752          * @return {boolean} true if this element is a DragDrop handle, false
19753          * otherwise
19754          * @static
19755          */
19756         isHandle: function(sDDId, sHandleId) {
19757             return ( this.handleIds[sDDId] &&
19758                             this.handleIds[sDDId][sHandleId] );
19759         },
19760
19761         /**
19762          * Returns the DragDrop instance for a given id
19763          * @method getDDById
19764          * @param {String} id the id of the DragDrop object
19765          * @return {DragDrop} the drag drop object, null if it is not found
19766          * @static
19767          */
19768         getDDById: function(id) {
19769             for (var i in this.ids) {
19770                 if (this.ids[i][id]) {
19771                     return this.ids[i][id];
19772                 }
19773             }
19774             return null;
19775         },
19776
19777         /**
19778          * Fired after a registered DragDrop object gets the mousedown event.
19779          * Sets up the events required to track the object being dragged
19780          * @method handleMouseDown
19781          * @param {Event} e the event
19782          * @param oDD the DragDrop object being dragged
19783          * @private
19784          * @static
19785          */
19786         handleMouseDown: function(e, oDD) {
19787             if(Roo.QuickTips){
19788                 Roo.QuickTips.disable();
19789             }
19790             this.currentTarget = e.getTarget();
19791
19792             this.dragCurrent = oDD;
19793
19794             var el = oDD.getEl();
19795
19796             // track start position
19797             this.startX = e.getPageX();
19798             this.startY = e.getPageY();
19799
19800             this.deltaX = this.startX - el.offsetLeft;
19801             this.deltaY = this.startY - el.offsetTop;
19802
19803             this.dragThreshMet = false;
19804
19805             this.clickTimeout = setTimeout(
19806                     function() {
19807                         var DDM = Roo.dd.DDM;
19808                         DDM.startDrag(DDM.startX, DDM.startY);
19809                     },
19810                     this.clickTimeThresh );
19811         },
19812
19813         /**
19814          * Fired when either the drag pixel threshol or the mousedown hold
19815          * time threshold has been met.
19816          * @method startDrag
19817          * @param x {int} the X position of the original mousedown
19818          * @param y {int} the Y position of the original mousedown
19819          * @static
19820          */
19821         startDrag: function(x, y) {
19822             clearTimeout(this.clickTimeout);
19823             if (this.dragCurrent) {
19824                 this.dragCurrent.b4StartDrag(x, y);
19825                 this.dragCurrent.startDrag(x, y);
19826             }
19827             this.dragThreshMet = true;
19828         },
19829
19830         /**
19831          * Internal function to handle the mouseup event.  Will be invoked
19832          * from the context of the document.
19833          * @method handleMouseUp
19834          * @param {Event} e the event
19835          * @private
19836          * @static
19837          */
19838         handleMouseUp: function(e) {
19839
19840             if(Roo.QuickTips){
19841                 Roo.QuickTips.enable();
19842             }
19843             if (! this.dragCurrent) {
19844                 return;
19845             }
19846
19847             clearTimeout(this.clickTimeout);
19848
19849             if (this.dragThreshMet) {
19850                 this.fireEvents(e, true);
19851             } else {
19852             }
19853
19854             this.stopDrag(e);
19855
19856             this.stopEvent(e);
19857         },
19858
19859         /**
19860          * Utility to stop event propagation and event default, if these
19861          * features are turned on.
19862          * @method stopEvent
19863          * @param {Event} e the event as returned by this.getEvent()
19864          * @static
19865          */
19866         stopEvent: function(e){
19867             if(this.stopPropagation) {
19868                 e.stopPropagation();
19869             }
19870
19871             if (this.preventDefault) {
19872                 e.preventDefault();
19873             }
19874         },
19875
19876         /**
19877          * Internal function to clean up event handlers after the drag
19878          * operation is complete
19879          * @method stopDrag
19880          * @param {Event} e the event
19881          * @private
19882          * @static
19883          */
19884         stopDrag: function(e) {
19885             // Fire the drag end event for the item that was dragged
19886             if (this.dragCurrent) {
19887                 if (this.dragThreshMet) {
19888                     this.dragCurrent.b4EndDrag(e);
19889                     this.dragCurrent.endDrag(e);
19890                 }
19891
19892                 this.dragCurrent.onMouseUp(e);
19893             }
19894
19895             this.dragCurrent = null;
19896             this.dragOvers = {};
19897         },
19898
19899         /**
19900          * Internal function to handle the mousemove event.  Will be invoked
19901          * from the context of the html element.
19902          *
19903          * @TODO figure out what we can do about mouse events lost when the
19904          * user drags objects beyond the window boundary.  Currently we can
19905          * detect this in internet explorer by verifying that the mouse is
19906          * down during the mousemove event.  Firefox doesn't give us the
19907          * button state on the mousemove event.
19908          * @method handleMouseMove
19909          * @param {Event} e the event
19910          * @private
19911          * @static
19912          */
19913         handleMouseMove: function(e) {
19914             if (! this.dragCurrent) {
19915                 return true;
19916             }
19917
19918             // var button = e.which || e.button;
19919
19920             // check for IE mouseup outside of page boundary
19921             if (Roo.isIE && (e.button !== 0 && e.button !== 1 && e.button !== 2)) {
19922                 this.stopEvent(e);
19923                 return this.handleMouseUp(e);
19924             }
19925
19926             if (!this.dragThreshMet) {
19927                 var diffX = Math.abs(this.startX - e.getPageX());
19928                 var diffY = Math.abs(this.startY - e.getPageY());
19929                 if (diffX > this.clickPixelThresh ||
19930                             diffY > this.clickPixelThresh) {
19931                     this.startDrag(this.startX, this.startY);
19932                 }
19933             }
19934
19935             if (this.dragThreshMet) {
19936                 this.dragCurrent.b4Drag(e);
19937                 this.dragCurrent.onDrag(e);
19938                 if(!this.dragCurrent.moveOnly){
19939                     this.fireEvents(e, false);
19940                 }
19941             }
19942
19943             this.stopEvent(e);
19944
19945             return true;
19946         },
19947
19948         /**
19949          * Iterates over all of the DragDrop elements to find ones we are
19950          * hovering over or dropping on
19951          * @method fireEvents
19952          * @param {Event} e the event
19953          * @param {boolean} isDrop is this a drop op or a mouseover op?
19954          * @private
19955          * @static
19956          */
19957         fireEvents: function(e, isDrop) {
19958             var dc = this.dragCurrent;
19959
19960             // If the user did the mouse up outside of the window, we could
19961             // get here even though we have ended the drag.
19962             if (!dc || dc.isLocked()) {
19963                 return;
19964             }
19965
19966             var pt = e.getPoint();
19967
19968             // cache the previous dragOver array
19969             var oldOvers = [];
19970
19971             var outEvts   = [];
19972             var overEvts  = [];
19973             var dropEvts  = [];
19974             var enterEvts = [];
19975
19976             // Check to see if the object(s) we were hovering over is no longer
19977             // being hovered over so we can fire the onDragOut event
19978             for (var i in this.dragOvers) {
19979
19980                 var ddo = this.dragOvers[i];
19981
19982                 if (! this.isTypeOfDD(ddo)) {
19983                     continue;
19984                 }
19985
19986                 if (! this.isOverTarget(pt, ddo, this.mode)) {
19987                     outEvts.push( ddo );
19988                 }
19989
19990                 oldOvers[i] = true;
19991                 delete this.dragOvers[i];
19992             }
19993
19994             for (var sGroup in dc.groups) {
19995
19996                 if ("string" != typeof sGroup) {
19997                     continue;
19998                 }
19999
20000                 for (i in this.ids[sGroup]) {
20001                     var oDD = this.ids[sGroup][i];
20002                     if (! this.isTypeOfDD(oDD)) {
20003                         continue;
20004                     }
20005
20006                     if (oDD.isTarget && !oDD.isLocked() && oDD != dc) {
20007                         if (this.isOverTarget(pt, oDD, this.mode)) {
20008                             // look for drop interactions
20009                             if (isDrop) {
20010                                 dropEvts.push( oDD );
20011                             // look for drag enter and drag over interactions
20012                             } else {
20013
20014                                 // initial drag over: dragEnter fires
20015                                 if (!oldOvers[oDD.id]) {
20016                                     enterEvts.push( oDD );
20017                                 // subsequent drag overs: dragOver fires
20018                                 } else {
20019                                     overEvts.push( oDD );
20020                                 }
20021
20022                                 this.dragOvers[oDD.id] = oDD;
20023                             }
20024                         }
20025                     }
20026                 }
20027             }
20028
20029             if (this.mode) {
20030                 if (outEvts.length) {
20031                     dc.b4DragOut(e, outEvts);
20032                     dc.onDragOut(e, outEvts);
20033                 }
20034
20035                 if (enterEvts.length) {
20036                     dc.onDragEnter(e, enterEvts);
20037                 }
20038
20039                 if (overEvts.length) {
20040                     dc.b4DragOver(e, overEvts);
20041                     dc.onDragOver(e, overEvts);
20042                 }
20043
20044                 if (dropEvts.length) {
20045                     dc.b4DragDrop(e, dropEvts);
20046                     dc.onDragDrop(e, dropEvts);
20047                 }
20048
20049             } else {
20050                 // fire dragout events
20051                 var len = 0;
20052                 for (i=0, len=outEvts.length; i<len; ++i) {
20053                     dc.b4DragOut(e, outEvts[i].id);
20054                     dc.onDragOut(e, outEvts[i].id);
20055                 }
20056
20057                 // fire enter events
20058                 for (i=0,len=enterEvts.length; i<len; ++i) {
20059                     // dc.b4DragEnter(e, oDD.id);
20060                     dc.onDragEnter(e, enterEvts[i].id);
20061                 }
20062
20063                 // fire over events
20064                 for (i=0,len=overEvts.length; i<len; ++i) {
20065                     dc.b4DragOver(e, overEvts[i].id);
20066                     dc.onDragOver(e, overEvts[i].id);
20067                 }
20068
20069                 // fire drop events
20070                 for (i=0, len=dropEvts.length; i<len; ++i) {
20071                     dc.b4DragDrop(e, dropEvts[i].id);
20072                     dc.onDragDrop(e, dropEvts[i].id);
20073                 }
20074
20075             }
20076
20077             // notify about a drop that did not find a target
20078             if (isDrop && !dropEvts.length) {
20079                 dc.onInvalidDrop(e);
20080             }
20081
20082         },
20083
20084         /**
20085          * Helper function for getting the best match from the list of drag
20086          * and drop objects returned by the drag and drop events when we are
20087          * in INTERSECT mode.  It returns either the first object that the
20088          * cursor is over, or the object that has the greatest overlap with
20089          * the dragged element.
20090          * @method getBestMatch
20091          * @param  {DragDrop[]} dds The array of drag and drop objects
20092          * targeted
20093          * @return {DragDrop}       The best single match
20094          * @static
20095          */
20096         getBestMatch: function(dds) {
20097             var winner = null;
20098             // Return null if the input is not what we expect
20099             //if (!dds || !dds.length || dds.length == 0) {
20100                // winner = null;
20101             // If there is only one item, it wins
20102             //} else if (dds.length == 1) {
20103
20104             var len = dds.length;
20105
20106             if (len == 1) {
20107                 winner = dds[0];
20108             } else {
20109                 // Loop through the targeted items
20110                 for (var i=0; i<len; ++i) {
20111                     var dd = dds[i];
20112                     // If the cursor is over the object, it wins.  If the
20113                     // cursor is over multiple matches, the first one we come
20114                     // to wins.
20115                     if (dd.cursorIsOver) {
20116                         winner = dd;
20117                         break;
20118                     // Otherwise the object with the most overlap wins
20119                     } else {
20120                         if (!winner ||
20121                             winner.overlap.getArea() < dd.overlap.getArea()) {
20122                             winner = dd;
20123                         }
20124                     }
20125                 }
20126             }
20127
20128             return winner;
20129         },
20130
20131         /**
20132          * Refreshes the cache of the top-left and bottom-right points of the
20133          * drag and drop objects in the specified group(s).  This is in the
20134          * format that is stored in the drag and drop instance, so typical
20135          * usage is:
20136          * <code>
20137          * Roo.dd.DragDropMgr.refreshCache(ddinstance.groups);
20138          * </code>
20139          * Alternatively:
20140          * <code>
20141          * Roo.dd.DragDropMgr.refreshCache({group1:true, group2:true});
20142          * </code>
20143          * @TODO this really should be an indexed array.  Alternatively this
20144          * method could accept both.
20145          * @method refreshCache
20146          * @param {Object} groups an associative array of groups to refresh
20147          * @static
20148          */
20149         refreshCache: function(groups) {
20150             for (var sGroup in groups) {
20151                 if ("string" != typeof sGroup) {
20152                     continue;
20153                 }
20154                 for (var i in this.ids[sGroup]) {
20155                     var oDD = this.ids[sGroup][i];
20156
20157                     if (this.isTypeOfDD(oDD)) {
20158                     // if (this.isTypeOfDD(oDD) && oDD.isTarget) {
20159                         var loc = this.getLocation(oDD);
20160                         if (loc) {
20161                             this.locationCache[oDD.id] = loc;
20162                         } else {
20163                             delete this.locationCache[oDD.id];
20164                             // this will unregister the drag and drop object if
20165                             // the element is not in a usable state
20166                             // oDD.unreg();
20167                         }
20168                     }
20169                 }
20170             }
20171         },
20172
20173         /**
20174          * This checks to make sure an element exists and is in the DOM.  The
20175          * main purpose is to handle cases where innerHTML is used to remove
20176          * drag and drop objects from the DOM.  IE provides an 'unspecified
20177          * error' when trying to access the offsetParent of such an element
20178          * @method verifyEl
20179          * @param {HTMLElement} el the element to check
20180          * @return {boolean} true if the element looks usable
20181          * @static
20182          */
20183         verifyEl: function(el) {
20184             if (el) {
20185                 var parent;
20186                 if(Roo.isIE){
20187                     try{
20188                         parent = el.offsetParent;
20189                     }catch(e){}
20190                 }else{
20191                     parent = el.offsetParent;
20192                 }
20193                 if (parent) {
20194                     return true;
20195                 }
20196             }
20197
20198             return false;
20199         },
20200
20201         /**
20202          * Returns a Region object containing the drag and drop element's position
20203          * and size, including the padding configured for it
20204          * @method getLocation
20205          * @param {DragDrop} oDD the drag and drop object to get the
20206          *                       location for
20207          * @return {Roo.lib.Region} a Region object representing the total area
20208          *                             the element occupies, including any padding
20209          *                             the instance is configured for.
20210          * @static
20211          */
20212         getLocation: function(oDD) {
20213             if (! this.isTypeOfDD(oDD)) {
20214                 return null;
20215             }
20216
20217             var el = oDD.getEl(), pos, x1, x2, y1, y2, t, r, b, l;
20218
20219             try {
20220                 pos= Roo.lib.Dom.getXY(el);
20221             } catch (e) { }
20222
20223             if (!pos) {
20224                 return null;
20225             }
20226
20227             x1 = pos[0];
20228             x2 = x1 + el.offsetWidth;
20229             y1 = pos[1];
20230             y2 = y1 + el.offsetHeight;
20231
20232             t = y1 - oDD.padding[0];
20233             r = x2 + oDD.padding[1];
20234             b = y2 + oDD.padding[2];
20235             l = x1 - oDD.padding[3];
20236
20237             return new Roo.lib.Region( t, r, b, l );
20238         },
20239
20240         /**
20241          * Checks the cursor location to see if it over the target
20242          * @method isOverTarget
20243          * @param {Roo.lib.Point} pt The point to evaluate
20244          * @param {DragDrop} oTarget the DragDrop object we are inspecting
20245          * @return {boolean} true if the mouse is over the target
20246          * @private
20247          * @static
20248          */
20249         isOverTarget: function(pt, oTarget, intersect) {
20250             // use cache if available
20251             var loc = this.locationCache[oTarget.id];
20252             if (!loc || !this.useCache) {
20253                 loc = this.getLocation(oTarget);
20254                 this.locationCache[oTarget.id] = loc;
20255
20256             }
20257
20258             if (!loc) {
20259                 return false;
20260             }
20261
20262             oTarget.cursorIsOver = loc.contains( pt );
20263
20264             // DragDrop is using this as a sanity check for the initial mousedown
20265             // in this case we are done.  In POINT mode, if the drag obj has no
20266             // contraints, we are also done. Otherwise we need to evaluate the
20267             // location of the target as related to the actual location of the
20268             // dragged element.
20269             var dc = this.dragCurrent;
20270             if (!dc || !dc.getTargetCoord ||
20271                     (!intersect && !dc.constrainX && !dc.constrainY)) {
20272                 return oTarget.cursorIsOver;
20273             }
20274
20275             oTarget.overlap = null;
20276
20277             // Get the current location of the drag element, this is the
20278             // location of the mouse event less the delta that represents
20279             // where the original mousedown happened on the element.  We
20280             // need to consider constraints and ticks as well.
20281             var pos = dc.getTargetCoord(pt.x, pt.y);
20282
20283             var el = dc.getDragEl();
20284             var curRegion = new Roo.lib.Region( pos.y,
20285                                                    pos.x + el.offsetWidth,
20286                                                    pos.y + el.offsetHeight,
20287                                                    pos.x );
20288
20289             var overlap = curRegion.intersect(loc);
20290
20291             if (overlap) {
20292                 oTarget.overlap = overlap;
20293                 return (intersect) ? true : oTarget.cursorIsOver;
20294             } else {
20295                 return false;
20296             }
20297         },
20298
20299         /**
20300          * unload event handler
20301          * @method _onUnload
20302          * @private
20303          * @static
20304          */
20305         _onUnload: function(e, me) {
20306             Roo.dd.DragDropMgr.unregAll();
20307         },
20308
20309         /**
20310          * Cleans up the drag and drop events and objects.
20311          * @method unregAll
20312          * @private
20313          * @static
20314          */
20315         unregAll: function() {
20316
20317             if (this.dragCurrent) {
20318                 this.stopDrag();
20319                 this.dragCurrent = null;
20320             }
20321
20322             this._execOnAll("unreg", []);
20323
20324             for (i in this.elementCache) {
20325                 delete this.elementCache[i];
20326             }
20327
20328             this.elementCache = {};
20329             this.ids = {};
20330         },
20331
20332         /**
20333          * A cache of DOM elements
20334          * @property elementCache
20335          * @private
20336          * @static
20337          */
20338         elementCache: {},
20339
20340         /**
20341          * Get the wrapper for the DOM element specified
20342          * @method getElWrapper
20343          * @param {String} id the id of the element to get
20344          * @return {Roo.dd.DDM.ElementWrapper} the wrapped element
20345          * @private
20346          * @deprecated This wrapper isn't that useful
20347          * @static
20348          */
20349         getElWrapper: function(id) {
20350             var oWrapper = this.elementCache[id];
20351             if (!oWrapper || !oWrapper.el) {
20352                 oWrapper = this.elementCache[id] =
20353                     new this.ElementWrapper(Roo.getDom(id));
20354             }
20355             return oWrapper;
20356         },
20357
20358         /**
20359          * Returns the actual DOM element
20360          * @method getElement
20361          * @param {String} id the id of the elment to get
20362          * @return {Object} The element
20363          * @deprecated use Roo.getDom instead
20364          * @static
20365          */
20366         getElement: function(id) {
20367             return Roo.getDom(id);
20368         },
20369
20370         /**
20371          * Returns the style property for the DOM element (i.e.,
20372          * document.getElById(id).style)
20373          * @method getCss
20374          * @param {String} id the id of the elment to get
20375          * @return {Object} The style property of the element
20376          * @deprecated use Roo.getDom instead
20377          * @static
20378          */
20379         getCss: function(id) {
20380             var el = Roo.getDom(id);
20381             return (el) ? el.style : null;
20382         },
20383
20384         /**
20385          * Inner class for cached elements
20386          * @class DragDropMgr.ElementWrapper
20387          * @for DragDropMgr
20388          * @private
20389          * @deprecated
20390          */
20391         ElementWrapper: function(el) {
20392                 /**
20393                  * The element
20394                  * @property el
20395                  */
20396                 this.el = el || null;
20397                 /**
20398                  * The element id
20399                  * @property id
20400                  */
20401                 this.id = this.el && el.id;
20402                 /**
20403                  * A reference to the style property
20404                  * @property css
20405                  */
20406                 this.css = this.el && el.style;
20407             },
20408
20409         /**
20410          * Returns the X position of an html element
20411          * @method getPosX
20412          * @param el the element for which to get the position
20413          * @return {int} the X coordinate
20414          * @for DragDropMgr
20415          * @deprecated use Roo.lib.Dom.getX instead
20416          * @static
20417          */
20418         getPosX: function(el) {
20419             return Roo.lib.Dom.getX(el);
20420         },
20421
20422         /**
20423          * Returns the Y position of an html element
20424          * @method getPosY
20425          * @param el the element for which to get the position
20426          * @return {int} the Y coordinate
20427          * @deprecated use Roo.lib.Dom.getY instead
20428          * @static
20429          */
20430         getPosY: function(el) {
20431             return Roo.lib.Dom.getY(el);
20432         },
20433
20434         /**
20435          * Swap two nodes.  In IE, we use the native method, for others we
20436          * emulate the IE behavior
20437          * @method swapNode
20438          * @param n1 the first node to swap
20439          * @param n2 the other node to swap
20440          * @static
20441          */
20442         swapNode: function(n1, n2) {
20443             if (n1.swapNode) {
20444                 n1.swapNode(n2);
20445             } else {
20446                 var p = n2.parentNode;
20447                 var s = n2.nextSibling;
20448
20449                 if (s == n1) {
20450                     p.insertBefore(n1, n2);
20451                 } else if (n2 == n1.nextSibling) {
20452                     p.insertBefore(n2, n1);
20453                 } else {
20454                     n1.parentNode.replaceChild(n2, n1);
20455                     p.insertBefore(n1, s);
20456                 }
20457             }
20458         },
20459
20460         /**
20461          * Returns the current scroll position
20462          * @method getScroll
20463          * @private
20464          * @static
20465          */
20466         getScroll: function () {
20467             var t, l, dde=document.documentElement, db=document.body;
20468             if (dde && (dde.scrollTop || dde.scrollLeft)) {
20469                 t = dde.scrollTop;
20470                 l = dde.scrollLeft;
20471             } else if (db) {
20472                 t = db.scrollTop;
20473                 l = db.scrollLeft;
20474             } else {
20475
20476             }
20477             return { top: t, left: l };
20478         },
20479
20480         /**
20481          * Returns the specified element style property
20482          * @method getStyle
20483          * @param {HTMLElement} el          the element
20484          * @param {string}      styleProp   the style property
20485          * @return {string} The value of the style property
20486          * @deprecated use Roo.lib.Dom.getStyle
20487          * @static
20488          */
20489         getStyle: function(el, styleProp) {
20490             return Roo.fly(el).getStyle(styleProp);
20491         },
20492
20493         /**
20494          * Gets the scrollTop
20495          * @method getScrollTop
20496          * @return {int} the document's scrollTop
20497          * @static
20498          */
20499         getScrollTop: function () { return this.getScroll().top; },
20500
20501         /**
20502          * Gets the scrollLeft
20503          * @method getScrollLeft
20504          * @return {int} the document's scrollTop
20505          * @static
20506          */
20507         getScrollLeft: function () { return this.getScroll().left; },
20508
20509         /**
20510          * Sets the x/y position of an element to the location of the
20511          * target element.
20512          * @method moveToEl
20513          * @param {HTMLElement} moveEl      The element to move
20514          * @param {HTMLElement} targetEl    The position reference element
20515          * @static
20516          */
20517         moveToEl: function (moveEl, targetEl) {
20518             var aCoord = Roo.lib.Dom.getXY(targetEl);
20519             Roo.lib.Dom.setXY(moveEl, aCoord);
20520         },
20521
20522         /**
20523          * Numeric array sort function
20524          * @method numericSort
20525          * @static
20526          */
20527         numericSort: function(a, b) { return (a - b); },
20528
20529         /**
20530          * Internal counter
20531          * @property _timeoutCount
20532          * @private
20533          * @static
20534          */
20535         _timeoutCount: 0,
20536
20537         /**
20538          * Trying to make the load order less important.  Without this we get
20539          * an error if this file is loaded before the Event Utility.
20540          * @method _addListeners
20541          * @private
20542          * @static
20543          */
20544         _addListeners: function() {
20545             var DDM = Roo.dd.DDM;
20546             if ( Roo.lib.Event && document ) {
20547                 DDM._onLoad();
20548             } else {
20549                 if (DDM._timeoutCount > 2000) {
20550                 } else {
20551                     setTimeout(DDM._addListeners, 10);
20552                     if (document && document.body) {
20553                         DDM._timeoutCount += 1;
20554                     }
20555                 }
20556             }
20557         },
20558
20559         /**
20560          * Recursively searches the immediate parent and all child nodes for
20561          * the handle element in order to determine wheter or not it was
20562          * clicked.
20563          * @method handleWasClicked
20564          * @param node the html element to inspect
20565          * @static
20566          */
20567         handleWasClicked: function(node, id) {
20568             if (this.isHandle(id, node.id)) {
20569                 return true;
20570             } else {
20571                 // check to see if this is a text node child of the one we want
20572                 var p = node.parentNode;
20573
20574                 while (p) {
20575                     if (this.isHandle(id, p.id)) {
20576                         return true;
20577                     } else {
20578                         p = p.parentNode;
20579                     }
20580                 }
20581             }
20582
20583             return false;
20584         }
20585
20586     };
20587
20588 }();
20589
20590 // shorter alias, save a few bytes
20591 Roo.dd.DDM = Roo.dd.DragDropMgr;
20592 Roo.dd.DDM._addListeners();
20593
20594 }/*
20595  * Based on:
20596  * Ext JS Library 1.1.1
20597  * Copyright(c) 2006-2007, Ext JS, LLC.
20598  *
20599  * Originally Released Under LGPL - original licence link has changed is not relivant.
20600  *
20601  * Fork - LGPL
20602  * <script type="text/javascript">
20603  */
20604
20605 /**
20606  * @class Roo.dd.DD
20607  * A DragDrop implementation where the linked element follows the
20608  * mouse cursor during a drag.
20609  * @extends Roo.dd.DragDrop
20610  * @constructor
20611  * @param {String} id the id of the linked element
20612  * @param {String} sGroup the group of related DragDrop items
20613  * @param {object} config an object containing configurable attributes
20614  *                Valid properties for DD:
20615  *                    scroll
20616  */
20617 Roo.dd.DD = function(id, sGroup, config) {
20618     if (id) {
20619         this.init(id, sGroup, config);
20620     }
20621 };
20622
20623 Roo.extend(Roo.dd.DD, Roo.dd.DragDrop, {
20624
20625     /**
20626      * When set to true, the utility automatically tries to scroll the browser
20627      * window wehn a drag and drop element is dragged near the viewport boundary.
20628      * Defaults to true.
20629      * @property scroll
20630      * @type boolean
20631      */
20632     scroll: true,
20633
20634     /**
20635      * Sets the pointer offset to the distance between the linked element's top
20636      * left corner and the location the element was clicked
20637      * @method autoOffset
20638      * @param {int} iPageX the X coordinate of the click
20639      * @param {int} iPageY the Y coordinate of the click
20640      */
20641     autoOffset: function(iPageX, iPageY) {
20642         var x = iPageX - this.startPageX;
20643         var y = iPageY - this.startPageY;
20644         this.setDelta(x, y);
20645     },
20646
20647     /**
20648      * Sets the pointer offset.  You can call this directly to force the
20649      * offset to be in a particular location (e.g., pass in 0,0 to set it
20650      * to the center of the object)
20651      * @method setDelta
20652      * @param {int} iDeltaX the distance from the left
20653      * @param {int} iDeltaY the distance from the top
20654      */
20655     setDelta: function(iDeltaX, iDeltaY) {
20656         this.deltaX = iDeltaX;
20657         this.deltaY = iDeltaY;
20658     },
20659
20660     /**
20661      * Sets the drag element to the location of the mousedown or click event,
20662      * maintaining the cursor location relative to the location on the element
20663      * that was clicked.  Override this if you want to place the element in a
20664      * location other than where the cursor is.
20665      * @method setDragElPos
20666      * @param {int} iPageX the X coordinate of the mousedown or drag event
20667      * @param {int} iPageY the Y coordinate of the mousedown or drag event
20668      */
20669     setDragElPos: function(iPageX, iPageY) {
20670         // the first time we do this, we are going to check to make sure
20671         // the element has css positioning
20672
20673         var el = this.getDragEl();
20674         this.alignElWithMouse(el, iPageX, iPageY);
20675     },
20676
20677     /**
20678      * Sets the element to the location of the mousedown or click event,
20679      * maintaining the cursor location relative to the location on the element
20680      * that was clicked.  Override this if you want to place the element in a
20681      * location other than where the cursor is.
20682      * @method alignElWithMouse
20683      * @param {HTMLElement} el the element to move
20684      * @param {int} iPageX the X coordinate of the mousedown or drag event
20685      * @param {int} iPageY the Y coordinate of the mousedown or drag event
20686      */
20687     alignElWithMouse: function(el, iPageX, iPageY) {
20688         var oCoord = this.getTargetCoord(iPageX, iPageY);
20689         var fly = el.dom ? el : Roo.fly(el);
20690         if (!this.deltaSetXY) {
20691             var aCoord = [oCoord.x, oCoord.y];
20692             fly.setXY(aCoord);
20693             var newLeft = fly.getLeft(true);
20694             var newTop  = fly.getTop(true);
20695             this.deltaSetXY = [ newLeft - oCoord.x, newTop - oCoord.y ];
20696         } else {
20697             fly.setLeftTop(oCoord.x + this.deltaSetXY[0], oCoord.y + this.deltaSetXY[1]);
20698         }
20699
20700         this.cachePosition(oCoord.x, oCoord.y);
20701         this.autoScroll(oCoord.x, oCoord.y, el.offsetHeight, el.offsetWidth);
20702         return oCoord;
20703     },
20704
20705     /**
20706      * Saves the most recent position so that we can reset the constraints and
20707      * tick marks on-demand.  We need to know this so that we can calculate the
20708      * number of pixels the element is offset from its original position.
20709      * @method cachePosition
20710      * @param iPageX the current x position (optional, this just makes it so we
20711      * don't have to look it up again)
20712      * @param iPageY the current y position (optional, this just makes it so we
20713      * don't have to look it up again)
20714      */
20715     cachePosition: function(iPageX, iPageY) {
20716         if (iPageX) {
20717             this.lastPageX = iPageX;
20718             this.lastPageY = iPageY;
20719         } else {
20720             var aCoord = Roo.lib.Dom.getXY(this.getEl());
20721             this.lastPageX = aCoord[0];
20722             this.lastPageY = aCoord[1];
20723         }
20724     },
20725
20726     /**
20727      * Auto-scroll the window if the dragged object has been moved beyond the
20728      * visible window boundary.
20729      * @method autoScroll
20730      * @param {int} x the drag element's x position
20731      * @param {int} y the drag element's y position
20732      * @param {int} h the height of the drag element
20733      * @param {int} w the width of the drag element
20734      * @private
20735      */
20736     autoScroll: function(x, y, h, w) {
20737
20738         if (this.scroll) {
20739             // The client height
20740             var clientH = Roo.lib.Dom.getViewWidth();
20741
20742             // The client width
20743             var clientW = Roo.lib.Dom.getViewHeight();
20744
20745             // The amt scrolled down
20746             var st = this.DDM.getScrollTop();
20747
20748             // The amt scrolled right
20749             var sl = this.DDM.getScrollLeft();
20750
20751             // Location of the bottom of the element
20752             var bot = h + y;
20753
20754             // Location of the right of the element
20755             var right = w + x;
20756
20757             // The distance from the cursor to the bottom of the visible area,
20758             // adjusted so that we don't scroll if the cursor is beyond the
20759             // element drag constraints
20760             var toBot = (clientH + st - y - this.deltaY);
20761
20762             // The distance from the cursor to the right of the visible area
20763             var toRight = (clientW + sl - x - this.deltaX);
20764
20765
20766             // How close to the edge the cursor must be before we scroll
20767             // var thresh = (document.all) ? 100 : 40;
20768             var thresh = 40;
20769
20770             // How many pixels to scroll per autoscroll op.  This helps to reduce
20771             // clunky scrolling. IE is more sensitive about this ... it needs this
20772             // value to be higher.
20773             var scrAmt = (document.all) ? 80 : 30;
20774
20775             // Scroll down if we are near the bottom of the visible page and the
20776             // obj extends below the crease
20777             if ( bot > clientH && toBot < thresh ) {
20778                 window.scrollTo(sl, st + scrAmt);
20779             }
20780
20781             // Scroll up if the window is scrolled down and the top of the object
20782             // goes above the top border
20783             if ( y < st && st > 0 && y - st < thresh ) {
20784                 window.scrollTo(sl, st - scrAmt);
20785             }
20786
20787             // Scroll right if the obj is beyond the right border and the cursor is
20788             // near the border.
20789             if ( right > clientW && toRight < thresh ) {
20790                 window.scrollTo(sl + scrAmt, st);
20791             }
20792
20793             // Scroll left if the window has been scrolled to the right and the obj
20794             // extends past the left border
20795             if ( x < sl && sl > 0 && x - sl < thresh ) {
20796                 window.scrollTo(sl - scrAmt, st);
20797             }
20798         }
20799     },
20800
20801     /**
20802      * Finds the location the element should be placed if we want to move
20803      * it to where the mouse location less the click offset would place us.
20804      * @method getTargetCoord
20805      * @param {int} iPageX the X coordinate of the click
20806      * @param {int} iPageY the Y coordinate of the click
20807      * @return an object that contains the coordinates (Object.x and Object.y)
20808      * @private
20809      */
20810     getTargetCoord: function(iPageX, iPageY) {
20811
20812
20813         var x = iPageX - this.deltaX;
20814         var y = iPageY - this.deltaY;
20815
20816         if (this.constrainX) {
20817             if (x < this.minX) { x = this.minX; }
20818             if (x > this.maxX) { x = this.maxX; }
20819         }
20820
20821         if (this.constrainY) {
20822             if (y < this.minY) { y = this.minY; }
20823             if (y > this.maxY) { y = this.maxY; }
20824         }
20825
20826         x = this.getTick(x, this.xTicks);
20827         y = this.getTick(y, this.yTicks);
20828
20829
20830         return {x:x, y:y};
20831     },
20832
20833     /*
20834      * Sets up config options specific to this class. Overrides
20835      * Roo.dd.DragDrop, but all versions of this method through the
20836      * inheritance chain are called
20837      */
20838     applyConfig: function() {
20839         Roo.dd.DD.superclass.applyConfig.call(this);
20840         this.scroll = (this.config.scroll !== false);
20841     },
20842
20843     /*
20844      * Event that fires prior to the onMouseDown event.  Overrides
20845      * Roo.dd.DragDrop.
20846      */
20847     b4MouseDown: function(e) {
20848         // this.resetConstraints();
20849         this.autoOffset(e.getPageX(),
20850                             e.getPageY());
20851     },
20852
20853     /*
20854      * Event that fires prior to the onDrag event.  Overrides
20855      * Roo.dd.DragDrop.
20856      */
20857     b4Drag: function(e) {
20858         this.setDragElPos(e.getPageX(),
20859                             e.getPageY());
20860     },
20861
20862     toString: function() {
20863         return ("DD " + this.id);
20864     }
20865
20866     //////////////////////////////////////////////////////////////////////////
20867     // Debugging ygDragDrop events that can be overridden
20868     //////////////////////////////////////////////////////////////////////////
20869     /*
20870     startDrag: function(x, y) {
20871     },
20872
20873     onDrag: function(e) {
20874     },
20875
20876     onDragEnter: function(e, id) {
20877     },
20878
20879     onDragOver: function(e, id) {
20880     },
20881
20882     onDragOut: function(e, id) {
20883     },
20884
20885     onDragDrop: function(e, id) {
20886     },
20887
20888     endDrag: function(e) {
20889     }
20890
20891     */
20892
20893 });/*
20894  * Based on:
20895  * Ext JS Library 1.1.1
20896  * Copyright(c) 2006-2007, Ext JS, LLC.
20897  *
20898  * Originally Released Under LGPL - original licence link has changed is not relivant.
20899  *
20900  * Fork - LGPL
20901  * <script type="text/javascript">
20902  */
20903
20904 /**
20905  * @class Roo.dd.DDProxy
20906  * A DragDrop implementation that inserts an empty, bordered div into
20907  * the document that follows the cursor during drag operations.  At the time of
20908  * the click, the frame div is resized to the dimensions of the linked html
20909  * element, and moved to the exact location of the linked element.
20910  *
20911  * References to the "frame" element refer to the single proxy element that
20912  * was created to be dragged in place of all DDProxy elements on the
20913  * page.
20914  *
20915  * @extends Roo.dd.DD
20916  * @constructor
20917  * @param {String} id the id of the linked html element
20918  * @param {String} sGroup the group of related DragDrop objects
20919  * @param {object} config an object containing configurable attributes
20920  *                Valid properties for DDProxy in addition to those in DragDrop:
20921  *                   resizeFrame, centerFrame, dragElId
20922  */
20923 Roo.dd.DDProxy = function(id, sGroup, config) {
20924     if (id) {
20925         this.init(id, sGroup, config);
20926         this.initFrame();
20927     }
20928 };
20929
20930 /**
20931  * The default drag frame div id
20932  * @property Roo.dd.DDProxy.dragElId
20933  * @type String
20934  * @static
20935  */
20936 Roo.dd.DDProxy.dragElId = "ygddfdiv";
20937
20938 Roo.extend(Roo.dd.DDProxy, Roo.dd.DD, {
20939
20940     /**
20941      * By default we resize the drag frame to be the same size as the element
20942      * we want to drag (this is to get the frame effect).  We can turn it off
20943      * if we want a different behavior.
20944      * @property resizeFrame
20945      * @type boolean
20946      */
20947     resizeFrame: true,
20948
20949     /**
20950      * By default the frame is positioned exactly where the drag element is, so
20951      * we use the cursor offset provided by Roo.dd.DD.  Another option that works only if
20952      * you do not have constraints on the obj is to have the drag frame centered
20953      * around the cursor.  Set centerFrame to true for this effect.
20954      * @property centerFrame
20955      * @type boolean
20956      */
20957     centerFrame: false,
20958
20959     /**
20960      * Creates the proxy element if it does not yet exist
20961      * @method createFrame
20962      */
20963     createFrame: function() {
20964         var self = this;
20965         var body = document.body;
20966
20967         if (!body || !body.firstChild) {
20968             setTimeout( function() { self.createFrame(); }, 50 );
20969             return;
20970         }
20971
20972         var div = this.getDragEl();
20973
20974         if (!div) {
20975             div    = document.createElement("div");
20976             div.id = this.dragElId;
20977             var s  = div.style;
20978
20979             s.position   = "absolute";
20980             s.visibility = "hidden";
20981             s.cursor     = "move";
20982             s.border     = "2px solid #aaa";
20983             s.zIndex     = 999;
20984
20985             // appendChild can blow up IE if invoked prior to the window load event
20986             // while rendering a table.  It is possible there are other scenarios
20987             // that would cause this to happen as well.
20988             body.insertBefore(div, body.firstChild);
20989         }
20990     },
20991
20992     /**
20993      * Initialization for the drag frame element.  Must be called in the
20994      * constructor of all subclasses
20995      * @method initFrame
20996      */
20997     initFrame: function() {
20998         this.createFrame();
20999     },
21000
21001     applyConfig: function() {
21002         Roo.dd.DDProxy.superclass.applyConfig.call(this);
21003
21004         this.resizeFrame = (this.config.resizeFrame !== false);
21005         this.centerFrame = (this.config.centerFrame);
21006         this.setDragElId(this.config.dragElId || Roo.dd.DDProxy.dragElId);
21007     },
21008
21009     /**
21010      * Resizes the drag frame to the dimensions of the clicked object, positions
21011      * it over the object, and finally displays it
21012      * @method showFrame
21013      * @param {int} iPageX X click position
21014      * @param {int} iPageY Y click position
21015      * @private
21016      */
21017     showFrame: function(iPageX, iPageY) {
21018         var el = this.getEl();
21019         var dragEl = this.getDragEl();
21020         var s = dragEl.style;
21021
21022         this._resizeProxy();
21023
21024         if (this.centerFrame) {
21025             this.setDelta( Math.round(parseInt(s.width,  10)/2),
21026                            Math.round(parseInt(s.height, 10)/2) );
21027         }
21028
21029         this.setDragElPos(iPageX, iPageY);
21030
21031         Roo.fly(dragEl).show();
21032     },
21033
21034     /**
21035      * The proxy is automatically resized to the dimensions of the linked
21036      * element when a drag is initiated, unless resizeFrame is set to false
21037      * @method _resizeProxy
21038      * @private
21039      */
21040     _resizeProxy: function() {
21041         if (this.resizeFrame) {
21042             var el = this.getEl();
21043             Roo.fly(this.getDragEl()).setSize(el.offsetWidth, el.offsetHeight);
21044         }
21045     },
21046
21047     // overrides Roo.dd.DragDrop
21048     b4MouseDown: function(e) {
21049         var x = e.getPageX();
21050         var y = e.getPageY();
21051         this.autoOffset(x, y);
21052         this.setDragElPos(x, y);
21053     },
21054
21055     // overrides Roo.dd.DragDrop
21056     b4StartDrag: function(x, y) {
21057         // show the drag frame
21058         this.showFrame(x, y);
21059     },
21060
21061     // overrides Roo.dd.DragDrop
21062     b4EndDrag: function(e) {
21063         Roo.fly(this.getDragEl()).hide();
21064     },
21065
21066     // overrides Roo.dd.DragDrop
21067     // By default we try to move the element to the last location of the frame.
21068     // This is so that the default behavior mirrors that of Roo.dd.DD.
21069     endDrag: function(e) {
21070
21071         var lel = this.getEl();
21072         var del = this.getDragEl();
21073
21074         // Show the drag frame briefly so we can get its position
21075         del.style.visibility = "";
21076
21077         this.beforeMove();
21078         // Hide the linked element before the move to get around a Safari
21079         // rendering bug.
21080         lel.style.visibility = "hidden";
21081         Roo.dd.DDM.moveToEl(lel, del);
21082         del.style.visibility = "hidden";
21083         lel.style.visibility = "";
21084
21085         this.afterDrag();
21086     },
21087
21088     beforeMove : function(){
21089
21090     },
21091
21092     afterDrag : function(){
21093
21094     },
21095
21096     toString: function() {
21097         return ("DDProxy " + this.id);
21098     }
21099
21100 });
21101 /*
21102  * Based on:
21103  * Ext JS Library 1.1.1
21104  * Copyright(c) 2006-2007, Ext JS, LLC.
21105  *
21106  * Originally Released Under LGPL - original licence link has changed is not relivant.
21107  *
21108  * Fork - LGPL
21109  * <script type="text/javascript">
21110  */
21111
21112  /**
21113  * @class Roo.dd.DDTarget
21114  * A DragDrop implementation that does not move, but can be a drop
21115  * target.  You would get the same result by simply omitting implementation
21116  * for the event callbacks, but this way we reduce the processing cost of the
21117  * event listener and the callbacks.
21118  * @extends Roo.dd.DragDrop
21119  * @constructor
21120  * @param {String} id the id of the element that is a drop target
21121  * @param {String} sGroup the group of related DragDrop objects
21122  * @param {object} config an object containing configurable attributes
21123  *                 Valid properties for DDTarget in addition to those in
21124  *                 DragDrop:
21125  *                    none
21126  */
21127 Roo.dd.DDTarget = function(id, sGroup, config) {
21128     if (id) {
21129         this.initTarget(id, sGroup, config);
21130     }
21131     if (config && (config.listeners || config.events)) { 
21132         Roo.dd.DragDrop.superclass.constructor.call(this,  { 
21133             listeners : config.listeners || {}, 
21134             events : config.events || {} 
21135         });    
21136     }
21137 };
21138
21139 // Roo.dd.DDTarget.prototype = new Roo.dd.DragDrop();
21140 Roo.extend(Roo.dd.DDTarget, Roo.dd.DragDrop, {
21141     toString: function() {
21142         return ("DDTarget " + this.id);
21143     }
21144 });
21145 /*
21146  * Based on:
21147  * Ext JS Library 1.1.1
21148  * Copyright(c) 2006-2007, Ext JS, LLC.
21149  *
21150  * Originally Released Under LGPL - original licence link has changed is not relivant.
21151  *
21152  * Fork - LGPL
21153  * <script type="text/javascript">
21154  */
21155  
21156
21157 /**
21158  * @class Roo.dd.ScrollManager
21159  * Provides automatic scrolling of overflow regions in the page during drag operations.<br><br>
21160  * <b>Note: This class uses "Point Mode" and is untested in "Intersect Mode".</b>
21161  * @singleton
21162  */
21163 Roo.dd.ScrollManager = function(){
21164     var ddm = Roo.dd.DragDropMgr;
21165     var els = {};
21166     var dragEl = null;
21167     var proc = {};
21168     
21169     
21170     
21171     var onStop = function(e){
21172         dragEl = null;
21173         clearProc();
21174     };
21175     
21176     var triggerRefresh = function(){
21177         if(ddm.dragCurrent){
21178              ddm.refreshCache(ddm.dragCurrent.groups);
21179         }
21180     };
21181     
21182     var doScroll = function(){
21183         if(ddm.dragCurrent){
21184             var dds = Roo.dd.ScrollManager;
21185             if(!dds.animate){
21186                 if(proc.el.scroll(proc.dir, dds.increment)){
21187                     triggerRefresh();
21188                 }
21189             }else{
21190                 proc.el.scroll(proc.dir, dds.increment, true, dds.animDuration, triggerRefresh);
21191             }
21192         }
21193     };
21194     
21195     var clearProc = function(){
21196         if(proc.id){
21197             clearInterval(proc.id);
21198         }
21199         proc.id = 0;
21200         proc.el = null;
21201         proc.dir = "";
21202     };
21203     
21204     var startProc = function(el, dir){
21205          Roo.log('scroll startproc');
21206         clearProc();
21207         proc.el = el;
21208         proc.dir = dir;
21209         proc.id = setInterval(doScroll, Roo.dd.ScrollManager.frequency);
21210     };
21211     
21212     var onFire = function(e, isDrop){
21213        
21214         if(isDrop || !ddm.dragCurrent){ return; }
21215         var dds = Roo.dd.ScrollManager;
21216         if(!dragEl || dragEl != ddm.dragCurrent){
21217             dragEl = ddm.dragCurrent;
21218             // refresh regions on drag start
21219             dds.refreshCache();
21220         }
21221         
21222         var xy = Roo.lib.Event.getXY(e);
21223         var pt = new Roo.lib.Point(xy[0], xy[1]);
21224         for(var id in els){
21225             var el = els[id], r = el._region;
21226             if(r && r.contains(pt) && el.isScrollable()){
21227                 if(r.bottom - pt.y <= dds.thresh){
21228                     if(proc.el != el){
21229                         startProc(el, "down");
21230                     }
21231                     return;
21232                 }else if(r.right - pt.x <= dds.thresh){
21233                     if(proc.el != el){
21234                         startProc(el, "left");
21235                     }
21236                     return;
21237                 }else if(pt.y - r.top <= dds.thresh){
21238                     if(proc.el != el){
21239                         startProc(el, "up");
21240                     }
21241                     return;
21242                 }else if(pt.x - r.left <= dds.thresh){
21243                     if(proc.el != el){
21244                         startProc(el, "right");
21245                     }
21246                     return;
21247                 }
21248             }
21249         }
21250         clearProc();
21251     };
21252     
21253     ddm.fireEvents = ddm.fireEvents.createSequence(onFire, ddm);
21254     ddm.stopDrag = ddm.stopDrag.createSequence(onStop, ddm);
21255     
21256     return {
21257         /**
21258          * Registers new overflow element(s) to auto scroll
21259          * @param {String/HTMLElement/Element/Array} el The id of or the element to be scrolled or an array of either
21260          */
21261         register : function(el){
21262             if(el instanceof Array){
21263                 for(var i = 0, len = el.length; i < len; i++) {
21264                         this.register(el[i]);
21265                 }
21266             }else{
21267                 el = Roo.get(el);
21268                 els[el.id] = el;
21269             }
21270             Roo.dd.ScrollManager.els = els;
21271         },
21272         
21273         /**
21274          * Unregisters overflow element(s) so they are no longer scrolled
21275          * @param {String/HTMLElement/Element/Array} el The id of or the element to be removed or an array of either
21276          */
21277         unregister : function(el){
21278             if(el instanceof Array){
21279                 for(var i = 0, len = el.length; i < len; i++) {
21280                         this.unregister(el[i]);
21281                 }
21282             }else{
21283                 el = Roo.get(el);
21284                 delete els[el.id];
21285             }
21286         },
21287         
21288         /**
21289          * The number of pixels from the edge of a container the pointer needs to be to 
21290          * trigger scrolling (defaults to 25)
21291          * @type Number
21292          */
21293         thresh : 25,
21294         
21295         /**
21296          * The number of pixels to scroll in each scroll increment (defaults to 50)
21297          * @type Number
21298          */
21299         increment : 100,
21300         
21301         /**
21302          * The frequency of scrolls in milliseconds (defaults to 500)
21303          * @type Number
21304          */
21305         frequency : 500,
21306         
21307         /**
21308          * True to animate the scroll (defaults to true)
21309          * @type Boolean
21310          */
21311         animate: true,
21312         
21313         /**
21314          * The animation duration in seconds - 
21315          * MUST BE less than Roo.dd.ScrollManager.frequency! (defaults to .4)
21316          * @type Number
21317          */
21318         animDuration: .4,
21319         
21320         /**
21321          * Manually trigger a cache refresh.
21322          */
21323         refreshCache : function(){
21324             for(var id in els){
21325                 if(typeof els[id] == 'object'){ // for people extending the object prototype
21326                     els[id]._region = els[id].getRegion();
21327                 }
21328             }
21329         }
21330     };
21331 }();/*
21332  * Based on:
21333  * Ext JS Library 1.1.1
21334  * Copyright(c) 2006-2007, Ext JS, LLC.
21335  *
21336  * Originally Released Under LGPL - original licence link has changed is not relivant.
21337  *
21338  * Fork - LGPL
21339  * <script type="text/javascript">
21340  */
21341  
21342
21343 /**
21344  * @class Roo.dd.Registry
21345  * Provides easy access to all drag drop components that are registered on a page.  Items can be retrieved either
21346  * directly by DOM node id, or by passing in the drag drop event that occurred and looking up the event target.
21347  * @singleton
21348  */
21349 Roo.dd.Registry = function(){
21350     var elements = {}; 
21351     var handles = {}; 
21352     var autoIdSeed = 0;
21353
21354     var getId = function(el, autogen){
21355         if(typeof el == "string"){
21356             return el;
21357         }
21358         var id = el.id;
21359         if(!id && autogen !== false){
21360             id = "roodd-" + (++autoIdSeed);
21361             el.id = id;
21362         }
21363         return id;
21364     };
21365     
21366     return {
21367     /**
21368      * Register a drag drop element
21369      * @param {String|HTMLElement} element The id or DOM node to register
21370      * @param {Object} data (optional) A custom data object that will be passed between the elements that are involved
21371      * in drag drop operations.  You can populate this object with any arbitrary properties that your own code
21372      * knows how to interpret, plus there are some specific properties known to the Registry that should be
21373      * populated in the data object (if applicable):
21374      * <pre>
21375 Value      Description<br />
21376 ---------  ------------------------------------------<br />
21377 handles    Array of DOM nodes that trigger dragging<br />
21378            for the element being registered<br />
21379 isHandle   True if the element passed in triggers<br />
21380            dragging itself, else false
21381 </pre>
21382      */
21383         register : function(el, data){
21384             data = data || {};
21385             if(typeof el == "string"){
21386                 el = document.getElementById(el);
21387             }
21388             data.ddel = el;
21389             elements[getId(el)] = data;
21390             if(data.isHandle !== false){
21391                 handles[data.ddel.id] = data;
21392             }
21393             if(data.handles){
21394                 var hs = data.handles;
21395                 for(var i = 0, len = hs.length; i < len; i++){
21396                         handles[getId(hs[i])] = data;
21397                 }
21398             }
21399         },
21400
21401     /**
21402      * Unregister a drag drop element
21403      * @param {String|HTMLElement}  element The id or DOM node to unregister
21404      */
21405         unregister : function(el){
21406             var id = getId(el, false);
21407             var data = elements[id];
21408             if(data){
21409                 delete elements[id];
21410                 if(data.handles){
21411                     var hs = data.handles;
21412                     for(var i = 0, len = hs.length; i < len; i++){
21413                         delete handles[getId(hs[i], false)];
21414                     }
21415                 }
21416             }
21417         },
21418
21419     /**
21420      * Returns the handle registered for a DOM Node by id
21421      * @param {String|HTMLElement} id The DOM node or id to look up
21422      * @return {Object} handle The custom handle data
21423      */
21424         getHandle : function(id){
21425             if(typeof id != "string"){ // must be element?
21426                 id = id.id;
21427             }
21428             return handles[id];
21429         },
21430
21431     /**
21432      * Returns the handle that is registered for the DOM node that is the target of the event
21433      * @param {Event} e The event
21434      * @return {Object} handle The custom handle data
21435      */
21436         getHandleFromEvent : function(e){
21437             var t = Roo.lib.Event.getTarget(e);
21438             return t ? handles[t.id] : null;
21439         },
21440
21441     /**
21442      * Returns a custom data object that is registered for a DOM node by id
21443      * @param {String|HTMLElement} id The DOM node or id to look up
21444      * @return {Object} data The custom data
21445      */
21446         getTarget : function(id){
21447             if(typeof id != "string"){ // must be element?
21448                 id = id.id;
21449             }
21450             return elements[id];
21451         },
21452
21453     /**
21454      * Returns a custom data object that is registered for the DOM node that is the target of the event
21455      * @param {Event} e The event
21456      * @return {Object} data The custom data
21457      */
21458         getTargetFromEvent : function(e){
21459             var t = Roo.lib.Event.getTarget(e);
21460             return t ? elements[t.id] || handles[t.id] : null;
21461         }
21462     };
21463 }();/*
21464  * Based on:
21465  * Ext JS Library 1.1.1
21466  * Copyright(c) 2006-2007, Ext JS, LLC.
21467  *
21468  * Originally Released Under LGPL - original licence link has changed is not relivant.
21469  *
21470  * Fork - LGPL
21471  * <script type="text/javascript">
21472  */
21473  
21474
21475 /**
21476  * @class Roo.dd.StatusProxy
21477  * A specialized drag proxy that supports a drop status icon, {@link Roo.Layer} styles and auto-repair.  This is the
21478  * default drag proxy used by all Roo.dd components.
21479  * @constructor
21480  * @param {Object} config
21481  */
21482 Roo.dd.StatusProxy = function(config){
21483     Roo.apply(this, config);
21484     this.id = this.id || Roo.id();
21485     this.el = new Roo.Layer({
21486         dh: {
21487             id: this.id, tag: "div", cls: "x-dd-drag-proxy "+this.dropNotAllowed, children: [
21488                 {tag: "div", cls: "x-dd-drop-icon"},
21489                 {tag: "div", cls: "x-dd-drag-ghost"}
21490             ]
21491         }, 
21492         shadow: !config || config.shadow !== false
21493     });
21494     this.ghost = Roo.get(this.el.dom.childNodes[1]);
21495     this.dropStatus = this.dropNotAllowed;
21496 };
21497
21498 Roo.dd.StatusProxy.prototype = {
21499     /**
21500      * @cfg {String} dropAllowed
21501      * The CSS class to apply to the status element when drop is allowed (defaults to "x-dd-drop-ok").
21502      */
21503     dropAllowed : "x-dd-drop-ok",
21504     /**
21505      * @cfg {String} dropNotAllowed
21506      * The CSS class to apply to the status element when drop is not allowed (defaults to "x-dd-drop-nodrop").
21507      */
21508     dropNotAllowed : "x-dd-drop-nodrop",
21509
21510     /**
21511      * Updates the proxy's visual element to indicate the status of whether or not drop is allowed
21512      * over the current target element.
21513      * @param {String} cssClass The css class for the new drop status indicator image
21514      */
21515     setStatus : function(cssClass){
21516         cssClass = cssClass || this.dropNotAllowed;
21517         if(this.dropStatus != cssClass){
21518             this.el.replaceClass(this.dropStatus, cssClass);
21519             this.dropStatus = cssClass;
21520         }
21521     },
21522
21523     /**
21524      * Resets the status indicator to the default dropNotAllowed value
21525      * @param {Boolean} clearGhost True to also remove all content from the ghost, false to preserve it
21526      */
21527     reset : function(clearGhost){
21528         this.el.dom.className = "x-dd-drag-proxy " + this.dropNotAllowed;
21529         this.dropStatus = this.dropNotAllowed;
21530         if(clearGhost){
21531             this.ghost.update("");
21532         }
21533     },
21534
21535     /**
21536      * Updates the contents of the ghost element
21537      * @param {String} html The html that will replace the current innerHTML of the ghost element
21538      */
21539     update : function(html){
21540         if(typeof html == "string"){
21541             this.ghost.update(html);
21542         }else{
21543             this.ghost.update("");
21544             html.style.margin = "0";
21545             this.ghost.dom.appendChild(html);
21546         }
21547         // ensure float = none set?? cant remember why though.
21548         var el = this.ghost.dom.firstChild;
21549                 if(el){
21550                         Roo.fly(el).setStyle('float', 'none');
21551                 }
21552     },
21553     
21554     /**
21555      * Returns the underlying proxy {@link Roo.Layer}
21556      * @return {Roo.Layer} el
21557     */
21558     getEl : function(){
21559         return this.el;
21560     },
21561
21562     /**
21563      * Returns the ghost element
21564      * @return {Roo.Element} el
21565      */
21566     getGhost : function(){
21567         return this.ghost;
21568     },
21569
21570     /**
21571      * Hides the proxy
21572      * @param {Boolean} clear True to reset the status and clear the ghost contents, false to preserve them
21573      */
21574     hide : function(clear){
21575         this.el.hide();
21576         if(clear){
21577             this.reset(true);
21578         }
21579     },
21580
21581     /**
21582      * Stops the repair animation if it's currently running
21583      */
21584     stop : function(){
21585         if(this.anim && this.anim.isAnimated && this.anim.isAnimated()){
21586             this.anim.stop();
21587         }
21588     },
21589
21590     /**
21591      * Displays this proxy
21592      */
21593     show : function(){
21594         this.el.show();
21595     },
21596
21597     /**
21598      * Force the Layer to sync its shadow and shim positions to the element
21599      */
21600     sync : function(){
21601         this.el.sync();
21602     },
21603
21604     /**
21605      * Causes the proxy to return to its position of origin via an animation.  Should be called after an
21606      * invalid drop operation by the item being dragged.
21607      * @param {Array} xy The XY position of the element ([x, y])
21608      * @param {Function} callback The function to call after the repair is complete
21609      * @param {Object} scope The scope in which to execute the callback
21610      */
21611     repair : function(xy, callback, scope){
21612         this.callback = callback;
21613         this.scope = scope;
21614         if(xy && this.animRepair !== false){
21615             this.el.addClass("x-dd-drag-repair");
21616             this.el.hideUnders(true);
21617             this.anim = this.el.shift({
21618                 duration: this.repairDuration || .5,
21619                 easing: 'easeOut',
21620                 xy: xy,
21621                 stopFx: true,
21622                 callback: this.afterRepair,
21623                 scope: this
21624             });
21625         }else{
21626             this.afterRepair();
21627         }
21628     },
21629
21630     // private
21631     afterRepair : function(){
21632         this.hide(true);
21633         if(typeof this.callback == "function"){
21634             this.callback.call(this.scope || this);
21635         }
21636         this.callback = null;
21637         this.scope = null;
21638     }
21639 };/*
21640  * Based on:
21641  * Ext JS Library 1.1.1
21642  * Copyright(c) 2006-2007, Ext JS, LLC.
21643  *
21644  * Originally Released Under LGPL - original licence link has changed is not relivant.
21645  *
21646  * Fork - LGPL
21647  * <script type="text/javascript">
21648  */
21649
21650 /**
21651  * @class Roo.dd.DragSource
21652  * @extends Roo.dd.DDProxy
21653  * A simple class that provides the basic implementation needed to make any element draggable.
21654  * @constructor
21655  * @param {String/HTMLElement/Element} el The container element
21656  * @param {Object} config
21657  */
21658 Roo.dd.DragSource = function(el, config){
21659     this.el = Roo.get(el);
21660     this.dragData = {};
21661     
21662     Roo.apply(this, config);
21663     
21664     if(!this.proxy){
21665         this.proxy = new Roo.dd.StatusProxy();
21666     }
21667
21668     Roo.dd.DragSource.superclass.constructor.call(this, this.el.dom, this.ddGroup || this.group,
21669           {dragElId : this.proxy.id, resizeFrame: false, isTarget: false, scroll: this.scroll === true});
21670     
21671     this.dragging = false;
21672 };
21673
21674 Roo.extend(Roo.dd.DragSource, Roo.dd.DDProxy, {
21675     /**
21676      * @cfg {String} dropAllowed
21677      * The CSS class returned to the drag source when drop is allowed (defaults to "x-dd-drop-ok").
21678      */
21679     dropAllowed : "x-dd-drop-ok",
21680     /**
21681      * @cfg {String} dropNotAllowed
21682      * The CSS class returned to the drag source when drop is not allowed (defaults to "x-dd-drop-nodrop").
21683      */
21684     dropNotAllowed : "x-dd-drop-nodrop",
21685
21686     /**
21687      * Returns the data object associated with this drag source
21688      * @return {Object} data An object containing arbitrary data
21689      */
21690     getDragData : function(e){
21691         return this.dragData;
21692     },
21693
21694     // private
21695     onDragEnter : function(e, id){
21696         var target = Roo.dd.DragDropMgr.getDDById(id);
21697         this.cachedTarget = target;
21698         if(this.beforeDragEnter(target, e, id) !== false){
21699             if(target.isNotifyTarget){
21700                 var status = target.notifyEnter(this, e, this.dragData);
21701                 this.proxy.setStatus(status);
21702             }else{
21703                 this.proxy.setStatus(this.dropAllowed);
21704             }
21705             
21706             if(this.afterDragEnter){
21707                 /**
21708                  * An empty function by default, but provided so that you can perform a custom action
21709                  * when the dragged item enters the drop target by providing an implementation.
21710                  * @param {Roo.dd.DragDrop} target The drop target
21711                  * @param {Event} e The event object
21712                  * @param {String} id The id of the dragged element
21713                  * @method afterDragEnter
21714                  */
21715                 this.afterDragEnter(target, e, id);
21716             }
21717         }
21718     },
21719
21720     /**
21721      * An empty function by default, but provided so that you can perform a custom action
21722      * before the dragged item enters the drop target and optionally cancel the onDragEnter.
21723      * @param {Roo.dd.DragDrop} target The drop target
21724      * @param {Event} e The event object
21725      * @param {String} id The id of the dragged element
21726      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21727      */
21728     beforeDragEnter : function(target, e, id){
21729         return true;
21730     },
21731
21732     // private
21733     alignElWithMouse: function() {
21734         Roo.dd.DragSource.superclass.alignElWithMouse.apply(this, arguments);
21735         this.proxy.sync();
21736     },
21737
21738     // private
21739     onDragOver : function(e, id){
21740         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21741         if(this.beforeDragOver(target, e, id) !== false){
21742             if(target.isNotifyTarget){
21743                 var status = target.notifyOver(this, e, this.dragData);
21744                 this.proxy.setStatus(status);
21745             }
21746
21747             if(this.afterDragOver){
21748                 /**
21749                  * An empty function by default, but provided so that you can perform a custom action
21750                  * while the dragged item is over the drop target by providing an implementation.
21751                  * @param {Roo.dd.DragDrop} target The drop target
21752                  * @param {Event} e The event object
21753                  * @param {String} id The id of the dragged element
21754                  * @method afterDragOver
21755                  */
21756                 this.afterDragOver(target, e, id);
21757             }
21758         }
21759     },
21760
21761     /**
21762      * An empty function by default, but provided so that you can perform a custom action
21763      * while the dragged item is over the drop target and optionally cancel the onDragOver.
21764      * @param {Roo.dd.DragDrop} target The drop target
21765      * @param {Event} e The event object
21766      * @param {String} id The id of the dragged element
21767      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21768      */
21769     beforeDragOver : function(target, e, id){
21770         return true;
21771     },
21772
21773     // private
21774     onDragOut : function(e, id){
21775         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21776         if(this.beforeDragOut(target, e, id) !== false){
21777             if(target.isNotifyTarget){
21778                 target.notifyOut(this, e, this.dragData);
21779             }
21780             this.proxy.reset();
21781             if(this.afterDragOut){
21782                 /**
21783                  * An empty function by default, but provided so that you can perform a custom action
21784                  * after the dragged item is dragged out of the target without dropping.
21785                  * @param {Roo.dd.DragDrop} target The drop target
21786                  * @param {Event} e The event object
21787                  * @param {String} id The id of the dragged element
21788                  * @method afterDragOut
21789                  */
21790                 this.afterDragOut(target, e, id);
21791             }
21792         }
21793         this.cachedTarget = null;
21794     },
21795
21796     /**
21797      * An empty function by default, but provided so that you can perform a custom action before the dragged
21798      * item is dragged out of the target without dropping, and optionally cancel the onDragOut.
21799      * @param {Roo.dd.DragDrop} target The drop target
21800      * @param {Event} e The event object
21801      * @param {String} id The id of the dragged element
21802      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21803      */
21804     beforeDragOut : function(target, e, id){
21805         return true;
21806     },
21807     
21808     // private
21809     onDragDrop : function(e, id){
21810         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21811         if(this.beforeDragDrop(target, e, id) !== false){
21812             if(target.isNotifyTarget){
21813                 if(target.notifyDrop(this, e, this.dragData)){ // valid drop?
21814                     this.onValidDrop(target, e, id);
21815                 }else{
21816                     this.onInvalidDrop(target, e, id);
21817                 }
21818             }else{
21819                 this.onValidDrop(target, e, id);
21820             }
21821             
21822             if(this.afterDragDrop){
21823                 /**
21824                  * An empty function by default, but provided so that you can perform a custom action
21825                  * after a valid drag drop has occurred by providing an implementation.
21826                  * @param {Roo.dd.DragDrop} target The drop target
21827                  * @param {Event} e The event object
21828                  * @param {String} id The id of the dropped element
21829                  * @method afterDragDrop
21830                  */
21831                 this.afterDragDrop(target, e, id);
21832             }
21833         }
21834         delete this.cachedTarget;
21835     },
21836
21837     /**
21838      * An empty function by default, but provided so that you can perform a custom action before the dragged
21839      * item is dropped onto the target and optionally cancel the onDragDrop.
21840      * @param {Roo.dd.DragDrop} target The drop target
21841      * @param {Event} e The event object
21842      * @param {String} id The id of the dragged element
21843      * @return {Boolean} isValid True if the drag drop event is valid, else false to cancel
21844      */
21845     beforeDragDrop : function(target, e, id){
21846         return true;
21847     },
21848
21849     // private
21850     onValidDrop : function(target, e, id){
21851         this.hideProxy();
21852         if(this.afterValidDrop){
21853             /**
21854              * An empty function by default, but provided so that you can perform a custom action
21855              * after a valid drop has occurred by providing an implementation.
21856              * @param {Object} target The target DD 
21857              * @param {Event} e The event object
21858              * @param {String} id The id of the dropped element
21859              * @method afterInvalidDrop
21860              */
21861             this.afterValidDrop(target, e, id);
21862         }
21863     },
21864
21865     // private
21866     getRepairXY : function(e, data){
21867         return this.el.getXY();  
21868     },
21869
21870     // private
21871     onInvalidDrop : function(target, e, id){
21872         this.beforeInvalidDrop(target, e, id);
21873         if(this.cachedTarget){
21874             if(this.cachedTarget.isNotifyTarget){
21875                 this.cachedTarget.notifyOut(this, e, this.dragData);
21876             }
21877             this.cacheTarget = null;
21878         }
21879         this.proxy.repair(this.getRepairXY(e, this.dragData), this.afterRepair, this);
21880
21881         if(this.afterInvalidDrop){
21882             /**
21883              * An empty function by default, but provided so that you can perform a custom action
21884              * after an invalid drop has occurred by providing an implementation.
21885              * @param {Event} e The event object
21886              * @param {String} id The id of the dropped element
21887              * @method afterInvalidDrop
21888              */
21889             this.afterInvalidDrop(e, id);
21890         }
21891     },
21892
21893     // private
21894     afterRepair : function(){
21895         if(Roo.enableFx){
21896             this.el.highlight(this.hlColor || "c3daf9");
21897         }
21898         this.dragging = false;
21899     },
21900
21901     /**
21902      * An empty function by default, but provided so that you can perform a custom action after an invalid
21903      * drop has occurred.
21904      * @param {Roo.dd.DragDrop} target The drop target
21905      * @param {Event} e The event object
21906      * @param {String} id The id of the dragged element
21907      * @return {Boolean} isValid True if the invalid drop should proceed, else false to cancel
21908      */
21909     beforeInvalidDrop : function(target, e, id){
21910         return true;
21911     },
21912
21913     // private
21914     handleMouseDown : function(e){
21915         if(this.dragging) {
21916             return;
21917         }
21918         var data = this.getDragData(e);
21919         if(data && this.onBeforeDrag(data, e) !== false){
21920             this.dragData = data;
21921             this.proxy.stop();
21922             Roo.dd.DragSource.superclass.handleMouseDown.apply(this, arguments);
21923         } 
21924     },
21925
21926     /**
21927      * An empty function by default, but provided so that you can perform a custom action before the initial
21928      * drag event begins and optionally cancel it.
21929      * @param {Object} data An object containing arbitrary data to be shared with drop targets
21930      * @param {Event} e The event object
21931      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21932      */
21933     onBeforeDrag : function(data, e){
21934         return true;
21935     },
21936
21937     /**
21938      * An empty function by default, but provided so that you can perform a custom action once the initial
21939      * drag event has begun.  The drag cannot be canceled from this function.
21940      * @param {Number} x The x position of the click on the dragged object
21941      * @param {Number} y The y position of the click on the dragged object
21942      */
21943     onStartDrag : Roo.emptyFn,
21944
21945     // private - YUI override
21946     startDrag : function(x, y){
21947         this.proxy.reset();
21948         this.dragging = true;
21949         this.proxy.update("");
21950         this.onInitDrag(x, y);
21951         this.proxy.show();
21952     },
21953
21954     // private
21955     onInitDrag : function(x, y){
21956         var clone = this.el.dom.cloneNode(true);
21957         clone.id = Roo.id(); // prevent duplicate ids
21958         this.proxy.update(clone);
21959         this.onStartDrag(x, y);
21960         return true;
21961     },
21962
21963     /**
21964      * Returns the drag source's underlying {@link Roo.dd.StatusProxy}
21965      * @return {Roo.dd.StatusProxy} proxy The StatusProxy
21966      */
21967     getProxy : function(){
21968         return this.proxy;  
21969     },
21970
21971     /**
21972      * Hides the drag source's {@link Roo.dd.StatusProxy}
21973      */
21974     hideProxy : function(){
21975         this.proxy.hide();  
21976         this.proxy.reset(true);
21977         this.dragging = false;
21978     },
21979
21980     // private
21981     triggerCacheRefresh : function(){
21982         Roo.dd.DDM.refreshCache(this.groups);
21983     },
21984
21985     // private - override to prevent hiding
21986     b4EndDrag: function(e) {
21987     },
21988
21989     // private - override to prevent moving
21990     endDrag : function(e){
21991         this.onEndDrag(this.dragData, e);
21992     },
21993
21994     // private
21995     onEndDrag : function(data, e){
21996     },
21997     
21998     // private - pin to cursor
21999     autoOffset : function(x, y) {
22000         this.setDelta(-12, -20);
22001     }    
22002 });/*
22003  * Based on:
22004  * Ext JS Library 1.1.1
22005  * Copyright(c) 2006-2007, Ext JS, LLC.
22006  *
22007  * Originally Released Under LGPL - original licence link has changed is not relivant.
22008  *
22009  * Fork - LGPL
22010  * <script type="text/javascript">
22011  */
22012
22013
22014 /**
22015  * @class Roo.dd.DropTarget
22016  * @extends Roo.dd.DDTarget
22017  * A simple class that provides the basic implementation needed to make any element a drop target that can have
22018  * draggable items dropped onto it.  The drop has no effect until an implementation of notifyDrop is provided.
22019  * @constructor
22020  * @param {String/HTMLElement/Element} el The container element
22021  * @param {Object} config
22022  */
22023 Roo.dd.DropTarget = function(el, config){
22024     this.el = Roo.get(el);
22025     
22026     var listeners = false; ;
22027     if (config && config.listeners) {
22028         listeners= config.listeners;
22029         delete config.listeners;
22030     }
22031     Roo.apply(this, config);
22032     
22033     if(this.containerScroll){
22034         Roo.dd.ScrollManager.register(this.el);
22035     }
22036     this.addEvents( {
22037          /**
22038          * @scope Roo.dd.DropTarget
22039          */
22040          
22041          /**
22042          * @event enter
22043          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the source is now over the
22044          * target.  This default implementation adds the CSS class specified by overClass (if any) to the drop element
22045          * and returns the dropAllowed config value.  This method should be overridden if drop validation is required.
22046          * 
22047          * IMPORTANT : it should set this.overClass and this.dropAllowed
22048          * 
22049          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22050          * @param {Event} e The event
22051          * @param {Object} data An object containing arbitrary data supplied by the drag source
22052          */
22053         "enter" : true,
22054         
22055          /**
22056          * @event over
22057          * The function a {@link Roo.dd.DragSource} calls continuously while it is being dragged over the target.
22058          * This method will be called on every mouse movement while the drag source is over the drop target.
22059          * This default implementation simply returns the dropAllowed config value.
22060          * 
22061          * IMPORTANT : it should set this.dropAllowed
22062          * 
22063          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22064          * @param {Event} e The event
22065          * @param {Object} data An object containing arbitrary data supplied by the drag source
22066          
22067          */
22068         "over" : true,
22069         /**
22070          * @event out
22071          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the source has been dragged
22072          * out of the target without dropping.  This default implementation simply removes the CSS class specified by
22073          * overClass (if any) from the drop element.
22074          * 
22075          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22076          * @param {Event} e The event
22077          * @param {Object} data An object containing arbitrary data supplied by the drag source
22078          */
22079          "out" : true,
22080          
22081         /**
22082          * @event drop
22083          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the dragged item has
22084          * been dropped on it.  This method has no default implementation and returns false, so you must provide an
22085          * implementation that does something to process the drop event and returns true so that the drag source's
22086          * repair action does not run.
22087          * 
22088          * IMPORTANT : it should set this.success
22089          * 
22090          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22091          * @param {Event} e The event
22092          * @param {Object} data An object containing arbitrary data supplied by the drag source
22093         */
22094          "drop" : true
22095     });
22096             
22097      
22098     Roo.dd.DropTarget.superclass.constructor.call(  this, 
22099         this.el.dom, 
22100         this.ddGroup || this.group,
22101         {
22102             isTarget: true,
22103             listeners : listeners || {} 
22104            
22105         
22106         }
22107     );
22108
22109 };
22110
22111 Roo.extend(Roo.dd.DropTarget, Roo.dd.DDTarget, {
22112     /**
22113      * @cfg {String} overClass
22114      * The CSS class applied to the drop target element while the drag source is over it (defaults to "").
22115      */
22116      /**
22117      * @cfg {String} ddGroup
22118      * The drag drop group to handle drop events for
22119      */
22120      
22121     /**
22122      * @cfg {String} dropAllowed
22123      * The CSS class returned to the drag source when drop is allowed (defaults to "x-dd-drop-ok").
22124      */
22125     dropAllowed : "x-dd-drop-ok",
22126     /**
22127      * @cfg {String} dropNotAllowed
22128      * The CSS class returned to the drag source when drop is not allowed (defaults to "x-dd-drop-nodrop").
22129      */
22130     dropNotAllowed : "x-dd-drop-nodrop",
22131     /**
22132      * @cfg {boolean} success
22133      * set this after drop listener.. 
22134      */
22135     success : false,
22136     /**
22137      * @cfg {boolean|String} valid true/false or string (ok-add/ok-sub/ok/nodrop)
22138      * if the drop point is valid for over/enter..
22139      */
22140     valid : false,
22141     // private
22142     isTarget : true,
22143
22144     // private
22145     isNotifyTarget : true,
22146     
22147     /**
22148      * @hide
22149      */
22150     notifyEnter : function(dd, e, data)
22151     {
22152         this.valid = true;
22153         this.fireEvent('enter', dd, e, data);
22154         if(this.overClass){
22155             this.el.addClass(this.overClass);
22156         }
22157         return typeof(this.valid) == 'string' ? 'x-dd-drop-' + this.valid : (
22158             this.valid ? this.dropAllowed : this.dropNotAllowed
22159         );
22160     },
22161
22162     /**
22163      * @hide
22164      */
22165     notifyOver : function(dd, e, data)
22166     {
22167         this.valid = true;
22168         this.fireEvent('over', dd, e, data);
22169         return typeof(this.valid) == 'string' ? 'x-dd-drop-' + this.valid : (
22170             this.valid ? this.dropAllowed : this.dropNotAllowed
22171         );
22172     },
22173
22174     /**
22175      * @hide
22176      */
22177     notifyOut : function(dd, e, data)
22178     {
22179         this.fireEvent('out', dd, e, data);
22180         if(this.overClass){
22181             this.el.removeClass(this.overClass);
22182         }
22183     },
22184
22185     /**
22186      * @hide
22187      */
22188     notifyDrop : function(dd, e, data)
22189     {
22190         this.success = false;
22191         this.fireEvent('drop', dd, e, data);
22192         return this.success;
22193     }
22194 });/*
22195  * Based on:
22196  * Ext JS Library 1.1.1
22197  * Copyright(c) 2006-2007, Ext JS, LLC.
22198  *
22199  * Originally Released Under LGPL - original licence link has changed is not relivant.
22200  *
22201  * Fork - LGPL
22202  * <script type="text/javascript">
22203  */
22204
22205
22206 /**
22207  * @class Roo.dd.DragZone
22208  * @extends Roo.dd.DragSource
22209  * This class provides a container DD instance that proxies for multiple child node sources.<br />
22210  * By default, this class requires that draggable child nodes are registered with {@link Roo.dd.Registry}.
22211  * @constructor
22212  * @param {String/HTMLElement/Element} el The container element
22213  * @param {Object} config
22214  */
22215 Roo.dd.DragZone = function(el, config){
22216     Roo.dd.DragZone.superclass.constructor.call(this, el, config);
22217     if(this.containerScroll){
22218         Roo.dd.ScrollManager.register(this.el);
22219     }
22220 };
22221
22222 Roo.extend(Roo.dd.DragZone, Roo.dd.DragSource, {
22223     /**
22224      * @cfg {Boolean} containerScroll True to register this container with the Scrollmanager
22225      * for auto scrolling during drag operations.
22226      */
22227     /**
22228      * @cfg {String} hlColor The color to use when visually highlighting the drag source in the afterRepair
22229      * method after a failed drop (defaults to "c3daf9" - light blue)
22230      */
22231
22232     /**
22233      * Called when a mousedown occurs in this container. Looks in {@link Roo.dd.Registry}
22234      * for a valid target to drag based on the mouse down. Override this method
22235      * to provide your own lookup logic (e.g. finding a child by class name). Make sure your returned
22236      * object has a "ddel" attribute (with an HTML Element) for other functions to work.
22237      * @param {EventObject} e The mouse down event
22238      * @return {Object} The dragData
22239      */
22240     getDragData : function(e){
22241         return Roo.dd.Registry.getHandleFromEvent(e);
22242     },
22243     
22244     /**
22245      * Called once drag threshold has been reached to initialize the proxy element. By default, it clones the
22246      * this.dragData.ddel
22247      * @param {Number} x The x position of the click on the dragged object
22248      * @param {Number} y The y position of the click on the dragged object
22249      * @return {Boolean} true to continue the drag, false to cancel
22250      */
22251     onInitDrag : function(x, y){
22252         this.proxy.update(this.dragData.ddel.cloneNode(true));
22253         this.onStartDrag(x, y);
22254         return true;
22255     },
22256     
22257     /**
22258      * Called after a repair of an invalid drop. By default, highlights this.dragData.ddel 
22259      */
22260     afterRepair : function(){
22261         if(Roo.enableFx){
22262             Roo.Element.fly(this.dragData.ddel).highlight(this.hlColor || "c3daf9");
22263         }
22264         this.dragging = false;
22265     },
22266
22267     /**
22268      * Called before a repair of an invalid drop to get the XY to animate to. By default returns
22269      * the XY of this.dragData.ddel
22270      * @param {EventObject} e The mouse up event
22271      * @return {Array} The xy location (e.g. [100, 200])
22272      */
22273     getRepairXY : function(e){
22274         return Roo.Element.fly(this.dragData.ddel).getXY();  
22275     }
22276 });/*
22277  * Based on:
22278  * Ext JS Library 1.1.1
22279  * Copyright(c) 2006-2007, Ext JS, LLC.
22280  *
22281  * Originally Released Under LGPL - original licence link has changed is not relivant.
22282  *
22283  * Fork - LGPL
22284  * <script type="text/javascript">
22285  */
22286 /**
22287  * @class Roo.dd.DropZone
22288  * @extends Roo.dd.DropTarget
22289  * This class provides a container DD instance that proxies for multiple child node targets.<br />
22290  * By default, this class requires that child nodes accepting drop are registered with {@link Roo.dd.Registry}.
22291  * @constructor
22292  * @param {String/HTMLElement/Element} el The container element
22293  * @param {Object} config
22294  */
22295 Roo.dd.DropZone = function(el, config){
22296     Roo.dd.DropZone.superclass.constructor.call(this, el, config);
22297 };
22298
22299 Roo.extend(Roo.dd.DropZone, Roo.dd.DropTarget, {
22300     /**
22301      * Returns a custom data object associated with the DOM node that is the target of the event.  By default
22302      * this looks up the event target in the {@link Roo.dd.Registry}, although you can override this method to
22303      * provide your own custom lookup.
22304      * @param {Event} e The event
22305      * @return {Object} data The custom data
22306      */
22307     getTargetFromEvent : function(e){
22308         return Roo.dd.Registry.getTargetFromEvent(e);
22309     },
22310
22311     /**
22312      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has entered a drop node
22313      * that it has registered.  This method has no default implementation and should be overridden to provide
22314      * node-specific processing if necessary.
22315      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from 
22316      * {@link #getTargetFromEvent} for this node)
22317      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22318      * @param {Event} e The event
22319      * @param {Object} data An object containing arbitrary data supplied by the drag source
22320      */
22321     onNodeEnter : function(n, dd, e, data){
22322         
22323     },
22324
22325     /**
22326      * Called internally while the DropZone determines that a {@link Roo.dd.DragSource} is over a drop node
22327      * that it has registered.  The default implementation returns this.dropNotAllowed, so it should be
22328      * overridden to provide the proper feedback.
22329      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22330      * {@link #getTargetFromEvent} for this node)
22331      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22332      * @param {Event} e The event
22333      * @param {Object} data An object containing arbitrary data supplied by the drag source
22334      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22335      * underlying {@link Roo.dd.StatusProxy} can be updated
22336      */
22337     onNodeOver : function(n, dd, e, data){
22338         return this.dropAllowed;
22339     },
22340
22341     /**
22342      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dragged out of
22343      * the drop node without dropping.  This method has no default implementation and should be overridden to provide
22344      * node-specific processing if necessary.
22345      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22346      * {@link #getTargetFromEvent} for this node)
22347      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22348      * @param {Event} e The event
22349      * @param {Object} data An object containing arbitrary data supplied by the drag source
22350      */
22351     onNodeOut : function(n, dd, e, data){
22352         
22353     },
22354
22355     /**
22356      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dropped onto
22357      * the drop node.  The default implementation returns false, so it should be overridden to provide the
22358      * appropriate processing of the drop event and return true so that the drag source's repair action does not run.
22359      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22360      * {@link #getTargetFromEvent} for this node)
22361      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22362      * @param {Event} e The event
22363      * @param {Object} data An object containing arbitrary data supplied by the drag source
22364      * @return {Boolean} True if the drop was valid, else false
22365      */
22366     onNodeDrop : function(n, dd, e, data){
22367         return false;
22368     },
22369
22370     /**
22371      * Called internally while the DropZone determines that a {@link Roo.dd.DragSource} is being dragged over it,
22372      * but not over any of its registered drop nodes.  The default implementation returns this.dropNotAllowed, so
22373      * it should be overridden to provide the proper feedback if necessary.
22374      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22375      * @param {Event} e The event
22376      * @param {Object} data An object containing arbitrary data supplied by the drag source
22377      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22378      * underlying {@link Roo.dd.StatusProxy} can be updated
22379      */
22380     onContainerOver : function(dd, e, data){
22381         return this.dropNotAllowed;
22382     },
22383
22384     /**
22385      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dropped on it,
22386      * but not on any of its registered drop nodes.  The default implementation returns false, so it should be
22387      * overridden to provide the appropriate processing of the drop event if you need the drop zone itself to
22388      * be able to accept drops.  It should return true when valid so that the drag source's repair action does not run.
22389      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22390      * @param {Event} e The event
22391      * @param {Object} data An object containing arbitrary data supplied by the drag source
22392      * @return {Boolean} True if the drop was valid, else false
22393      */
22394     onContainerDrop : function(dd, e, data){
22395         return false;
22396     },
22397
22398     /**
22399      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the source is now over
22400      * the zone.  The default implementation returns this.dropNotAllowed and expects that only registered drop
22401      * nodes can process drag drop operations, so if you need the drop zone itself to be able to process drops
22402      * you should override this method and provide a custom implementation.
22403      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22404      * @param {Event} e The event
22405      * @param {Object} data An object containing arbitrary data supplied by the drag source
22406      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22407      * underlying {@link Roo.dd.StatusProxy} can be updated
22408      */
22409     notifyEnter : function(dd, e, data){
22410         return this.dropNotAllowed;
22411     },
22412
22413     /**
22414      * The function a {@link Roo.dd.DragSource} calls continuously while it is being dragged over the drop zone.
22415      * This method will be called on every mouse movement while the drag source is over the drop zone.
22416      * It will call {@link #onNodeOver} while the drag source is over a registered node, and will also automatically
22417      * delegate to the appropriate node-specific methods as necessary when the drag source enters and exits
22418      * registered nodes ({@link #onNodeEnter}, {@link #onNodeOut}). If the drag source is not currently over a
22419      * registered node, it will call {@link #onContainerOver}.
22420      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22421      * @param {Event} e The event
22422      * @param {Object} data An object containing arbitrary data supplied by the drag source
22423      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22424      * underlying {@link Roo.dd.StatusProxy} can be updated
22425      */
22426     notifyOver : function(dd, e, data){
22427         var n = this.getTargetFromEvent(e);
22428         if(!n){ // not over valid drop target
22429             if(this.lastOverNode){
22430                 this.onNodeOut(this.lastOverNode, dd, e, data);
22431                 this.lastOverNode = null;
22432             }
22433             return this.onContainerOver(dd, e, data);
22434         }
22435         if(this.lastOverNode != n){
22436             if(this.lastOverNode){
22437                 this.onNodeOut(this.lastOverNode, dd, e, data);
22438             }
22439             this.onNodeEnter(n, dd, e, data);
22440             this.lastOverNode = n;
22441         }
22442         return this.onNodeOver(n, dd, e, data);
22443     },
22444
22445     /**
22446      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the source has been dragged
22447      * out of the zone without dropping.  If the drag source is currently over a registered node, the notification
22448      * will be delegated to {@link #onNodeOut} for node-specific handling, otherwise it will be ignored.
22449      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22450      * @param {Event} e The event
22451      * @param {Object} data An object containing arbitrary data supplied by the drag zone
22452      */
22453     notifyOut : function(dd, e, data){
22454         if(this.lastOverNode){
22455             this.onNodeOut(this.lastOverNode, dd, e, data);
22456             this.lastOverNode = null;
22457         }
22458     },
22459
22460     /**
22461      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the dragged item has
22462      * been dropped on it.  The drag zone will look up the target node based on the event passed in, and if there
22463      * is a node registered for that event, it will delegate to {@link #onNodeDrop} for node-specific handling,
22464      * otherwise it will call {@link #onContainerDrop}.
22465      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22466      * @param {Event} e The event
22467      * @param {Object} data An object containing arbitrary data supplied by the drag source
22468      * @return {Boolean} True if the drop was valid, else false
22469      */
22470     notifyDrop : function(dd, e, data){
22471         if(this.lastOverNode){
22472             this.onNodeOut(this.lastOverNode, dd, e, data);
22473             this.lastOverNode = null;
22474         }
22475         var n = this.getTargetFromEvent(e);
22476         return n ?
22477             this.onNodeDrop(n, dd, e, data) :
22478             this.onContainerDrop(dd, e, data);
22479     },
22480
22481     // private
22482     triggerCacheRefresh : function(){
22483         Roo.dd.DDM.refreshCache(this.groups);
22484     }  
22485 });/*
22486  * Based on:
22487  * Ext JS Library 1.1.1
22488  * Copyright(c) 2006-2007, Ext JS, LLC.
22489  *
22490  * Originally Released Under LGPL - original licence link has changed is not relivant.
22491  *
22492  * Fork - LGPL
22493  * <script type="text/javascript">
22494  */
22495
22496
22497 /**
22498  * @class Roo.data.SortTypes
22499  * @singleton
22500  * Defines the default sorting (casting?) comparison functions used when sorting data.
22501  */
22502 Roo.data.SortTypes = {
22503     /**
22504      * Default sort that does nothing
22505      * @param {Mixed} s The value being converted
22506      * @return {Mixed} The comparison value
22507      */
22508     none : function(s){
22509         return s;
22510     },
22511     
22512     /**
22513      * The regular expression used to strip tags
22514      * @type {RegExp}
22515      * @property
22516      */
22517     stripTagsRE : /<\/?[^>]+>/gi,
22518     
22519     /**
22520      * Strips all HTML tags to sort on text only
22521      * @param {Mixed} s The value being converted
22522      * @return {String} The comparison value
22523      */
22524     asText : function(s){
22525         return String(s).replace(this.stripTagsRE, "");
22526     },
22527     
22528     /**
22529      * Strips all HTML tags to sort on text only - Case insensitive
22530      * @param {Mixed} s The value being converted
22531      * @return {String} The comparison value
22532      */
22533     asUCText : function(s){
22534         return String(s).toUpperCase().replace(this.stripTagsRE, "");
22535     },
22536     
22537     /**
22538      * Case insensitive string
22539      * @param {Mixed} s The value being converted
22540      * @return {String} The comparison value
22541      */
22542     asUCString : function(s) {
22543         return String(s).toUpperCase();
22544     },
22545     
22546     /**
22547      * Date sorting
22548      * @param {Mixed} s The value being converted
22549      * @return {Number} The comparison value
22550      */
22551     asDate : function(s) {
22552         if(!s){
22553             return 0;
22554         }
22555         if(s instanceof Date){
22556             return s.getTime();
22557         }
22558         return Date.parse(String(s));
22559     },
22560     
22561     /**
22562      * Float sorting
22563      * @param {Mixed} s The value being converted
22564      * @return {Float} The comparison value
22565      */
22566     asFloat : function(s) {
22567         var val = parseFloat(String(s).replace(/,/g, ""));
22568         if(isNaN(val)) {
22569             val = 0;
22570         }
22571         return val;
22572     },
22573     
22574     /**
22575      * Integer sorting
22576      * @param {Mixed} s The value being converted
22577      * @return {Number} The comparison value
22578      */
22579     asInt : function(s) {
22580         var val = parseInt(String(s).replace(/,/g, ""));
22581         if(isNaN(val)) {
22582             val = 0;
22583         }
22584         return val;
22585     }
22586 };/*
22587  * Based on:
22588  * Ext JS Library 1.1.1
22589  * Copyright(c) 2006-2007, Ext JS, LLC.
22590  *
22591  * Originally Released Under LGPL - original licence link has changed is not relivant.
22592  *
22593  * Fork - LGPL
22594  * <script type="text/javascript">
22595  */
22596
22597 /**
22598 * @class Roo.data.Record
22599  * Instances of this class encapsulate both record <em>definition</em> information, and record
22600  * <em>value</em> information for use in {@link Roo.data.Store} objects, or any code which needs
22601  * to access Records cached in an {@link Roo.data.Store} object.<br>
22602  * <p>
22603  * Constructors for this class are generated by passing an Array of field definition objects to {@link #create}.
22604  * Instances are usually only created by {@link Roo.data.Reader} implementations when processing unformatted data
22605  * objects.<br>
22606  * <p>
22607  * Record objects generated by this constructor inherit all the methods of Roo.data.Record listed below.
22608  * @constructor
22609  * This constructor should not be used to create Record objects. Instead, use the constructor generated by
22610  * {@link #create}. The parameters are the same.
22611  * @param {Array} data An associative Array of data values keyed by the field name.
22612  * @param {Object} id (Optional) The id of the record. This id should be unique, and is used by the
22613  * {@link Roo.data.Store} object which owns the Record to index its collection of Records. If
22614  * not specified an integer id is generated.
22615  */
22616 Roo.data.Record = function(data, id){
22617     this.id = (id || id === 0) ? id : ++Roo.data.Record.AUTO_ID;
22618     this.data = data;
22619 };
22620
22621 /**
22622  * Generate a constructor for a specific record layout.
22623  * @param {Array} o An Array of field definition objects which specify field names, and optionally,
22624  * data types, and a mapping for an {@link Roo.data.Reader} to extract the field's value from a data object.
22625  * Each field definition object may contain the following properties: <ul>
22626  * <li><b>name</b> : String<p style="margin-left:1em">The name by which the field is referenced within the Record. This is referenced by,
22627  * for example the <em>dataIndex</em> property in column definition objects passed to {@link Roo.grid.ColumnModel}</p></li>
22628  * <li><b>mapping</b> : String<p style="margin-left:1em">(Optional) A path specification for use by the {@link Roo.data.Reader} implementation
22629  * that is creating the Record to access the data value from the data object. If an {@link Roo.data.JsonReader}
22630  * is being used, then this is a string containing the javascript expression to reference the data relative to 
22631  * the record item's root. If an {@link Roo.data.XmlReader} is being used, this is an {@link Roo.DomQuery} path
22632  * to the data item relative to the record element. If the mapping expression is the same as the field name,
22633  * this may be omitted.</p></li>
22634  * <li><b>type</b> : String<p style="margin-left:1em">(Optional) The data type for conversion to displayable value. Possible values are
22635  * <ul><li>auto (Default, implies no conversion)</li>
22636  * <li>string</li>
22637  * <li>int</li>
22638  * <li>float</li>
22639  * <li>boolean</li>
22640  * <li>date</li></ul></p></li>
22641  * <li><b>sortType</b> : Mixed<p style="margin-left:1em">(Optional) A member of {@link Roo.data.SortTypes}.</p></li>
22642  * <li><b>sortDir</b> : String<p style="margin-left:1em">(Optional) Initial direction to sort. "ASC" or "DESC"</p></li>
22643  * <li><b>convert</b> : Function<p style="margin-left:1em">(Optional) A function which converts the value provided
22644  * by the Reader into an object that will be stored in the Record. It is passed the
22645  * following parameters:<ul>
22646  * <li><b>v</b> : Mixed<p style="margin-left:1em">The data value as read by the Reader.</p></li>
22647  * </ul></p></li>
22648  * <li><b>dateFormat</b> : String<p style="margin-left:1em">(Optional) A format String for the Date.parseDate function.</p></li>
22649  * </ul>
22650  * <br>usage:<br><pre><code>
22651 var TopicRecord = Roo.data.Record.create(
22652     {name: 'title', mapping: 'topic_title'},
22653     {name: 'author', mapping: 'username'},
22654     {name: 'totalPosts', mapping: 'topic_replies', type: 'int'},
22655     {name: 'lastPost', mapping: 'post_time', type: 'date'},
22656     {name: 'lastPoster', mapping: 'user2'},
22657     {name: 'excerpt', mapping: 'post_text'}
22658 );
22659
22660 var myNewRecord = new TopicRecord({
22661     title: 'Do my job please',
22662     author: 'noobie',
22663     totalPosts: 1,
22664     lastPost: new Date(),
22665     lastPoster: 'Animal',
22666     excerpt: 'No way dude!'
22667 });
22668 myStore.add(myNewRecord);
22669 </code></pre>
22670  * @method create
22671  * @static
22672  */
22673 Roo.data.Record.create = function(o){
22674     var f = function(){
22675         f.superclass.constructor.apply(this, arguments);
22676     };
22677     Roo.extend(f, Roo.data.Record);
22678     var p = f.prototype;
22679     p.fields = new Roo.util.MixedCollection(false, function(field){
22680         return field.name;
22681     });
22682     for(var i = 0, len = o.length; i < len; i++){
22683         p.fields.add(new Roo.data.Field(o[i]));
22684     }
22685     f.getField = function(name){
22686         return p.fields.get(name);  
22687     };
22688     return f;
22689 };
22690
22691 Roo.data.Record.AUTO_ID = 1000;
22692 Roo.data.Record.EDIT = 'edit';
22693 Roo.data.Record.REJECT = 'reject';
22694 Roo.data.Record.COMMIT = 'commit';
22695
22696 Roo.data.Record.prototype = {
22697     /**
22698      * Readonly flag - true if this record has been modified.
22699      * @type Boolean
22700      */
22701     dirty : false,
22702     editing : false,
22703     error: null,
22704     modified: null,
22705
22706     // private
22707     join : function(store){
22708         this.store = store;
22709     },
22710
22711     /**
22712      * Set the named field to the specified value.
22713      * @param {String} name The name of the field to set.
22714      * @param {Object} value The value to set the field to.
22715      */
22716     set : function(name, value){
22717         if(this.data[name] == value){
22718             return;
22719         }
22720         this.dirty = true;
22721         if(!this.modified){
22722             this.modified = {};
22723         }
22724         if(typeof this.modified[name] == 'undefined'){
22725             this.modified[name] = this.data[name];
22726         }
22727         this.data[name] = value;
22728         if(!this.editing && this.store){
22729             this.store.afterEdit(this);
22730         }       
22731     },
22732
22733     /**
22734      * Get the value of the named field.
22735      * @param {String} name The name of the field to get the value of.
22736      * @return {Object} The value of the field.
22737      */
22738     get : function(name){
22739         return this.data[name]; 
22740     },
22741
22742     // private
22743     beginEdit : function(){
22744         this.editing = true;
22745         this.modified = {}; 
22746     },
22747
22748     // private
22749     cancelEdit : function(){
22750         this.editing = false;
22751         delete this.modified;
22752     },
22753
22754     // private
22755     endEdit : function(){
22756         this.editing = false;
22757         if(this.dirty && this.store){
22758             this.store.afterEdit(this);
22759         }
22760     },
22761
22762     /**
22763      * Usually called by the {@link Roo.data.Store} which owns the Record.
22764      * Rejects all changes made to the Record since either creation, or the last commit operation.
22765      * Modified fields are reverted to their original values.
22766      * <p>
22767      * Developers should subscribe to the {@link Roo.data.Store#update} event to have their code notified
22768      * of reject operations.
22769      */
22770     reject : function(){
22771         var m = this.modified;
22772         for(var n in m){
22773             if(typeof m[n] != "function"){
22774                 this.data[n] = m[n];
22775             }
22776         }
22777         this.dirty = false;
22778         delete this.modified;
22779         this.editing = false;
22780         if(this.store){
22781             this.store.afterReject(this);
22782         }
22783     },
22784
22785     /**
22786      * Usually called by the {@link Roo.data.Store} which owns the Record.
22787      * Commits all changes made to the Record since either creation, or the last commit operation.
22788      * <p>
22789      * Developers should subscribe to the {@link Roo.data.Store#update} event to have their code notified
22790      * of commit operations.
22791      */
22792     commit : function(){
22793         this.dirty = false;
22794         delete this.modified;
22795         this.editing = false;
22796         if(this.store){
22797             this.store.afterCommit(this);
22798         }
22799     },
22800
22801     // private
22802     hasError : function(){
22803         return this.error != null;
22804     },
22805
22806     // private
22807     clearError : function(){
22808         this.error = null;
22809     },
22810
22811     /**
22812      * Creates a copy of this record.
22813      * @param {String} id (optional) A new record id if you don't want to use this record's id
22814      * @return {Record}
22815      */
22816     copy : function(newId) {
22817         return new this.constructor(Roo.apply({}, this.data), newId || this.id);
22818     }
22819 };/*
22820  * Based on:
22821  * Ext JS Library 1.1.1
22822  * Copyright(c) 2006-2007, Ext JS, LLC.
22823  *
22824  * Originally Released Under LGPL - original licence link has changed is not relivant.
22825  *
22826  * Fork - LGPL
22827  * <script type="text/javascript">
22828  */
22829
22830
22831
22832 /**
22833  * @class Roo.data.Store
22834  * @extends Roo.util.Observable
22835  * The Store class encapsulates a client side cache of {@link Roo.data.Record} objects which provide input data
22836  * for widgets such as the Roo.grid.Grid, or the Roo.form.ComboBox.<br>
22837  * <p>
22838  * A Store object uses an implementation of {@link Roo.data.DataProxy} to access a data object unless you call loadData() directly and pass in your data. The Store object
22839  * has no knowledge of the format of the data returned by the Proxy.<br>
22840  * <p>
22841  * A Store object uses its configured implementation of {@link Roo.data.DataReader} to create {@link Roo.data.Record}
22842  * instances from the data object. These records are cached and made available through accessor functions.
22843  * @constructor
22844  * Creates a new Store.
22845  * @param {Object} config A config object containing the objects needed for the Store to access data,
22846  * and read the data into Records.
22847  */
22848 Roo.data.Store = function(config){
22849     this.data = new Roo.util.MixedCollection(false);
22850     this.data.getKey = function(o){
22851         return o.id;
22852     };
22853     this.baseParams = {};
22854     // private
22855     this.paramNames = {
22856         "start" : "start",
22857         "limit" : "limit",
22858         "sort" : "sort",
22859         "dir" : "dir",
22860         "multisort" : "_multisort"
22861     };
22862
22863     if(config && config.data){
22864         this.inlineData = config.data;
22865         delete config.data;
22866     }
22867
22868     Roo.apply(this, config);
22869     
22870     if(this.reader){ // reader passed
22871         this.reader = Roo.factory(this.reader, Roo.data);
22872         this.reader.xmodule = this.xmodule || false;
22873         if(!this.recordType){
22874             this.recordType = this.reader.recordType;
22875         }
22876         if(this.reader.onMetaChange){
22877             this.reader.onMetaChange = this.onMetaChange.createDelegate(this);
22878         }
22879     }
22880
22881     if(this.recordType){
22882         this.fields = this.recordType.prototype.fields;
22883     }
22884     this.modified = [];
22885
22886     this.addEvents({
22887         /**
22888          * @event datachanged
22889          * Fires when the data cache has changed, and a widget which is using this Store
22890          * as a Record cache should refresh its view.
22891          * @param {Store} this
22892          */
22893         datachanged : true,
22894         /**
22895          * @event metachange
22896          * Fires when this store's reader provides new metadata (fields). This is currently only support for JsonReaders.
22897          * @param {Store} this
22898          * @param {Object} meta The JSON metadata
22899          */
22900         metachange : true,
22901         /**
22902          * @event add
22903          * Fires when Records have been added to the Store
22904          * @param {Store} this
22905          * @param {Roo.data.Record[]} records The array of Records added
22906          * @param {Number} index The index at which the record(s) were added
22907          */
22908         add : true,
22909         /**
22910          * @event remove
22911          * Fires when a Record has been removed from the Store
22912          * @param {Store} this
22913          * @param {Roo.data.Record} record The Record that was removed
22914          * @param {Number} index The index at which the record was removed
22915          */
22916         remove : true,
22917         /**
22918          * @event update
22919          * Fires when a Record has been updated
22920          * @param {Store} this
22921          * @param {Roo.data.Record} record The Record that was updated
22922          * @param {String} operation The update operation being performed.  Value may be one of:
22923          * <pre><code>
22924  Roo.data.Record.EDIT
22925  Roo.data.Record.REJECT
22926  Roo.data.Record.COMMIT
22927          * </code></pre>
22928          */
22929         update : true,
22930         /**
22931          * @event clear
22932          * Fires when the data cache has been cleared.
22933          * @param {Store} this
22934          */
22935         clear : true,
22936         /**
22937          * @event beforeload
22938          * Fires before a request is made for a new data object.  If the beforeload handler returns false
22939          * the load action will be canceled.
22940          * @param {Store} this
22941          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22942          */
22943         beforeload : true,
22944         /**
22945          * @event beforeloadadd
22946          * Fires after a new set of Records has been loaded.
22947          * @param {Store} this
22948          * @param {Roo.data.Record[]} records The Records that were loaded
22949          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22950          */
22951         beforeloadadd : true,
22952         /**
22953          * @event load
22954          * Fires after a new set of Records has been loaded, before they are added to the store.
22955          * @param {Store} this
22956          * @param {Roo.data.Record[]} records The Records that were loaded
22957          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22958          * @params {Object} return from reader
22959          */
22960         load : true,
22961         /**
22962          * @event loadexception
22963          * Fires if an exception occurs in the Proxy during loading.
22964          * Called with the signature of the Proxy's "loadexception" event.
22965          * If you return Json { data: [] , success: false, .... } then this will be thrown with the following args
22966          * 
22967          * @param {Proxy} 
22968          * @param {Object} return from JsonData.reader() - success, totalRecords, records
22969          * @param {Object} load options 
22970          * @param {Object} jsonData from your request (normally this contains the Exception)
22971          */
22972         loadexception : true
22973     });
22974     
22975     if(this.proxy){
22976         this.proxy = Roo.factory(this.proxy, Roo.data);
22977         this.proxy.xmodule = this.xmodule || false;
22978         this.relayEvents(this.proxy,  ["loadexception"]);
22979     }
22980     this.sortToggle = {};
22981     this.sortOrder = []; // array of order of sorting - updated by grid if multisort is enabled.
22982
22983     Roo.data.Store.superclass.constructor.call(this);
22984
22985     if(this.inlineData){
22986         this.loadData(this.inlineData);
22987         delete this.inlineData;
22988     }
22989 };
22990
22991 Roo.extend(Roo.data.Store, Roo.util.Observable, {
22992      /**
22993     * @cfg {boolean} isLocal   flag if data is locally available (and can be always looked up
22994     * without a remote query - used by combo/forms at present.
22995     */
22996     
22997     /**
22998     * @cfg {Roo.data.DataProxy} proxy The Proxy object which provides access to a data object.
22999     */
23000     /**
23001     * @cfg {Array} data Inline data to be loaded when the store is initialized.
23002     */
23003     /**
23004     * @cfg {Roo.data.Reader} reader The Reader object which processes the data object and returns
23005     * an Array of Roo.data.record objects which are cached keyed by their <em>id</em> property.
23006     */
23007     /**
23008     * @cfg {Object} baseParams An object containing properties which are to be sent as parameters
23009     * on any HTTP request
23010     */
23011     /**
23012     * @cfg {Object} sortInfo A config object in the format: {field: "fieldName", direction: "ASC|DESC"}
23013     */
23014     /**
23015     * @cfg {Boolean} multiSort enable multi column sorting (sort is based on the order of columns, remote only at present)
23016     */
23017     multiSort: false,
23018     /**
23019     * @cfg {boolean} remoteSort True if sorting is to be handled by requesting the Proxy to provide a refreshed
23020     * version of the data object in sorted order, as opposed to sorting the Record cache in place (defaults to false).
23021     */
23022     remoteSort : false,
23023
23024     /**
23025     * @cfg {boolean} pruneModifiedRecords True to clear all modified record information each time the store is
23026      * loaded or when a record is removed. (defaults to false).
23027     */
23028     pruneModifiedRecords : false,
23029
23030     // private
23031     lastOptions : null,
23032
23033     /**
23034      * Add Records to the Store and fires the add event.
23035      * @param {Roo.data.Record[]} records An Array of Roo.data.Record objects to add to the cache.
23036      */
23037     add : function(records){
23038         records = [].concat(records);
23039         for(var i = 0, len = records.length; i < len; i++){
23040             records[i].join(this);
23041         }
23042         var index = this.data.length;
23043         this.data.addAll(records);
23044         this.fireEvent("add", this, records, index);
23045     },
23046
23047     /**
23048      * Remove a Record from the Store and fires the remove event.
23049      * @param {Ext.data.Record} record The Roo.data.Record object to remove from the cache.
23050      */
23051     remove : function(record){
23052         var index = this.data.indexOf(record);
23053         this.data.removeAt(index);
23054  
23055         if(this.pruneModifiedRecords){
23056             this.modified.remove(record);
23057         }
23058         this.fireEvent("remove", this, record, index);
23059     },
23060
23061     /**
23062      * Remove all Records from the Store and fires the clear event.
23063      */
23064     removeAll : function(){
23065         this.data.clear();
23066         if(this.pruneModifiedRecords){
23067             this.modified = [];
23068         }
23069         this.fireEvent("clear", this);
23070     },
23071
23072     /**
23073      * Inserts Records to the Store at the given index and fires the add event.
23074      * @param {Number} index The start index at which to insert the passed Records.
23075      * @param {Roo.data.Record[]} records An Array of Roo.data.Record objects to add to the cache.
23076      */
23077     insert : function(index, records){
23078         records = [].concat(records);
23079         for(var i = 0, len = records.length; i < len; i++){
23080             this.data.insert(index, records[i]);
23081             records[i].join(this);
23082         }
23083         this.fireEvent("add", this, records, index);
23084     },
23085
23086     /**
23087      * Get the index within the cache of the passed Record.
23088      * @param {Roo.data.Record} record The Roo.data.Record object to to find.
23089      * @return {Number} The index of the passed Record. Returns -1 if not found.
23090      */
23091     indexOf : function(record){
23092         return this.data.indexOf(record);
23093     },
23094
23095     /**
23096      * Get the index within the cache of the Record with the passed id.
23097      * @param {String} id The id of the Record to find.
23098      * @return {Number} The index of the Record. Returns -1 if not found.
23099      */
23100     indexOfId : function(id){
23101         return this.data.indexOfKey(id);
23102     },
23103
23104     /**
23105      * Get the Record with the specified id.
23106      * @param {String} id The id of the Record to find.
23107      * @return {Roo.data.Record} The Record with the passed id. Returns undefined if not found.
23108      */
23109     getById : function(id){
23110         return this.data.key(id);
23111     },
23112
23113     /**
23114      * Get the Record at the specified index.
23115      * @param {Number} index The index of the Record to find.
23116      * @return {Roo.data.Record} The Record at the passed index. Returns undefined if not found.
23117      */
23118     getAt : function(index){
23119         return this.data.itemAt(index);
23120     },
23121
23122     /**
23123      * Returns a range of Records between specified indices.
23124      * @param {Number} startIndex (optional) The starting index (defaults to 0)
23125      * @param {Number} endIndex (optional) The ending index (defaults to the last Record in the Store)
23126      * @return {Roo.data.Record[]} An array of Records
23127      */
23128     getRange : function(start, end){
23129         return this.data.getRange(start, end);
23130     },
23131
23132     // private
23133     storeOptions : function(o){
23134         o = Roo.apply({}, o);
23135         delete o.callback;
23136         delete o.scope;
23137         this.lastOptions = o;
23138     },
23139
23140     /**
23141      * Loads the Record cache from the configured Proxy using the configured Reader.
23142      * <p>
23143      * If using remote paging, then the first load call must specify the <em>start</em>
23144      * and <em>limit</em> properties in the options.params property to establish the initial
23145      * position within the dataset, and the number of Records to cache on each read from the Proxy.
23146      * <p>
23147      * <strong>It is important to note that for remote data sources, loading is asynchronous,
23148      * and this call will return before the new data has been loaded. Perform any post-processing
23149      * in a callback function, or in a "load" event handler.</strong>
23150      * <p>
23151      * @param {Object} options An object containing properties which control loading options:<ul>
23152      * <li>params {Object} An object containing properties to pass as HTTP parameters to a remote data source.</li>
23153      * <li>callback {Function} A function to be called after the Records have been loaded. The callback is
23154      * passed the following arguments:<ul>
23155      * <li>r : Roo.data.Record[]</li>
23156      * <li>options: Options object from the load call</li>
23157      * <li>success: Boolean success indicator</li></ul></li>
23158      * <li>scope {Object} Scope with which to call the callback (defaults to the Store object)</li>
23159      * <li>add {Boolean} indicator to append loaded records rather than replace the current cache.</li>
23160      * </ul>
23161      */
23162     load : function(options){
23163         options = options || {};
23164         if(this.fireEvent("beforeload", this, options) !== false){
23165             this.storeOptions(options);
23166             var p = Roo.apply(options.params || {}, this.baseParams);
23167             // if meta was not loaded from remote source.. try requesting it.
23168             if (!this.reader.metaFromRemote) {
23169                 p._requestMeta = 1;
23170             }
23171             if(this.sortInfo && this.remoteSort){
23172                 var pn = this.paramNames;
23173                 p[pn["sort"]] = this.sortInfo.field;
23174                 p[pn["dir"]] = this.sortInfo.direction;
23175             }
23176             if (this.multiSort) {
23177                 var pn = this.paramNames;
23178                 p[pn["multisort"]] = Roo.encode( { sort : this.sortToggle, order: this.sortOrder });
23179             }
23180             
23181             this.proxy.load(p, this.reader, this.loadRecords, this, options);
23182         }
23183     },
23184
23185     /**
23186      * Reloads the Record cache from the configured Proxy using the configured Reader and
23187      * the options from the last load operation performed.
23188      * @param {Object} options (optional) An object containing properties which may override the options
23189      * used in the last load operation. See {@link #load} for details (defaults to null, in which case
23190      * the most recently used options are reused).
23191      */
23192     reload : function(options){
23193         this.load(Roo.applyIf(options||{}, this.lastOptions));
23194     },
23195
23196     // private
23197     // Called as a callback by the Reader during a load operation.
23198     loadRecords : function(o, options, success){
23199         if(!o || success === false){
23200             if(success !== false){
23201                 this.fireEvent("load", this, [], options, o);
23202             }
23203             if(options.callback){
23204                 options.callback.call(options.scope || this, [], options, false);
23205             }
23206             return;
23207         }
23208         // if data returned failure - throw an exception.
23209         if (o.success === false) {
23210             // show a message if no listener is registered.
23211             if (!this.hasListener('loadexception') && typeof(o.raw.errorMsg) != 'undefined') {
23212                     Roo.MessageBox.alert("Error loading",o.raw.errorMsg);
23213             }
23214             // loadmask wil be hooked into this..
23215             this.fireEvent("loadexception", this, o, options, o.raw.errorMsg);
23216             return;
23217         }
23218         var r = o.records, t = o.totalRecords || r.length;
23219         
23220         this.fireEvent("beforeloadadd", this, r, options, o);
23221         
23222         if(!options || options.add !== true){
23223             if(this.pruneModifiedRecords){
23224                 this.modified = [];
23225             }
23226             for(var i = 0, len = r.length; i < len; i++){
23227                 r[i].join(this);
23228             }
23229             if(this.snapshot){
23230                 this.data = this.snapshot;
23231                 delete this.snapshot;
23232             }
23233             this.data.clear();
23234             this.data.addAll(r);
23235             this.totalLength = t;
23236             this.applySort();
23237             this.fireEvent("datachanged", this);
23238         }else{
23239             this.totalLength = Math.max(t, this.data.length+r.length);
23240             this.add(r);
23241         }
23242         
23243         if(this.parent && !Roo.isIOS && !this.useNativeIOS && this.parent.emptyTitle.length) {
23244                 
23245             var e = new Roo.data.Record({});
23246
23247             e.set(this.parent.displayField, this.parent.emptyTitle);
23248             e.set(this.parent.valueField, '');
23249
23250             this.insert(0, e);
23251         }
23252             
23253         this.fireEvent("load", this, r, options, o);
23254         if(options.callback){
23255             options.callback.call(options.scope || this, r, options, true);
23256         }
23257     },
23258
23259
23260     /**
23261      * Loads data from a passed data block. A Reader which understands the format of the data
23262      * must have been configured in the constructor.
23263      * @param {Object} data The data block from which to read the Records.  The format of the data expected
23264      * is dependent on the type of Reader that is configured and should correspond to that Reader's readRecords parameter.
23265      * @param {Boolean} append (Optional) True to append the new Records rather than replace the existing cache.
23266      */
23267     loadData : function(o, append){
23268         var r = this.reader.readRecords(o);
23269         this.loadRecords(r, {add: append}, true);
23270     },
23271
23272     /**
23273      * Gets the number of cached records.
23274      * <p>
23275      * <em>If using paging, this may not be the total size of the dataset. If the data object
23276      * used by the Reader contains the dataset size, then the getTotalCount() function returns
23277      * the data set size</em>
23278      */
23279     getCount : function(){
23280         return this.data.length || 0;
23281     },
23282
23283     /**
23284      * Gets the total number of records in the dataset as returned by the server.
23285      * <p>
23286      * <em>If using paging, for this to be accurate, the data object used by the Reader must contain
23287      * the dataset size</em>
23288      */
23289     getTotalCount : function(){
23290         return this.totalLength || 0;
23291     },
23292
23293     /**
23294      * Returns the sort state of the Store as an object with two properties:
23295      * <pre><code>
23296  field {String} The name of the field by which the Records are sorted
23297  direction {String} The sort order, "ASC" or "DESC"
23298      * </code></pre>
23299      */
23300     getSortState : function(){
23301         return this.sortInfo;
23302     },
23303
23304     // private
23305     applySort : function(){
23306         if(this.sortInfo && !this.remoteSort){
23307             var s = this.sortInfo, f = s.field;
23308             var st = this.fields.get(f).sortType;
23309             var fn = function(r1, r2){
23310                 var v1 = st(r1.data[f]), v2 = st(r2.data[f]);
23311                 return v1 > v2 ? 1 : (v1 < v2 ? -1 : 0);
23312             };
23313             this.data.sort(s.direction, fn);
23314             if(this.snapshot && this.snapshot != this.data){
23315                 this.snapshot.sort(s.direction, fn);
23316             }
23317         }
23318     },
23319
23320     /**
23321      * Sets the default sort column and order to be used by the next load operation.
23322      * @param {String} fieldName The name of the field to sort by.
23323      * @param {String} dir (optional) The sort order, "ASC" or "DESC" (defaults to "ASC")
23324      */
23325     setDefaultSort : function(field, dir){
23326         this.sortInfo = {field: field, direction: dir ? dir.toUpperCase() : "ASC"};
23327     },
23328
23329     /**
23330      * Sort the Records.
23331      * If remote sorting is used, the sort is performed on the server, and the cache is
23332      * reloaded. If local sorting is used, the cache is sorted internally.
23333      * @param {String} fieldName The name of the field to sort by.
23334      * @param {String} dir (optional) The sort order, "ASC" or "DESC" (defaults to "ASC")
23335      */
23336     sort : function(fieldName, dir){
23337         var f = this.fields.get(fieldName);
23338         if(!dir){
23339             this.sortToggle[f.name] = this.sortToggle[f.name] || f.sortDir;
23340             
23341             if(this.multiSort || (this.sortInfo && this.sortInfo.field == f.name) ){ // toggle sort dir
23342                 dir = (this.sortToggle[f.name] || "ASC").toggle("ASC", "DESC");
23343             }else{
23344                 dir = f.sortDir;
23345             }
23346         }
23347         this.sortToggle[f.name] = dir;
23348         this.sortInfo = {field: f.name, direction: dir};
23349         if(!this.remoteSort){
23350             this.applySort();
23351             this.fireEvent("datachanged", this);
23352         }else{
23353             this.load(this.lastOptions);
23354         }
23355     },
23356
23357     /**
23358      * Calls the specified function for each of the Records in the cache.
23359      * @param {Function} fn The function to call. The Record is passed as the first parameter.
23360      * Returning <em>false</em> aborts and exits the iteration.
23361      * @param {Object} scope (optional) The scope in which to call the function (defaults to the Record).
23362      */
23363     each : function(fn, scope){
23364         this.data.each(fn, scope);
23365     },
23366
23367     /**
23368      * Gets all records modified since the last commit.  Modified records are persisted across load operations
23369      * (e.g., during paging).
23370      * @return {Roo.data.Record[]} An array of Records containing outstanding modifications.
23371      */
23372     getModifiedRecords : function(){
23373         return this.modified;
23374     },
23375
23376     // private
23377     createFilterFn : function(property, value, anyMatch){
23378         if(!value.exec){ // not a regex
23379             value = String(value);
23380             if(value.length == 0){
23381                 return false;
23382             }
23383             value = new RegExp((anyMatch === true ? '' : '^') + Roo.escapeRe(value), "i");
23384         }
23385         return function(r){
23386             return value.test(r.data[property]);
23387         };
23388     },
23389
23390     /**
23391      * Sums the value of <i>property</i> for each record between start and end and returns the result.
23392      * @param {String} property A field on your records
23393      * @param {Number} start The record index to start at (defaults to 0)
23394      * @param {Number} end The last record index to include (defaults to length - 1)
23395      * @return {Number} The sum
23396      */
23397     sum : function(property, start, end){
23398         var rs = this.data.items, v = 0;
23399         start = start || 0;
23400         end = (end || end === 0) ? end : rs.length-1;
23401
23402         for(var i = start; i <= end; i++){
23403             v += (rs[i].data[property] || 0);
23404         }
23405         return v;
23406     },
23407
23408     /**
23409      * Filter the records by a specified property.
23410      * @param {String} field A field on your records
23411      * @param {String/RegExp} value Either a string that the field
23412      * should start with or a RegExp to test against the field
23413      * @param {Boolean} anyMatch True to match any part not just the beginning
23414      */
23415     filter : function(property, value, anyMatch){
23416         var fn = this.createFilterFn(property, value, anyMatch);
23417         return fn ? this.filterBy(fn) : this.clearFilter();
23418     },
23419
23420     /**
23421      * Filter by a function. The specified function will be called with each
23422      * record in this data source. If the function returns true the record is included,
23423      * otherwise it is filtered.
23424      * @param {Function} fn The function to be called, it will receive 2 args (record, id)
23425      * @param {Object} scope (optional) The scope of the function (defaults to this)
23426      */
23427     filterBy : function(fn, scope){
23428         this.snapshot = this.snapshot || this.data;
23429         this.data = this.queryBy(fn, scope||this);
23430         this.fireEvent("datachanged", this);
23431     },
23432
23433     /**
23434      * Query the records by a specified property.
23435      * @param {String} field A field on your records
23436      * @param {String/RegExp} value Either a string that the field
23437      * should start with or a RegExp to test against the field
23438      * @param {Boolean} anyMatch True to match any part not just the beginning
23439      * @return {MixedCollection} Returns an Roo.util.MixedCollection of the matched records
23440      */
23441     query : function(property, value, anyMatch){
23442         var fn = this.createFilterFn(property, value, anyMatch);
23443         return fn ? this.queryBy(fn) : this.data.clone();
23444     },
23445
23446     /**
23447      * Query by a function. The specified function will be called with each
23448      * record in this data source. If the function returns true the record is included
23449      * in the results.
23450      * @param {Function} fn The function to be called, it will receive 2 args (record, id)
23451      * @param {Object} scope (optional) The scope of the function (defaults to this)
23452       @return {MixedCollection} Returns an Roo.util.MixedCollection of the matched records
23453      **/
23454     queryBy : function(fn, scope){
23455         var data = this.snapshot || this.data;
23456         return data.filterBy(fn, scope||this);
23457     },
23458
23459     /**
23460      * Collects unique values for a particular dataIndex from this store.
23461      * @param {String} dataIndex The property to collect
23462      * @param {Boolean} allowNull (optional) Pass true to allow null, undefined or empty string values
23463      * @param {Boolean} bypassFilter (optional) Pass true to collect from all records, even ones which are filtered
23464      * @return {Array} An array of the unique values
23465      **/
23466     collect : function(dataIndex, allowNull, bypassFilter){
23467         var d = (bypassFilter === true && this.snapshot) ?
23468                 this.snapshot.items : this.data.items;
23469         var v, sv, r = [], l = {};
23470         for(var i = 0, len = d.length; i < len; i++){
23471             v = d[i].data[dataIndex];
23472             sv = String(v);
23473             if((allowNull || !Roo.isEmpty(v)) && !l[sv]){
23474                 l[sv] = true;
23475                 r[r.length] = v;
23476             }
23477         }
23478         return r;
23479     },
23480
23481     /**
23482      * Revert to a view of the Record cache with no filtering applied.
23483      * @param {Boolean} suppressEvent If true the filter is cleared silently without notifying listeners
23484      */
23485     clearFilter : function(suppressEvent){
23486         if(this.snapshot && this.snapshot != this.data){
23487             this.data = this.snapshot;
23488             delete this.snapshot;
23489             if(suppressEvent !== true){
23490                 this.fireEvent("datachanged", this);
23491             }
23492         }
23493     },
23494
23495     // private
23496     afterEdit : function(record){
23497         if(this.modified.indexOf(record) == -1){
23498             this.modified.push(record);
23499         }
23500         this.fireEvent("update", this, record, Roo.data.Record.EDIT);
23501     },
23502     
23503     // private
23504     afterReject : function(record){
23505         this.modified.remove(record);
23506         this.fireEvent("update", this, record, Roo.data.Record.REJECT);
23507     },
23508
23509     // private
23510     afterCommit : function(record){
23511         this.modified.remove(record);
23512         this.fireEvent("update", this, record, Roo.data.Record.COMMIT);
23513     },
23514
23515     /**
23516      * Commit all Records with outstanding changes. To handle updates for changes, subscribe to the
23517      * Store's "update" event, and perform updating when the third parameter is Roo.data.Record.COMMIT.
23518      */
23519     commitChanges : function(){
23520         var m = this.modified.slice(0);
23521         this.modified = [];
23522         for(var i = 0, len = m.length; i < len; i++){
23523             m[i].commit();
23524         }
23525     },
23526
23527     /**
23528      * Cancel outstanding changes on all changed records.
23529      */
23530     rejectChanges : function(){
23531         var m = this.modified.slice(0);
23532         this.modified = [];
23533         for(var i = 0, len = m.length; i < len; i++){
23534             m[i].reject();
23535         }
23536     },
23537
23538     onMetaChange : function(meta, rtype, o){
23539         this.recordType = rtype;
23540         this.fields = rtype.prototype.fields;
23541         delete this.snapshot;
23542         this.sortInfo = meta.sortInfo || this.sortInfo;
23543         this.modified = [];
23544         this.fireEvent('metachange', this, this.reader.meta);
23545     },
23546     
23547     moveIndex : function(data, type)
23548     {
23549         var index = this.indexOf(data);
23550         
23551         var newIndex = index + type;
23552         
23553         this.remove(data);
23554         
23555         this.insert(newIndex, data);
23556         
23557     }
23558 });/*
23559  * Based on:
23560  * Ext JS Library 1.1.1
23561  * Copyright(c) 2006-2007, Ext JS, LLC.
23562  *
23563  * Originally Released Under LGPL - original licence link has changed is not relivant.
23564  *
23565  * Fork - LGPL
23566  * <script type="text/javascript">
23567  */
23568
23569 /**
23570  * @class Roo.data.SimpleStore
23571  * @extends Roo.data.Store
23572  * Small helper class to make creating Stores from Array data easier.
23573  * @cfg {Number} id The array index of the record id. Leave blank to auto generate ids.
23574  * @cfg {Array} fields An array of field definition objects, or field name strings.
23575  * @cfg {Object} an existing reader (eg. copied from another store)
23576  * @cfg {Array} data The multi-dimensional array of data
23577  * @constructor
23578  * @param {Object} config
23579  */
23580 Roo.data.SimpleStore = function(config)
23581 {
23582     Roo.data.SimpleStore.superclass.constructor.call(this, {
23583         isLocal : true,
23584         reader: typeof(config.reader) != 'undefined' ? config.reader : new Roo.data.ArrayReader({
23585                 id: config.id
23586             },
23587             Roo.data.Record.create(config.fields)
23588         ),
23589         proxy : new Roo.data.MemoryProxy(config.data)
23590     });
23591     this.load();
23592 };
23593 Roo.extend(Roo.data.SimpleStore, Roo.data.Store);/*
23594  * Based on:
23595  * Ext JS Library 1.1.1
23596  * Copyright(c) 2006-2007, Ext JS, LLC.
23597  *
23598  * Originally Released Under LGPL - original licence link has changed is not relivant.
23599  *
23600  * Fork - LGPL
23601  * <script type="text/javascript">
23602  */
23603
23604 /**
23605 /**
23606  * @extends Roo.data.Store
23607  * @class Roo.data.JsonStore
23608  * Small helper class to make creating Stores for JSON data easier. <br/>
23609 <pre><code>
23610 var store = new Roo.data.JsonStore({
23611     url: 'get-images.php',
23612     root: 'images',
23613     fields: ['name', 'url', {name:'size', type: 'float'}, {name:'lastmod', type:'date'}]
23614 });
23615 </code></pre>
23616  * <b>Note: Although they are not listed, this class inherits all of the config options of Store,
23617  * JsonReader and HttpProxy (unless inline data is provided).</b>
23618  * @cfg {Array} fields An array of field definition objects, or field name strings.
23619  * @constructor
23620  * @param {Object} config
23621  */
23622 Roo.data.JsonStore = function(c){
23623     Roo.data.JsonStore.superclass.constructor.call(this, Roo.apply(c, {
23624         proxy: !c.data ? new Roo.data.HttpProxy({url: c.url}) : undefined,
23625         reader: new Roo.data.JsonReader(c, c.fields)
23626     }));
23627 };
23628 Roo.extend(Roo.data.JsonStore, Roo.data.Store);/*
23629  * Based on:
23630  * Ext JS Library 1.1.1
23631  * Copyright(c) 2006-2007, Ext JS, LLC.
23632  *
23633  * Originally Released Under LGPL - original licence link has changed is not relivant.
23634  *
23635  * Fork - LGPL
23636  * <script type="text/javascript">
23637  */
23638
23639  
23640 Roo.data.Field = function(config){
23641     if(typeof config == "string"){
23642         config = {name: config};
23643     }
23644     Roo.apply(this, config);
23645     
23646     if(!this.type){
23647         this.type = "auto";
23648     }
23649     
23650     var st = Roo.data.SortTypes;
23651     // named sortTypes are supported, here we look them up
23652     if(typeof this.sortType == "string"){
23653         this.sortType = st[this.sortType];
23654     }
23655     
23656     // set default sortType for strings and dates
23657     if(!this.sortType){
23658         switch(this.type){
23659             case "string":
23660                 this.sortType = st.asUCString;
23661                 break;
23662             case "date":
23663                 this.sortType = st.asDate;
23664                 break;
23665             default:
23666                 this.sortType = st.none;
23667         }
23668     }
23669
23670     // define once
23671     var stripRe = /[\$,%]/g;
23672
23673     // prebuilt conversion function for this field, instead of
23674     // switching every time we're reading a value
23675     if(!this.convert){
23676         var cv, dateFormat = this.dateFormat;
23677         switch(this.type){
23678             case "":
23679             case "auto":
23680             case undefined:
23681                 cv = function(v){ return v; };
23682                 break;
23683             case "string":
23684                 cv = function(v){ return (v === undefined || v === null) ? '' : String(v); };
23685                 break;
23686             case "int":
23687                 cv = function(v){
23688                     return v !== undefined && v !== null && v !== '' ?
23689                            parseInt(String(v).replace(stripRe, ""), 10) : '';
23690                     };
23691                 break;
23692             case "float":
23693                 cv = function(v){
23694                     return v !== undefined && v !== null && v !== '' ?
23695                            parseFloat(String(v).replace(stripRe, ""), 10) : ''; 
23696                     };
23697                 break;
23698             case "bool":
23699             case "boolean":
23700                 cv = function(v){ return v === true || v === "true" || v == 1; };
23701                 break;
23702             case "date":
23703                 cv = function(v){
23704                     if(!v){
23705                         return '';
23706                     }
23707                     if(v instanceof Date){
23708                         return v;
23709                     }
23710                     if(dateFormat){
23711                         if(dateFormat == "timestamp"){
23712                             return new Date(v*1000);
23713                         }
23714                         return Date.parseDate(v, dateFormat);
23715                     }
23716                     var parsed = Date.parse(v);
23717                     return parsed ? new Date(parsed) : null;
23718                 };
23719              break;
23720             
23721         }
23722         this.convert = cv;
23723     }
23724 };
23725
23726 Roo.data.Field.prototype = {
23727     dateFormat: null,
23728     defaultValue: "",
23729     mapping: null,
23730     sortType : null,
23731     sortDir : "ASC"
23732 };/*
23733  * Based on:
23734  * Ext JS Library 1.1.1
23735  * Copyright(c) 2006-2007, Ext JS, LLC.
23736  *
23737  * Originally Released Under LGPL - original licence link has changed is not relivant.
23738  *
23739  * Fork - LGPL
23740  * <script type="text/javascript">
23741  */
23742  
23743 // Base class for reading structured data from a data source.  This class is intended to be
23744 // extended (see ArrayReader, JsonReader and XmlReader) and should not be created directly.
23745
23746 /**
23747  * @class Roo.data.DataReader
23748  * Base class for reading structured data from a data source.  This class is intended to be
23749  * extended (see {Roo.data.ArrayReader}, {Roo.data.JsonReader} and {Roo.data.XmlReader}) and should not be created directly.
23750  */
23751
23752 Roo.data.DataReader = function(meta, recordType){
23753     
23754     this.meta = meta;
23755     
23756     this.recordType = recordType instanceof Array ? 
23757         Roo.data.Record.create(recordType) : recordType;
23758 };
23759
23760 Roo.data.DataReader.prototype = {
23761      /**
23762      * Create an empty record
23763      * @param {Object} data (optional) - overlay some values
23764      * @return {Roo.data.Record} record created.
23765      */
23766     newRow :  function(d) {
23767         var da =  {};
23768         this.recordType.prototype.fields.each(function(c) {
23769             switch( c.type) {
23770                 case 'int' : da[c.name] = 0; break;
23771                 case 'date' : da[c.name] = new Date(); break;
23772                 case 'float' : da[c.name] = 0.0; break;
23773                 case 'boolean' : da[c.name] = false; break;
23774                 default : da[c.name] = ""; break;
23775             }
23776             
23777         });
23778         return new this.recordType(Roo.apply(da, d));
23779     }
23780     
23781     
23782 };/*
23783  * Based on:
23784  * Ext JS Library 1.1.1
23785  * Copyright(c) 2006-2007, Ext JS, LLC.
23786  *
23787  * Originally Released Under LGPL - original licence link has changed is not relivant.
23788  *
23789  * Fork - LGPL
23790  * <script type="text/javascript">
23791  */
23792
23793 /**
23794  * @class Roo.data.DataProxy
23795  * @extends Roo.data.Observable
23796  * This class is an abstract base class for implementations which provide retrieval of
23797  * unformatted data objects.<br>
23798  * <p>
23799  * DataProxy implementations are usually used in conjunction with an implementation of Roo.data.DataReader
23800  * (of the appropriate type which knows how to parse the data object) to provide a block of
23801  * {@link Roo.data.Records} to an {@link Roo.data.Store}.<br>
23802  * <p>
23803  * Custom implementations must implement the load method as described in
23804  * {@link Roo.data.HttpProxy#load}.
23805  */
23806 Roo.data.DataProxy = function(){
23807     this.addEvents({
23808         /**
23809          * @event beforeload
23810          * Fires before a network request is made to retrieve a data object.
23811          * @param {Object} This DataProxy object.
23812          * @param {Object} params The params parameter to the load function.
23813          */
23814         beforeload : true,
23815         /**
23816          * @event load
23817          * Fires before the load method's callback is called.
23818          * @param {Object} This DataProxy object.
23819          * @param {Object} o The data object.
23820          * @param {Object} arg The callback argument object passed to the load function.
23821          */
23822         load : true,
23823         /**
23824          * @event loadexception
23825          * Fires if an Exception occurs during data retrieval.
23826          * @param {Object} This DataProxy object.
23827          * @param {Object} o The data object.
23828          * @param {Object} arg The callback argument object passed to the load function.
23829          * @param {Object} e The Exception.
23830          */
23831         loadexception : true
23832     });
23833     Roo.data.DataProxy.superclass.constructor.call(this);
23834 };
23835
23836 Roo.extend(Roo.data.DataProxy, Roo.util.Observable);
23837
23838     /**
23839      * @cfg {void} listeners (Not available) Constructor blocks listeners from being set
23840      */
23841 /*
23842  * Based on:
23843  * Ext JS Library 1.1.1
23844  * Copyright(c) 2006-2007, Ext JS, LLC.
23845  *
23846  * Originally Released Under LGPL - original licence link has changed is not relivant.
23847  *
23848  * Fork - LGPL
23849  * <script type="text/javascript">
23850  */
23851 /**
23852  * @class Roo.data.MemoryProxy
23853  * An implementation of Roo.data.DataProxy that simply passes the data specified in its constructor
23854  * to the Reader when its load method is called.
23855  * @constructor
23856  * @param {Object} data The data object which the Reader uses to construct a block of Roo.data.Records.
23857  */
23858 Roo.data.MemoryProxy = function(data){
23859     if (data.data) {
23860         data = data.data;
23861     }
23862     Roo.data.MemoryProxy.superclass.constructor.call(this);
23863     this.data = data;
23864 };
23865
23866 Roo.extend(Roo.data.MemoryProxy, Roo.data.DataProxy, {
23867     
23868     /**
23869      * Load data from the requested source (in this case an in-memory
23870      * data object passed to the constructor), read the data object into
23871      * a block of Roo.data.Records using the passed Roo.data.DataReader implementation, and
23872      * process that block using the passed callback.
23873      * @param {Object} params This parameter is not used by the MemoryProxy class.
23874      * @param {Roo.data.DataReader} reader The Reader object which converts the data
23875      * object into a block of Roo.data.Records.
23876      * @param {Function} callback The function into which to pass the block of Roo.data.records.
23877      * The function must be passed <ul>
23878      * <li>The Record block object</li>
23879      * <li>The "arg" argument from the load function</li>
23880      * <li>A boolean success indicator</li>
23881      * </ul>
23882      * @param {Object} scope The scope in which to call the callback
23883      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
23884      */
23885     load : function(params, reader, callback, scope, arg){
23886         params = params || {};
23887         var result;
23888         try {
23889             result = reader.readRecords(params.data ? params.data :this.data);
23890         }catch(e){
23891             this.fireEvent("loadexception", this, arg, null, e);
23892             callback.call(scope, null, arg, false);
23893             return;
23894         }
23895         callback.call(scope, result, arg, true);
23896     },
23897     
23898     // private
23899     update : function(params, records){
23900         
23901     }
23902 });/*
23903  * Based on:
23904  * Ext JS Library 1.1.1
23905  * Copyright(c) 2006-2007, Ext JS, LLC.
23906  *
23907  * Originally Released Under LGPL - original licence link has changed is not relivant.
23908  *
23909  * Fork - LGPL
23910  * <script type="text/javascript">
23911  */
23912 /**
23913  * @class Roo.data.HttpProxy
23914  * @extends Roo.data.DataProxy
23915  * An implementation of {@link Roo.data.DataProxy} that reads a data object from an {@link Roo.data.Connection} object
23916  * configured to reference a certain URL.<br><br>
23917  * <p>
23918  * <em>Note that this class cannot be used to retrieve data from a domain other than the domain
23919  * from which the running page was served.<br><br>
23920  * <p>
23921  * For cross-domain access to remote data, use an {@link Roo.data.ScriptTagProxy}.</em><br><br>
23922  * <p>
23923  * Be aware that to enable the browser to parse an XML document, the server must set
23924  * the Content-Type header in the HTTP response to "text/xml".
23925  * @constructor
23926  * @param {Object} conn Connection config options to add to each request (e.g. {url: 'foo.php'} or
23927  * an {@link Roo.data.Connection} object.  If a Connection config is passed, the singleton {@link Roo.Ajax} object
23928  * will be used to make the request.
23929  */
23930 Roo.data.HttpProxy = function(conn){
23931     Roo.data.HttpProxy.superclass.constructor.call(this);
23932     // is conn a conn config or a real conn?
23933     this.conn = conn;
23934     this.useAjax = !conn || !conn.events;
23935   
23936 };
23937
23938 Roo.extend(Roo.data.HttpProxy, Roo.data.DataProxy, {
23939     // thse are take from connection...
23940     
23941     /**
23942      * @cfg {String} url (Optional) The default URL to be used for requests to the server. (defaults to undefined)
23943      */
23944     /**
23945      * @cfg {Object} extraParams (Optional) An object containing properties which are used as
23946      * extra parameters to each request made by this object. (defaults to undefined)
23947      */
23948     /**
23949      * @cfg {Object} defaultHeaders (Optional) An object containing request headers which are added
23950      *  to each request made by this object. (defaults to undefined)
23951      */
23952     /**
23953      * @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)
23954      */
23955     /**
23956      * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
23957      */
23958      /**
23959      * @cfg {Boolean} autoAbort (Optional) Whether this request should abort any pending requests. (defaults to false)
23960      * @type Boolean
23961      */
23962   
23963
23964     /**
23965      * @cfg {Boolean} disableCaching (Optional) True to add a unique cache-buster param to GET requests. (defaults to true)
23966      * @type Boolean
23967      */
23968     /**
23969      * Return the {@link Roo.data.Connection} object being used by this Proxy.
23970      * @return {Connection} The Connection object. This object may be used to subscribe to events on
23971      * a finer-grained basis than the DataProxy events.
23972      */
23973     getConnection : function(){
23974         return this.useAjax ? Roo.Ajax : this.conn;
23975     },
23976
23977     /**
23978      * Load data from the configured {@link Roo.data.Connection}, read the data object into
23979      * a block of Roo.data.Records using the passed {@link Roo.data.DataReader} implementation, and
23980      * process that block using the passed callback.
23981      * @param {Object} params An object containing properties which are to be used as HTTP parameters
23982      * for the request to the remote server.
23983      * @param {Roo.data.DataReader} reader The Reader object which converts the data
23984      * object into a block of Roo.data.Records.
23985      * @param {Function} callback The function into which to pass the block of Roo.data.Records.
23986      * The function must be passed <ul>
23987      * <li>The Record block object</li>
23988      * <li>The "arg" argument from the load function</li>
23989      * <li>A boolean success indicator</li>
23990      * </ul>
23991      * @param {Object} scope The scope in which to call the callback
23992      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
23993      */
23994     load : function(params, reader, callback, scope, arg){
23995         if(this.fireEvent("beforeload", this, params) !== false){
23996             var  o = {
23997                 params : params || {},
23998                 request: {
23999                     callback : callback,
24000                     scope : scope,
24001                     arg : arg
24002                 },
24003                 reader: reader,
24004                 callback : this.loadResponse,
24005                 scope: this
24006             };
24007             if(this.useAjax){
24008                 Roo.applyIf(o, this.conn);
24009                 if(this.activeRequest){
24010                     Roo.Ajax.abort(this.activeRequest);
24011                 }
24012                 this.activeRequest = Roo.Ajax.request(o);
24013             }else{
24014                 this.conn.request(o);
24015             }
24016         }else{
24017             callback.call(scope||this, null, arg, false);
24018         }
24019     },
24020
24021     // private
24022     loadResponse : function(o, success, response){
24023         delete this.activeRequest;
24024         if(!success){
24025             this.fireEvent("loadexception", this, o, response);
24026             o.request.callback.call(o.request.scope, null, o.request.arg, false);
24027             return;
24028         }
24029         var result;
24030         try {
24031             result = o.reader.read(response);
24032         }catch(e){
24033             this.fireEvent("loadexception", this, o, response, e);
24034             o.request.callback.call(o.request.scope, null, o.request.arg, false);
24035             return;
24036         }
24037         
24038         this.fireEvent("load", this, o, o.request.arg);
24039         o.request.callback.call(o.request.scope, result, o.request.arg, true);
24040     },
24041
24042     // private
24043     update : function(dataSet){
24044
24045     },
24046
24047     // private
24048     updateResponse : function(dataSet){
24049
24050     }
24051 });/*
24052  * Based on:
24053  * Ext JS Library 1.1.1
24054  * Copyright(c) 2006-2007, Ext JS, LLC.
24055  *
24056  * Originally Released Under LGPL - original licence link has changed is not relivant.
24057  *
24058  * Fork - LGPL
24059  * <script type="text/javascript">
24060  */
24061
24062 /**
24063  * @class Roo.data.ScriptTagProxy
24064  * An implementation of Roo.data.DataProxy that reads a data object from a URL which may be in a domain
24065  * other than the originating domain of the running page.<br><br>
24066  * <p>
24067  * <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
24068  * of the running page, you must use this class, rather than DataProxy.</em><br><br>
24069  * <p>
24070  * The content passed back from a server resource requested by a ScriptTagProxy is executable JavaScript
24071  * source code that is used as the source inside a &lt;script> tag.<br><br>
24072  * <p>
24073  * In order for the browser to process the returned data, the server must wrap the data object
24074  * with a call to a callback function, the name of which is passed as a parameter by the ScriptTagProxy.
24075  * Below is a Java example for a servlet which returns data for either a ScriptTagProxy, or an HttpProxy
24076  * depending on whether the callback name was passed:
24077  * <p>
24078  * <pre><code>
24079 boolean scriptTag = false;
24080 String cb = request.getParameter("callback");
24081 if (cb != null) {
24082     scriptTag = true;
24083     response.setContentType("text/javascript");
24084 } else {
24085     response.setContentType("application/x-json");
24086 }
24087 Writer out = response.getWriter();
24088 if (scriptTag) {
24089     out.write(cb + "(");
24090 }
24091 out.print(dataBlock.toJsonString());
24092 if (scriptTag) {
24093     out.write(");");
24094 }
24095 </pre></code>
24096  *
24097  * @constructor
24098  * @param {Object} config A configuration object.
24099  */
24100 Roo.data.ScriptTagProxy = function(config){
24101     Roo.data.ScriptTagProxy.superclass.constructor.call(this);
24102     Roo.apply(this, config);
24103     this.head = document.getElementsByTagName("head")[0];
24104 };
24105
24106 Roo.data.ScriptTagProxy.TRANS_ID = 1000;
24107
24108 Roo.extend(Roo.data.ScriptTagProxy, Roo.data.DataProxy, {
24109     /**
24110      * @cfg {String} url The URL from which to request the data object.
24111      */
24112     /**
24113      * @cfg {Number} timeout (Optional) The number of milliseconds to wait for a response. Defaults to 30 seconds.
24114      */
24115     timeout : 30000,
24116     /**
24117      * @cfg {String} callbackParam (Optional) The name of the parameter to pass to the server which tells
24118      * the server the name of the callback function set up by the load call to process the returned data object.
24119      * Defaults to "callback".<p>The server-side processing must read this parameter value, and generate
24120      * javascript output which calls this named function passing the data object as its only parameter.
24121      */
24122     callbackParam : "callback",
24123     /**
24124      *  @cfg {Boolean} nocache (Optional) Defaults to true. Disable cacheing by adding a unique parameter
24125      * name to the request.
24126      */
24127     nocache : true,
24128
24129     /**
24130      * Load data from the configured URL, read the data object into
24131      * a block of Roo.data.Records using the passed Roo.data.DataReader implementation, and
24132      * process that block using the passed callback.
24133      * @param {Object} params An object containing properties which are to be used as HTTP parameters
24134      * for the request to the remote server.
24135      * @param {Roo.data.DataReader} reader The Reader object which converts the data
24136      * object into a block of Roo.data.Records.
24137      * @param {Function} callback The function into which to pass the block of Roo.data.Records.
24138      * The function must be passed <ul>
24139      * <li>The Record block object</li>
24140      * <li>The "arg" argument from the load function</li>
24141      * <li>A boolean success indicator</li>
24142      * </ul>
24143      * @param {Object} scope The scope in which to call the callback
24144      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
24145      */
24146     load : function(params, reader, callback, scope, arg){
24147         if(this.fireEvent("beforeload", this, params) !== false){
24148
24149             var p = Roo.urlEncode(Roo.apply(params, this.extraParams));
24150
24151             var url = this.url;
24152             url += (url.indexOf("?") != -1 ? "&" : "?") + p;
24153             if(this.nocache){
24154                 url += "&_dc=" + (new Date().getTime());
24155             }
24156             var transId = ++Roo.data.ScriptTagProxy.TRANS_ID;
24157             var trans = {
24158                 id : transId,
24159                 cb : "stcCallback"+transId,
24160                 scriptId : "stcScript"+transId,
24161                 params : params,
24162                 arg : arg,
24163                 url : url,
24164                 callback : callback,
24165                 scope : scope,
24166                 reader : reader
24167             };
24168             var conn = this;
24169
24170             window[trans.cb] = function(o){
24171                 conn.handleResponse(o, trans);
24172             };
24173
24174             url += String.format("&{0}={1}", this.callbackParam, trans.cb);
24175
24176             if(this.autoAbort !== false){
24177                 this.abort();
24178             }
24179
24180             trans.timeoutId = this.handleFailure.defer(this.timeout, this, [trans]);
24181
24182             var script = document.createElement("script");
24183             script.setAttribute("src", url);
24184             script.setAttribute("type", "text/javascript");
24185             script.setAttribute("id", trans.scriptId);
24186             this.head.appendChild(script);
24187
24188             this.trans = trans;
24189         }else{
24190             callback.call(scope||this, null, arg, false);
24191         }
24192     },
24193
24194     // private
24195     isLoading : function(){
24196         return this.trans ? true : false;
24197     },
24198
24199     /**
24200      * Abort the current server request.
24201      */
24202     abort : function(){
24203         if(this.isLoading()){
24204             this.destroyTrans(this.trans);
24205         }
24206     },
24207
24208     // private
24209     destroyTrans : function(trans, isLoaded){
24210         this.head.removeChild(document.getElementById(trans.scriptId));
24211         clearTimeout(trans.timeoutId);
24212         if(isLoaded){
24213             window[trans.cb] = undefined;
24214             try{
24215                 delete window[trans.cb];
24216             }catch(e){}
24217         }else{
24218             // if hasn't been loaded, wait for load to remove it to prevent script error
24219             window[trans.cb] = function(){
24220                 window[trans.cb] = undefined;
24221                 try{
24222                     delete window[trans.cb];
24223                 }catch(e){}
24224             };
24225         }
24226     },
24227
24228     // private
24229     handleResponse : function(o, trans){
24230         this.trans = false;
24231         this.destroyTrans(trans, true);
24232         var result;
24233         try {
24234             result = trans.reader.readRecords(o);
24235         }catch(e){
24236             this.fireEvent("loadexception", this, o, trans.arg, e);
24237             trans.callback.call(trans.scope||window, null, trans.arg, false);
24238             return;
24239         }
24240         this.fireEvent("load", this, o, trans.arg);
24241         trans.callback.call(trans.scope||window, result, trans.arg, true);
24242     },
24243
24244     // private
24245     handleFailure : function(trans){
24246         this.trans = false;
24247         this.destroyTrans(trans, false);
24248         this.fireEvent("loadexception", this, null, trans.arg);
24249         trans.callback.call(trans.scope||window, null, trans.arg, false);
24250     }
24251 });/*
24252  * Based on:
24253  * Ext JS Library 1.1.1
24254  * Copyright(c) 2006-2007, Ext JS, LLC.
24255  *
24256  * Originally Released Under LGPL - original licence link has changed is not relivant.
24257  *
24258  * Fork - LGPL
24259  * <script type="text/javascript">
24260  */
24261
24262 /**
24263  * @class Roo.data.JsonReader
24264  * @extends Roo.data.DataReader
24265  * Data reader class to create an Array of Roo.data.Record objects from a JSON response
24266  * based on mappings in a provided Roo.data.Record constructor.
24267  * 
24268  * The default behaviour of a store is to send ?_requestMeta=1, unless the class has recieved 'metaData' property
24269  * in the reply previously. 
24270  * 
24271  * <p>
24272  * Example code:
24273  * <pre><code>
24274 var RecordDef = Roo.data.Record.create([
24275     {name: 'name', mapping: 'name'},     // "mapping" property not needed if it's the same as "name"
24276     {name: 'occupation'}                 // This field will use "occupation" as the mapping.
24277 ]);
24278 var myReader = new Roo.data.JsonReader({
24279     totalProperty: "results",    // The property which contains the total dataset size (optional)
24280     root: "rows",                // The property which contains an Array of row objects
24281     id: "id"                     // The property within each row object that provides an ID for the record (optional)
24282 }, RecordDef);
24283 </code></pre>
24284  * <p>
24285  * This would consume a JSON file like this:
24286  * <pre><code>
24287 { 'results': 2, 'rows': [
24288     { 'id': 1, 'name': 'Bill', occupation: 'Gardener' },
24289     { 'id': 2, 'name': 'Ben', occupation: 'Horticulturalist' } ]
24290 }
24291 </code></pre>
24292  * @cfg {String} totalProperty Name of the property from which to retrieve the total number of records
24293  * in the dataset. This is only needed if the whole dataset is not passed in one go, but is being
24294  * paged from the remote server.
24295  * @cfg {String} successProperty Name of the property from which to retrieve the success attribute used by forms.
24296  * @cfg {String} root name of the property which contains the Array of row objects.
24297  * @cfg {String} id Name of the property within a row object that contains a record identifier value.
24298  * @cfg {Array} fields Array of field definition objects
24299  * @constructor
24300  * Create a new JsonReader
24301  * @param {Object} meta Metadata configuration options
24302  * @param {Object} recordType Either an Array of field definition objects,
24303  * or an {@link Roo.data.Record} object created using {@link Roo.data.Record#create}.
24304  */
24305 Roo.data.JsonReader = function(meta, recordType){
24306     
24307     meta = meta || {};
24308     // set some defaults:
24309     Roo.applyIf(meta, {
24310         totalProperty: 'total',
24311         successProperty : 'success',
24312         root : 'data',
24313         id : 'id'
24314     });
24315     
24316     Roo.data.JsonReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24317 };
24318 Roo.extend(Roo.data.JsonReader, Roo.data.DataReader, {
24319     
24320     /**
24321      * @prop {Boolean} metaFromRemote  - if the meta data was loaded from the remote source.
24322      * Used by Store query builder to append _requestMeta to params.
24323      * 
24324      */
24325     metaFromRemote : false,
24326     /**
24327      * This method is only used by a DataProxy which has retrieved data from a remote server.
24328      * @param {Object} response The XHR object which contains the JSON data in its responseText.
24329      * @return {Object} data A data block which is used by an Roo.data.Store object as
24330      * a cache of Roo.data.Records.
24331      */
24332     read : function(response){
24333         var json = response.responseText;
24334        
24335         var o = /* eval:var:o */ eval("("+json+")");
24336         if(!o) {
24337             throw {message: "JsonReader.read: Json object not found"};
24338         }
24339         
24340         if(o.metaData){
24341             
24342             delete this.ef;
24343             this.metaFromRemote = true;
24344             this.meta = o.metaData;
24345             this.recordType = Roo.data.Record.create(o.metaData.fields);
24346             this.onMetaChange(this.meta, this.recordType, o);
24347         }
24348         return this.readRecords(o);
24349     },
24350
24351     // private function a store will implement
24352     onMetaChange : function(meta, recordType, o){
24353
24354     },
24355
24356     /**
24357          * @ignore
24358          */
24359     simpleAccess: function(obj, subsc) {
24360         return obj[subsc];
24361     },
24362
24363         /**
24364          * @ignore
24365          */
24366     getJsonAccessor: function(){
24367         var re = /[\[\.]/;
24368         return function(expr) {
24369             try {
24370                 return(re.test(expr))
24371                     ? new Function("obj", "return obj." + expr)
24372                     : function(obj){
24373                         return obj[expr];
24374                     };
24375             } catch(e){}
24376             return Roo.emptyFn;
24377         };
24378     }(),
24379
24380     /**
24381      * Create a data block containing Roo.data.Records from an XML document.
24382      * @param {Object} o An object which contains an Array of row objects in the property specified
24383      * in the config as 'root, and optionally a property, specified in the config as 'totalProperty'
24384      * which contains the total size of the dataset.
24385      * @return {Object} data A data block which is used by an Roo.data.Store object as
24386      * a cache of Roo.data.Records.
24387      */
24388     readRecords : function(o){
24389         /**
24390          * After any data loads, the raw JSON data is available for further custom processing.
24391          * @type Object
24392          */
24393         this.o = o;
24394         var s = this.meta, Record = this.recordType,
24395             f = Record ? Record.prototype.fields : null, fi = f ? f.items : [], fl = f ? f.length : 0;
24396
24397 //      Generate extraction functions for the totalProperty, the root, the id, and for each field
24398         if (!this.ef) {
24399             if(s.totalProperty) {
24400                     this.getTotal = this.getJsonAccessor(s.totalProperty);
24401                 }
24402                 if(s.successProperty) {
24403                     this.getSuccess = this.getJsonAccessor(s.successProperty);
24404                 }
24405                 this.getRoot = s.root ? this.getJsonAccessor(s.root) : function(p){return p;};
24406                 if (s.id) {
24407                         var g = this.getJsonAccessor(s.id);
24408                         this.getId = function(rec) {
24409                                 var r = g(rec);  
24410                                 return (r === undefined || r === "") ? null : r;
24411                         };
24412                 } else {
24413                         this.getId = function(){return null;};
24414                 }
24415             this.ef = [];
24416             for(var jj = 0; jj < fl; jj++){
24417                 f = fi[jj];
24418                 var map = (f.mapping !== undefined && f.mapping !== null) ? f.mapping : f.name;
24419                 this.ef[jj] = this.getJsonAccessor(map);
24420             }
24421         }
24422
24423         var root = this.getRoot(o), c = root.length, totalRecords = c, success = true;
24424         if(s.totalProperty){
24425             var vt = parseInt(this.getTotal(o), 10);
24426             if(!isNaN(vt)){
24427                 totalRecords = vt;
24428             }
24429         }
24430         if(s.successProperty){
24431             var vs = this.getSuccess(o);
24432             if(vs === false || vs === 'false'){
24433                 success = false;
24434             }
24435         }
24436         var records = [];
24437         for(var i = 0; i < c; i++){
24438                 var n = root[i];
24439             var values = {};
24440             var id = this.getId(n);
24441             for(var j = 0; j < fl; j++){
24442                 f = fi[j];
24443             var v = this.ef[j](n);
24444             if (!f.convert) {
24445                 Roo.log('missing convert for ' + f.name);
24446                 Roo.log(f);
24447                 continue;
24448             }
24449             values[f.name] = f.convert((v !== undefined) ? v : f.defaultValue);
24450             }
24451             var record = new Record(values, id);
24452             record.json = n;
24453             records[i] = record;
24454         }
24455         return {
24456             raw : o,
24457             success : success,
24458             records : records,
24459             totalRecords : totalRecords
24460         };
24461     }
24462 });/*
24463  * Based on:
24464  * Ext JS Library 1.1.1
24465  * Copyright(c) 2006-2007, Ext JS, LLC.
24466  *
24467  * Originally Released Under LGPL - original licence link has changed is not relivant.
24468  *
24469  * Fork - LGPL
24470  * <script type="text/javascript">
24471  */
24472
24473 /**
24474  * @class Roo.data.XmlReader
24475  * @extends Roo.data.DataReader
24476  * Data reader class to create an Array of {@link Roo.data.Record} objects from an XML document
24477  * based on mappings in a provided Roo.data.Record constructor.<br><br>
24478  * <p>
24479  * <em>Note that in order for the browser to parse a returned XML document, the Content-Type
24480  * header in the HTTP response must be set to "text/xml".</em>
24481  * <p>
24482  * Example code:
24483  * <pre><code>
24484 var RecordDef = Roo.data.Record.create([
24485    {name: 'name', mapping: 'name'},     // "mapping" property not needed if it's the same as "name"
24486    {name: 'occupation'}                 // This field will use "occupation" as the mapping.
24487 ]);
24488 var myReader = new Roo.data.XmlReader({
24489    totalRecords: "results", // The element which contains the total dataset size (optional)
24490    record: "row",           // The repeated element which contains row information
24491    id: "id"                 // The element within the row that provides an ID for the record (optional)
24492 }, RecordDef);
24493 </code></pre>
24494  * <p>
24495  * This would consume an XML file like this:
24496  * <pre><code>
24497 &lt;?xml?>
24498 &lt;dataset>
24499  &lt;results>2&lt;/results>
24500  &lt;row>
24501    &lt;id>1&lt;/id>
24502    &lt;name>Bill&lt;/name>
24503    &lt;occupation>Gardener&lt;/occupation>
24504  &lt;/row>
24505  &lt;row>
24506    &lt;id>2&lt;/id>
24507    &lt;name>Ben&lt;/name>
24508    &lt;occupation>Horticulturalist&lt;/occupation>
24509  &lt;/row>
24510 &lt;/dataset>
24511 </code></pre>
24512  * @cfg {String} totalRecords The DomQuery path from which to retrieve the total number of records
24513  * in the dataset. This is only needed if the whole dataset is not passed in one go, but is being
24514  * paged from the remote server.
24515  * @cfg {String} record The DomQuery path to the repeated element which contains record information.
24516  * @cfg {String} success The DomQuery path to the success attribute used by forms.
24517  * @cfg {String} id The DomQuery path relative from the record element to the element that contains
24518  * a record identifier value.
24519  * @constructor
24520  * Create a new XmlReader
24521  * @param {Object} meta Metadata configuration options
24522  * @param {Mixed} recordType The definition of the data record type to produce.  This can be either a valid
24523  * Record subclass created with {@link Roo.data.Record#create}, or an array of objects with which to call
24524  * Roo.data.Record.create.  See the {@link Roo.data.Record} class for more details.
24525  */
24526 Roo.data.XmlReader = function(meta, recordType){
24527     meta = meta || {};
24528     Roo.data.XmlReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24529 };
24530 Roo.extend(Roo.data.XmlReader, Roo.data.DataReader, {
24531     /**
24532      * This method is only used by a DataProxy which has retrieved data from a remote server.
24533          * @param {Object} response The XHR object which contains the parsed XML document.  The response is expected
24534          * to contain a method called 'responseXML' that returns an XML document object.
24535      * @return {Object} records A data block which is used by an {@link Roo.data.Store} as
24536      * a cache of Roo.data.Records.
24537      */
24538     read : function(response){
24539         var doc = response.responseXML;
24540         if(!doc) {
24541             throw {message: "XmlReader.read: XML Document not available"};
24542         }
24543         return this.readRecords(doc);
24544     },
24545
24546     /**
24547      * Create a data block containing Roo.data.Records from an XML document.
24548          * @param {Object} doc A parsed XML document.
24549      * @return {Object} records A data block which is used by an {@link Roo.data.Store} as
24550      * a cache of Roo.data.Records.
24551      */
24552     readRecords : function(doc){
24553         /**
24554          * After any data loads/reads, the raw XML Document is available for further custom processing.
24555          * @type XMLDocument
24556          */
24557         this.xmlData = doc;
24558         var root = doc.documentElement || doc;
24559         var q = Roo.DomQuery;
24560         var recordType = this.recordType, fields = recordType.prototype.fields;
24561         var sid = this.meta.id;
24562         var totalRecords = 0, success = true;
24563         if(this.meta.totalRecords){
24564             totalRecords = q.selectNumber(this.meta.totalRecords, root, 0);
24565         }
24566         
24567         if(this.meta.success){
24568             var sv = q.selectValue(this.meta.success, root, true);
24569             success = sv !== false && sv !== 'false';
24570         }
24571         var records = [];
24572         var ns = q.select(this.meta.record, root);
24573         for(var i = 0, len = ns.length; i < len; i++) {
24574                 var n = ns[i];
24575                 var values = {};
24576                 var id = sid ? q.selectValue(sid, n) : undefined;
24577                 for(var j = 0, jlen = fields.length; j < jlen; j++){
24578                     var f = fields.items[j];
24579                 var v = q.selectValue(f.mapping || f.name, n, f.defaultValue);
24580                     v = f.convert(v);
24581                     values[f.name] = v;
24582                 }
24583                 var record = new recordType(values, id);
24584                 record.node = n;
24585                 records[records.length] = record;
24586             }
24587
24588             return {
24589                 success : success,
24590                 records : records,
24591                 totalRecords : totalRecords || records.length
24592             };
24593     }
24594 });/*
24595  * Based on:
24596  * Ext JS Library 1.1.1
24597  * Copyright(c) 2006-2007, Ext JS, LLC.
24598  *
24599  * Originally Released Under LGPL - original licence link has changed is not relivant.
24600  *
24601  * Fork - LGPL
24602  * <script type="text/javascript">
24603  */
24604
24605 /**
24606  * @class Roo.data.ArrayReader
24607  * @extends Roo.data.DataReader
24608  * Data reader class to create an Array of Roo.data.Record objects from an Array.
24609  * Each element of that Array represents a row of data fields. The
24610  * fields are pulled into a Record object using as a subscript, the <em>mapping</em> property
24611  * of the field definition if it exists, or the field's ordinal position in the definition.<br>
24612  * <p>
24613  * Example code:.
24614  * <pre><code>
24615 var RecordDef = Roo.data.Record.create([
24616     {name: 'name', mapping: 1},         // "mapping" only needed if an "id" field is present which
24617     {name: 'occupation', mapping: 2}    // precludes using the ordinal position as the index.
24618 ]);
24619 var myReader = new Roo.data.ArrayReader({
24620     id: 0                     // The subscript within row Array that provides an ID for the Record (optional)
24621 }, RecordDef);
24622 </code></pre>
24623  * <p>
24624  * This would consume an Array like this:
24625  * <pre><code>
24626 [ [1, 'Bill', 'Gardener'], [2, 'Ben', 'Horticulturalist'] ]
24627   </code></pre>
24628  
24629  * @constructor
24630  * Create a new JsonReader
24631  * @param {Object} meta Metadata configuration options.
24632  * @param {Object|Array} recordType Either an Array of field definition objects
24633  * 
24634  * @cfg {Array} fields Array of field definition objects
24635  * @cfg {String} id Name of the property within a row object that contains a record identifier value.
24636  * as specified to {@link Roo.data.Record#create},
24637  * or an {@link Roo.data.Record} object
24638  *
24639  * 
24640  * created using {@link Roo.data.Record#create}.
24641  */
24642 Roo.data.ArrayReader = function(meta, recordType)
24643 {    
24644     Roo.data.ArrayReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24645 };
24646
24647 Roo.extend(Roo.data.ArrayReader, Roo.data.JsonReader, {
24648     /**
24649      * Create a data block containing Roo.data.Records from an XML document.
24650      * @param {Object} o An Array of row objects which represents the dataset.
24651      * @return {Object} A data block which is used by an {@link Roo.data.Store} object as
24652      * a cache of Roo.data.Records.
24653      */
24654     readRecords : function(o)
24655     {
24656         var sid = this.meta ? this.meta.id : null;
24657         var recordType = this.recordType, fields = recordType.prototype.fields;
24658         var records = [];
24659         var root = o;
24660         for(var i = 0; i < root.length; i++){
24661                 var n = root[i];
24662             var values = {};
24663             var id = ((sid || sid === 0) && n[sid] !== undefined && n[sid] !== "" ? n[sid] : null);
24664             for(var j = 0, jlen = fields.length; j < jlen; j++){
24665                 var f = fields.items[j];
24666                 var k = f.mapping !== undefined && f.mapping !== null ? f.mapping : j;
24667                 var v = n[k] !== undefined ? n[k] : f.defaultValue;
24668                 v = f.convert(v);
24669                 values[f.name] = v;
24670             }
24671             var record = new recordType(values, id);
24672             record.json = n;
24673             records[records.length] = record;
24674         }
24675         return {
24676             records : records,
24677             totalRecords : records.length
24678         };
24679     }
24680 });/*
24681  * Based on:
24682  * Ext JS Library 1.1.1
24683  * Copyright(c) 2006-2007, Ext JS, LLC.
24684  *
24685  * Originally Released Under LGPL - original licence link has changed is not relivant.
24686  *
24687  * Fork - LGPL
24688  * <script type="text/javascript">
24689  */
24690
24691
24692 /**
24693  * @class Roo.data.Tree
24694  * @extends Roo.util.Observable
24695  * Represents a tree data structure and bubbles all the events for its nodes. The nodes
24696  * in the tree have most standard DOM functionality.
24697  * @constructor
24698  * @param {Node} root (optional) The root node
24699  */
24700 Roo.data.Tree = function(root){
24701    this.nodeHash = {};
24702    /**
24703     * The root node for this tree
24704     * @type Node
24705     */
24706    this.root = null;
24707    if(root){
24708        this.setRootNode(root);
24709    }
24710    this.addEvents({
24711        /**
24712         * @event append
24713         * Fires when a new child node is appended to a node in this tree.
24714         * @param {Tree} tree The owner tree
24715         * @param {Node} parent The parent node
24716         * @param {Node} node The newly appended node
24717         * @param {Number} index The index of the newly appended node
24718         */
24719        "append" : true,
24720        /**
24721         * @event remove
24722         * Fires when a child node is removed from a node in this tree.
24723         * @param {Tree} tree The owner tree
24724         * @param {Node} parent The parent node
24725         * @param {Node} node The child node removed
24726         */
24727        "remove" : true,
24728        /**
24729         * @event move
24730         * Fires when a node is moved to a new location in the tree
24731         * @param {Tree} tree The owner tree
24732         * @param {Node} node The node moved
24733         * @param {Node} oldParent The old parent of this node
24734         * @param {Node} newParent The new parent of this node
24735         * @param {Number} index The index it was moved to
24736         */
24737        "move" : true,
24738        /**
24739         * @event insert
24740         * Fires when a new child node is inserted in a node in this tree.
24741         * @param {Tree} tree The owner tree
24742         * @param {Node} parent The parent node
24743         * @param {Node} node The child node inserted
24744         * @param {Node} refNode The child node the node was inserted before
24745         */
24746        "insert" : true,
24747        /**
24748         * @event beforeappend
24749         * Fires before a new child is appended to a node in this tree, return false to cancel the append.
24750         * @param {Tree} tree The owner tree
24751         * @param {Node} parent The parent node
24752         * @param {Node} node The child node to be appended
24753         */
24754        "beforeappend" : true,
24755        /**
24756         * @event beforeremove
24757         * Fires before a child is removed from a node in this tree, return false to cancel the remove.
24758         * @param {Tree} tree The owner tree
24759         * @param {Node} parent The parent node
24760         * @param {Node} node The child node to be removed
24761         */
24762        "beforeremove" : true,
24763        /**
24764         * @event beforemove
24765         * Fires before a node is moved to a new location in the tree. Return false to cancel the move.
24766         * @param {Tree} tree The owner tree
24767         * @param {Node} node The node being moved
24768         * @param {Node} oldParent The parent of the node
24769         * @param {Node} newParent The new parent the node is moving to
24770         * @param {Number} index The index it is being moved to
24771         */
24772        "beforemove" : true,
24773        /**
24774         * @event beforeinsert
24775         * Fires before a new child is inserted in a node in this tree, return false to cancel the insert.
24776         * @param {Tree} tree The owner tree
24777         * @param {Node} parent The parent node
24778         * @param {Node} node The child node to be inserted
24779         * @param {Node} refNode The child node the node is being inserted before
24780         */
24781        "beforeinsert" : true
24782    });
24783
24784     Roo.data.Tree.superclass.constructor.call(this);
24785 };
24786
24787 Roo.extend(Roo.data.Tree, Roo.util.Observable, {
24788     pathSeparator: "/",
24789
24790     proxyNodeEvent : function(){
24791         return this.fireEvent.apply(this, arguments);
24792     },
24793
24794     /**
24795      * Returns the root node for this tree.
24796      * @return {Node}
24797      */
24798     getRootNode : function(){
24799         return this.root;
24800     },
24801
24802     /**
24803      * Sets the root node for this tree.
24804      * @param {Node} node
24805      * @return {Node}
24806      */
24807     setRootNode : function(node){
24808         this.root = node;
24809         node.ownerTree = this;
24810         node.isRoot = true;
24811         this.registerNode(node);
24812         return node;
24813     },
24814
24815     /**
24816      * Gets a node in this tree by its id.
24817      * @param {String} id
24818      * @return {Node}
24819      */
24820     getNodeById : function(id){
24821         return this.nodeHash[id];
24822     },
24823
24824     registerNode : function(node){
24825         this.nodeHash[node.id] = node;
24826     },
24827
24828     unregisterNode : function(node){
24829         delete this.nodeHash[node.id];
24830     },
24831
24832     toString : function(){
24833         return "[Tree"+(this.id?" "+this.id:"")+"]";
24834     }
24835 });
24836
24837 /**
24838  * @class Roo.data.Node
24839  * @extends Roo.util.Observable
24840  * @cfg {Boolean} leaf true if this node is a leaf and does not have children
24841  * @cfg {String} id The id for this node. If one is not specified, one is generated.
24842  * @constructor
24843  * @param {Object} attributes The attributes/config for the node
24844  */
24845 Roo.data.Node = function(attributes){
24846     /**
24847      * The attributes supplied for the node. You can use this property to access any custom attributes you supplied.
24848      * @type {Object}
24849      */
24850     this.attributes = attributes || {};
24851     this.leaf = this.attributes.leaf;
24852     /**
24853      * The node id. @type String
24854      */
24855     this.id = this.attributes.id;
24856     if(!this.id){
24857         this.id = Roo.id(null, "ynode-");
24858         this.attributes.id = this.id;
24859     }
24860      
24861     
24862     /**
24863      * All child nodes of this node. @type Array
24864      */
24865     this.childNodes = [];
24866     if(!this.childNodes.indexOf){ // indexOf is a must
24867         this.childNodes.indexOf = function(o){
24868             for(var i = 0, len = this.length; i < len; i++){
24869                 if(this[i] == o) {
24870                     return i;
24871                 }
24872             }
24873             return -1;
24874         };
24875     }
24876     /**
24877      * The parent node for this node. @type Node
24878      */
24879     this.parentNode = null;
24880     /**
24881      * The first direct child node of this node, or null if this node has no child nodes. @type Node
24882      */
24883     this.firstChild = null;
24884     /**
24885      * The last direct child node of this node, or null if this node has no child nodes. @type Node
24886      */
24887     this.lastChild = null;
24888     /**
24889      * The node immediately preceding this node in the tree, or null if there is no sibling node. @type Node
24890      */
24891     this.previousSibling = null;
24892     /**
24893      * The node immediately following this node in the tree, or null if there is no sibling node. @type Node
24894      */
24895     this.nextSibling = null;
24896
24897     this.addEvents({
24898        /**
24899         * @event append
24900         * Fires when a new child node is appended
24901         * @param {Tree} tree The owner tree
24902         * @param {Node} this This node
24903         * @param {Node} node The newly appended node
24904         * @param {Number} index The index of the newly appended node
24905         */
24906        "append" : true,
24907        /**
24908         * @event remove
24909         * Fires when a child node is removed
24910         * @param {Tree} tree The owner tree
24911         * @param {Node} this This node
24912         * @param {Node} node The removed node
24913         */
24914        "remove" : true,
24915        /**
24916         * @event move
24917         * Fires when this node is moved to a new location in the tree
24918         * @param {Tree} tree The owner tree
24919         * @param {Node} this This node
24920         * @param {Node} oldParent The old parent of this node
24921         * @param {Node} newParent The new parent of this node
24922         * @param {Number} index The index it was moved to
24923         */
24924        "move" : true,
24925        /**
24926         * @event insert
24927         * Fires when a new child node is inserted.
24928         * @param {Tree} tree The owner tree
24929         * @param {Node} this This node
24930         * @param {Node} node The child node inserted
24931         * @param {Node} refNode The child node the node was inserted before
24932         */
24933        "insert" : true,
24934        /**
24935         * @event beforeappend
24936         * Fires before a new child is appended, return false to cancel the append.
24937         * @param {Tree} tree The owner tree
24938         * @param {Node} this This node
24939         * @param {Node} node The child node to be appended
24940         */
24941        "beforeappend" : true,
24942        /**
24943         * @event beforeremove
24944         * Fires before a child is removed, return false to cancel the remove.
24945         * @param {Tree} tree The owner tree
24946         * @param {Node} this This node
24947         * @param {Node} node The child node to be removed
24948         */
24949        "beforeremove" : true,
24950        /**
24951         * @event beforemove
24952         * Fires before this node is moved to a new location in the tree. Return false to cancel the move.
24953         * @param {Tree} tree The owner tree
24954         * @param {Node} this This node
24955         * @param {Node} oldParent The parent of this node
24956         * @param {Node} newParent The new parent this node is moving to
24957         * @param {Number} index The index it is being moved to
24958         */
24959        "beforemove" : true,
24960        /**
24961         * @event beforeinsert
24962         * Fires before a new child is inserted, return false to cancel the insert.
24963         * @param {Tree} tree The owner tree
24964         * @param {Node} this This node
24965         * @param {Node} node The child node to be inserted
24966         * @param {Node} refNode The child node the node is being inserted before
24967         */
24968        "beforeinsert" : true
24969    });
24970     this.listeners = this.attributes.listeners;
24971     Roo.data.Node.superclass.constructor.call(this);
24972 };
24973
24974 Roo.extend(Roo.data.Node, Roo.util.Observable, {
24975     fireEvent : function(evtName){
24976         // first do standard event for this node
24977         if(Roo.data.Node.superclass.fireEvent.apply(this, arguments) === false){
24978             return false;
24979         }
24980         // then bubble it up to the tree if the event wasn't cancelled
24981         var ot = this.getOwnerTree();
24982         if(ot){
24983             if(ot.proxyNodeEvent.apply(ot, arguments) === false){
24984                 return false;
24985             }
24986         }
24987         return true;
24988     },
24989
24990     /**
24991      * Returns true if this node is a leaf
24992      * @return {Boolean}
24993      */
24994     isLeaf : function(){
24995         return this.leaf === true;
24996     },
24997
24998     // private
24999     setFirstChild : function(node){
25000         this.firstChild = node;
25001     },
25002
25003     //private
25004     setLastChild : function(node){
25005         this.lastChild = node;
25006     },
25007
25008
25009     /**
25010      * Returns true if this node is the last child of its parent
25011      * @return {Boolean}
25012      */
25013     isLast : function(){
25014        return (!this.parentNode ? true : this.parentNode.lastChild == this);
25015     },
25016
25017     /**
25018      * Returns true if this node is the first child of its parent
25019      * @return {Boolean}
25020      */
25021     isFirst : function(){
25022        return (!this.parentNode ? true : this.parentNode.firstChild == this);
25023     },
25024
25025     hasChildNodes : function(){
25026         return !this.isLeaf() && this.childNodes.length > 0;
25027     },
25028
25029     /**
25030      * Insert node(s) as the last child node of this node.
25031      * @param {Node/Array} node The node or Array of nodes to append
25032      * @return {Node} The appended node if single append, or null if an array was passed
25033      */
25034     appendChild : function(node){
25035         var multi = false;
25036         if(node instanceof Array){
25037             multi = node;
25038         }else if(arguments.length > 1){
25039             multi = arguments;
25040         }
25041         
25042         // if passed an array or multiple args do them one by one
25043         if(multi){
25044             for(var i = 0, len = multi.length; i < len; i++) {
25045                 this.appendChild(multi[i]);
25046             }
25047         }else{
25048             if(this.fireEvent("beforeappend", this.ownerTree, this, node) === false){
25049                 return false;
25050             }
25051             var index = this.childNodes.length;
25052             var oldParent = node.parentNode;
25053             // it's a move, make sure we move it cleanly
25054             if(oldParent){
25055                 if(node.fireEvent("beforemove", node.getOwnerTree(), node, oldParent, this, index) === false){
25056                     return false;
25057                 }
25058                 oldParent.removeChild(node);
25059             }
25060             
25061             index = this.childNodes.length;
25062             if(index == 0){
25063                 this.setFirstChild(node);
25064             }
25065             this.childNodes.push(node);
25066             node.parentNode = this;
25067             var ps = this.childNodes[index-1];
25068             if(ps){
25069                 node.previousSibling = ps;
25070                 ps.nextSibling = node;
25071             }else{
25072                 node.previousSibling = null;
25073             }
25074             node.nextSibling = null;
25075             this.setLastChild(node);
25076             node.setOwnerTree(this.getOwnerTree());
25077             this.fireEvent("append", this.ownerTree, this, node, index);
25078             if(this.ownerTree) {
25079                 this.ownerTree.fireEvent("appendnode", this, node, index);
25080             }
25081             if(oldParent){
25082                 node.fireEvent("move", this.ownerTree, node, oldParent, this, index);
25083             }
25084             return node;
25085         }
25086     },
25087
25088     /**
25089      * Removes a child node from this node.
25090      * @param {Node} node The node to remove
25091      * @return {Node} The removed node
25092      */
25093     removeChild : function(node){
25094         var index = this.childNodes.indexOf(node);
25095         if(index == -1){
25096             return false;
25097         }
25098         if(this.fireEvent("beforeremove", this.ownerTree, this, node) === false){
25099             return false;
25100         }
25101
25102         // remove it from childNodes collection
25103         this.childNodes.splice(index, 1);
25104
25105         // update siblings
25106         if(node.previousSibling){
25107             node.previousSibling.nextSibling = node.nextSibling;
25108         }
25109         if(node.nextSibling){
25110             node.nextSibling.previousSibling = node.previousSibling;
25111         }
25112
25113         // update child refs
25114         if(this.firstChild == node){
25115             this.setFirstChild(node.nextSibling);
25116         }
25117         if(this.lastChild == node){
25118             this.setLastChild(node.previousSibling);
25119         }
25120
25121         node.setOwnerTree(null);
25122         // clear any references from the node
25123         node.parentNode = null;
25124         node.previousSibling = null;
25125         node.nextSibling = null;
25126         this.fireEvent("remove", this.ownerTree, this, node);
25127         return node;
25128     },
25129
25130     /**
25131      * Inserts the first node before the second node in this nodes childNodes collection.
25132      * @param {Node} node The node to insert
25133      * @param {Node} refNode The node to insert before (if null the node is appended)
25134      * @return {Node} The inserted node
25135      */
25136     insertBefore : function(node, refNode){
25137         if(!refNode){ // like standard Dom, refNode can be null for append
25138             return this.appendChild(node);
25139         }
25140         // nothing to do
25141         if(node == refNode){
25142             return false;
25143         }
25144
25145         if(this.fireEvent("beforeinsert", this.ownerTree, this, node, refNode) === false){
25146             return false;
25147         }
25148         var index = this.childNodes.indexOf(refNode);
25149         var oldParent = node.parentNode;
25150         var refIndex = index;
25151
25152         // when moving internally, indexes will change after remove
25153         if(oldParent == this && this.childNodes.indexOf(node) < index){
25154             refIndex--;
25155         }
25156
25157         // it's a move, make sure we move it cleanly
25158         if(oldParent){
25159             if(node.fireEvent("beforemove", node.getOwnerTree(), node, oldParent, this, index, refNode) === false){
25160                 return false;
25161             }
25162             oldParent.removeChild(node);
25163         }
25164         if(refIndex == 0){
25165             this.setFirstChild(node);
25166         }
25167         this.childNodes.splice(refIndex, 0, node);
25168         node.parentNode = this;
25169         var ps = this.childNodes[refIndex-1];
25170         if(ps){
25171             node.previousSibling = ps;
25172             ps.nextSibling = node;
25173         }else{
25174             node.previousSibling = null;
25175         }
25176         node.nextSibling = refNode;
25177         refNode.previousSibling = node;
25178         node.setOwnerTree(this.getOwnerTree());
25179         this.fireEvent("insert", this.ownerTree, this, node, refNode);
25180         if(oldParent){
25181             node.fireEvent("move", this.ownerTree, node, oldParent, this, refIndex, refNode);
25182         }
25183         return node;
25184     },
25185
25186     /**
25187      * Returns the child node at the specified index.
25188      * @param {Number} index
25189      * @return {Node}
25190      */
25191     item : function(index){
25192         return this.childNodes[index];
25193     },
25194
25195     /**
25196      * Replaces one child node in this node with another.
25197      * @param {Node} newChild The replacement node
25198      * @param {Node} oldChild The node to replace
25199      * @return {Node} The replaced node
25200      */
25201     replaceChild : function(newChild, oldChild){
25202         this.insertBefore(newChild, oldChild);
25203         this.removeChild(oldChild);
25204         return oldChild;
25205     },
25206
25207     /**
25208      * Returns the index of a child node
25209      * @param {Node} node
25210      * @return {Number} The index of the node or -1 if it was not found
25211      */
25212     indexOf : function(child){
25213         return this.childNodes.indexOf(child);
25214     },
25215
25216     /**
25217      * Returns the tree this node is in.
25218      * @return {Tree}
25219      */
25220     getOwnerTree : function(){
25221         // if it doesn't have one, look for one
25222         if(!this.ownerTree){
25223             var p = this;
25224             while(p){
25225                 if(p.ownerTree){
25226                     this.ownerTree = p.ownerTree;
25227                     break;
25228                 }
25229                 p = p.parentNode;
25230             }
25231         }
25232         return this.ownerTree;
25233     },
25234
25235     /**
25236      * Returns depth of this node (the root node has a depth of 0)
25237      * @return {Number}
25238      */
25239     getDepth : function(){
25240         var depth = 0;
25241         var p = this;
25242         while(p.parentNode){
25243             ++depth;
25244             p = p.parentNode;
25245         }
25246         return depth;
25247     },
25248
25249     // private
25250     setOwnerTree : function(tree){
25251         // if it's move, we need to update everyone
25252         if(tree != this.ownerTree){
25253             if(this.ownerTree){
25254                 this.ownerTree.unregisterNode(this);
25255             }
25256             this.ownerTree = tree;
25257             var cs = this.childNodes;
25258             for(var i = 0, len = cs.length; i < len; i++) {
25259                 cs[i].setOwnerTree(tree);
25260             }
25261             if(tree){
25262                 tree.registerNode(this);
25263             }
25264         }
25265     },
25266
25267     /**
25268      * Returns the path for this node. The path can be used to expand or select this node programmatically.
25269      * @param {String} attr (optional) The attr to use for the path (defaults to the node's id)
25270      * @return {String} The path
25271      */
25272     getPath : function(attr){
25273         attr = attr || "id";
25274         var p = this.parentNode;
25275         var b = [this.attributes[attr]];
25276         while(p){
25277             b.unshift(p.attributes[attr]);
25278             p = p.parentNode;
25279         }
25280         var sep = this.getOwnerTree().pathSeparator;
25281         return sep + b.join(sep);
25282     },
25283
25284     /**
25285      * Bubbles up the tree from this node, calling the specified function with each node. The scope (<i>this</i>) of
25286      * function call will be the scope provided or the current node. The arguments to the function
25287      * will be the args provided or the current node. If the function returns false at any point,
25288      * the bubble is stopped.
25289      * @param {Function} fn The function to call
25290      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25291      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25292      */
25293     bubble : function(fn, scope, args){
25294         var p = this;
25295         while(p){
25296             if(fn.call(scope || p, args || p) === false){
25297                 break;
25298             }
25299             p = p.parentNode;
25300         }
25301     },
25302
25303     /**
25304      * Cascades down the tree from this node, calling the specified function with each node. The scope (<i>this</i>) of
25305      * function call will be the scope provided or the current node. The arguments to the function
25306      * will be the args provided or the current node. If the function returns false at any point,
25307      * the cascade is stopped on that branch.
25308      * @param {Function} fn The function to call
25309      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25310      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25311      */
25312     cascade : function(fn, scope, args){
25313         if(fn.call(scope || this, args || this) !== false){
25314             var cs = this.childNodes;
25315             for(var i = 0, len = cs.length; i < len; i++) {
25316                 cs[i].cascade(fn, scope, args);
25317             }
25318         }
25319     },
25320
25321     /**
25322      * Interates the child nodes of this node, calling the specified function with each node. The scope (<i>this</i>) of
25323      * function call will be the scope provided or the current node. The arguments to the function
25324      * will be the args provided or the current node. If the function returns false at any point,
25325      * the iteration stops.
25326      * @param {Function} fn The function to call
25327      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25328      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25329      */
25330     eachChild : function(fn, scope, args){
25331         var cs = this.childNodes;
25332         for(var i = 0, len = cs.length; i < len; i++) {
25333                 if(fn.call(scope || this, args || cs[i]) === false){
25334                     break;
25335                 }
25336         }
25337     },
25338
25339     /**
25340      * Finds the first child that has the attribute with the specified value.
25341      * @param {String} attribute The attribute name
25342      * @param {Mixed} value The value to search for
25343      * @return {Node} The found child or null if none was found
25344      */
25345     findChild : function(attribute, value){
25346         var cs = this.childNodes;
25347         for(var i = 0, len = cs.length; i < len; i++) {
25348                 if(cs[i].attributes[attribute] == value){
25349                     return cs[i];
25350                 }
25351         }
25352         return null;
25353     },
25354
25355     /**
25356      * Finds the first child by a custom function. The child matches if the function passed
25357      * returns true.
25358      * @param {Function} fn
25359      * @param {Object} scope (optional)
25360      * @return {Node} The found child or null if none was found
25361      */
25362     findChildBy : function(fn, scope){
25363         var cs = this.childNodes;
25364         for(var i = 0, len = cs.length; i < len; i++) {
25365                 if(fn.call(scope||cs[i], cs[i]) === true){
25366                     return cs[i];
25367                 }
25368         }
25369         return null;
25370     },
25371
25372     /**
25373      * Sorts this nodes children using the supplied sort function
25374      * @param {Function} fn
25375      * @param {Object} scope (optional)
25376      */
25377     sort : function(fn, scope){
25378         var cs = this.childNodes;
25379         var len = cs.length;
25380         if(len > 0){
25381             var sortFn = scope ? function(){fn.apply(scope, arguments);} : fn;
25382             cs.sort(sortFn);
25383             for(var i = 0; i < len; i++){
25384                 var n = cs[i];
25385                 n.previousSibling = cs[i-1];
25386                 n.nextSibling = cs[i+1];
25387                 if(i == 0){
25388                     this.setFirstChild(n);
25389                 }
25390                 if(i == len-1){
25391                     this.setLastChild(n);
25392                 }
25393             }
25394         }
25395     },
25396
25397     /**
25398      * Returns true if this node is an ancestor (at any point) of the passed node.
25399      * @param {Node} node
25400      * @return {Boolean}
25401      */
25402     contains : function(node){
25403         return node.isAncestor(this);
25404     },
25405
25406     /**
25407      * Returns true if the passed node is an ancestor (at any point) of this node.
25408      * @param {Node} node
25409      * @return {Boolean}
25410      */
25411     isAncestor : function(node){
25412         var p = this.parentNode;
25413         while(p){
25414             if(p == node){
25415                 return true;
25416             }
25417             p = p.parentNode;
25418         }
25419         return false;
25420     },
25421
25422     toString : function(){
25423         return "[Node"+(this.id?" "+this.id:"")+"]";
25424     }
25425 });/*
25426  * Based on:
25427  * Ext JS Library 1.1.1
25428  * Copyright(c) 2006-2007, Ext JS, LLC.
25429  *
25430  * Originally Released Under LGPL - original licence link has changed is not relivant.
25431  *
25432  * Fork - LGPL
25433  * <script type="text/javascript">
25434  */
25435  (function(){ 
25436 /**
25437  * @class Roo.Layer
25438  * @extends Roo.Element
25439  * An extended {@link Roo.Element} object that supports a shadow and shim, constrain to viewport and
25440  * automatic maintaining of shadow/shim positions.
25441  * @cfg {Boolean} shim False to disable the iframe shim in browsers which need one (defaults to true)
25442  * @cfg {String/Boolean} shadow True to create a shadow element with default class "x-layer-shadow", or
25443  * you can pass a string with a CSS class name. False turns off the shadow.
25444  * @cfg {Object} dh DomHelper object config to create element with (defaults to {tag: "div", cls: "x-layer"}).
25445  * @cfg {Boolean} constrain False to disable constrain to viewport (defaults to true)
25446  * @cfg {String} cls CSS class to add to the element
25447  * @cfg {Number} zindex Starting z-index (defaults to 11000)
25448  * @cfg {Number} shadowOffset Number of pixels to offset the shadow (defaults to 3)
25449  * @constructor
25450  * @param {Object} config An object with config options.
25451  * @param {String/HTMLElement} existingEl (optional) Uses an existing DOM element. If the element is not found it creates it.
25452  */
25453
25454 Roo.Layer = function(config, existingEl){
25455     config = config || {};
25456     var dh = Roo.DomHelper;
25457     var cp = config.parentEl, pel = cp ? Roo.getDom(cp) : document.body;
25458     if(existingEl){
25459         this.dom = Roo.getDom(existingEl);
25460     }
25461     if(!this.dom){
25462         var o = config.dh || {tag: "div", cls: "x-layer"};
25463         this.dom = dh.append(pel, o);
25464     }
25465     if(config.cls){
25466         this.addClass(config.cls);
25467     }
25468     this.constrain = config.constrain !== false;
25469     this.visibilityMode = Roo.Element.VISIBILITY;
25470     if(config.id){
25471         this.id = this.dom.id = config.id;
25472     }else{
25473         this.id = Roo.id(this.dom);
25474     }
25475     this.zindex = config.zindex || this.getZIndex();
25476     this.position("absolute", this.zindex);
25477     if(config.shadow){
25478         this.shadowOffset = config.shadowOffset || 4;
25479         this.shadow = new Roo.Shadow({
25480             offset : this.shadowOffset,
25481             mode : config.shadow
25482         });
25483     }else{
25484         this.shadowOffset = 0;
25485     }
25486     this.useShim = config.shim !== false && Roo.useShims;
25487     this.useDisplay = config.useDisplay;
25488     this.hide();
25489 };
25490
25491 var supr = Roo.Element.prototype;
25492
25493 // shims are shared among layer to keep from having 100 iframes
25494 var shims = [];
25495
25496 Roo.extend(Roo.Layer, Roo.Element, {
25497
25498     getZIndex : function(){
25499         return this.zindex || parseInt(this.getStyle("z-index"), 10) || 11000;
25500     },
25501
25502     getShim : function(){
25503         if(!this.useShim){
25504             return null;
25505         }
25506         if(this.shim){
25507             return this.shim;
25508         }
25509         var shim = shims.shift();
25510         if(!shim){
25511             shim = this.createShim();
25512             shim.enableDisplayMode('block');
25513             shim.dom.style.display = 'none';
25514             shim.dom.style.visibility = 'visible';
25515         }
25516         var pn = this.dom.parentNode;
25517         if(shim.dom.parentNode != pn){
25518             pn.insertBefore(shim.dom, this.dom);
25519         }
25520         shim.setStyle('z-index', this.getZIndex()-2);
25521         this.shim = shim;
25522         return shim;
25523     },
25524
25525     hideShim : function(){
25526         if(this.shim){
25527             this.shim.setDisplayed(false);
25528             shims.push(this.shim);
25529             delete this.shim;
25530         }
25531     },
25532
25533     disableShadow : function(){
25534         if(this.shadow){
25535             this.shadowDisabled = true;
25536             this.shadow.hide();
25537             this.lastShadowOffset = this.shadowOffset;
25538             this.shadowOffset = 0;
25539         }
25540     },
25541
25542     enableShadow : function(show){
25543         if(this.shadow){
25544             this.shadowDisabled = false;
25545             this.shadowOffset = this.lastShadowOffset;
25546             delete this.lastShadowOffset;
25547             if(show){
25548                 this.sync(true);
25549             }
25550         }
25551     },
25552
25553     // private
25554     // this code can execute repeatedly in milliseconds (i.e. during a drag) so
25555     // code size was sacrificed for effeciency (e.g. no getBox/setBox, no XY calls)
25556     sync : function(doShow){
25557         var sw = this.shadow;
25558         if(!this.updating && this.isVisible() && (sw || this.useShim)){
25559             var sh = this.getShim();
25560
25561             var w = this.getWidth(),
25562                 h = this.getHeight();
25563
25564             var l = this.getLeft(true),
25565                 t = this.getTop(true);
25566
25567             if(sw && !this.shadowDisabled){
25568                 if(doShow && !sw.isVisible()){
25569                     sw.show(this);
25570                 }else{
25571                     sw.realign(l, t, w, h);
25572                 }
25573                 if(sh){
25574                     if(doShow){
25575                        sh.show();
25576                     }
25577                     // fit the shim behind the shadow, so it is shimmed too
25578                     var a = sw.adjusts, s = sh.dom.style;
25579                     s.left = (Math.min(l, l+a.l))+"px";
25580                     s.top = (Math.min(t, t+a.t))+"px";
25581                     s.width = (w+a.w)+"px";
25582                     s.height = (h+a.h)+"px";
25583                 }
25584             }else if(sh){
25585                 if(doShow){
25586                    sh.show();
25587                 }
25588                 sh.setSize(w, h);
25589                 sh.setLeftTop(l, t);
25590             }
25591             
25592         }
25593     },
25594
25595     // private
25596     destroy : function(){
25597         this.hideShim();
25598         if(this.shadow){
25599             this.shadow.hide();
25600         }
25601         this.removeAllListeners();
25602         var pn = this.dom.parentNode;
25603         if(pn){
25604             pn.removeChild(this.dom);
25605         }
25606         Roo.Element.uncache(this.id);
25607     },
25608
25609     remove : function(){
25610         this.destroy();
25611     },
25612
25613     // private
25614     beginUpdate : function(){
25615         this.updating = true;
25616     },
25617
25618     // private
25619     endUpdate : function(){
25620         this.updating = false;
25621         this.sync(true);
25622     },
25623
25624     // private
25625     hideUnders : function(negOffset){
25626         if(this.shadow){
25627             this.shadow.hide();
25628         }
25629         this.hideShim();
25630     },
25631
25632     // private
25633     constrainXY : function(){
25634         if(this.constrain){
25635             var vw = Roo.lib.Dom.getViewWidth(),
25636                 vh = Roo.lib.Dom.getViewHeight();
25637             var s = Roo.get(document).getScroll();
25638
25639             var xy = this.getXY();
25640             var x = xy[0], y = xy[1];   
25641             var w = this.dom.offsetWidth+this.shadowOffset, h = this.dom.offsetHeight+this.shadowOffset;
25642             // only move it if it needs it
25643             var moved = false;
25644             // first validate right/bottom
25645             if((x + w) > vw+s.left){
25646                 x = vw - w - this.shadowOffset;
25647                 moved = true;
25648             }
25649             if((y + h) > vh+s.top){
25650                 y = vh - h - this.shadowOffset;
25651                 moved = true;
25652             }
25653             // then make sure top/left isn't negative
25654             if(x < s.left){
25655                 x = s.left;
25656                 moved = true;
25657             }
25658             if(y < s.top){
25659                 y = s.top;
25660                 moved = true;
25661             }
25662             if(moved){
25663                 if(this.avoidY){
25664                     var ay = this.avoidY;
25665                     if(y <= ay && (y+h) >= ay){
25666                         y = ay-h-5;   
25667                     }
25668                 }
25669                 xy = [x, y];
25670                 this.storeXY(xy);
25671                 supr.setXY.call(this, xy);
25672                 this.sync();
25673             }
25674         }
25675     },
25676
25677     isVisible : function(){
25678         return this.visible;    
25679     },
25680
25681     // private
25682     showAction : function(){
25683         this.visible = true; // track visibility to prevent getStyle calls
25684         if(this.useDisplay === true){
25685             this.setDisplayed("");
25686         }else if(this.lastXY){
25687             supr.setXY.call(this, this.lastXY);
25688         }else if(this.lastLT){
25689             supr.setLeftTop.call(this, this.lastLT[0], this.lastLT[1]);
25690         }
25691     },
25692
25693     // private
25694     hideAction : function(){
25695         this.visible = false;
25696         if(this.useDisplay === true){
25697             this.setDisplayed(false);
25698         }else{
25699             this.setLeftTop(-10000,-10000);
25700         }
25701     },
25702
25703     // overridden Element method
25704     setVisible : function(v, a, d, c, e){
25705         if(v){
25706             this.showAction();
25707         }
25708         if(a && v){
25709             var cb = function(){
25710                 this.sync(true);
25711                 if(c){
25712                     c();
25713                 }
25714             }.createDelegate(this);
25715             supr.setVisible.call(this, true, true, d, cb, e);
25716         }else{
25717             if(!v){
25718                 this.hideUnders(true);
25719             }
25720             var cb = c;
25721             if(a){
25722                 cb = function(){
25723                     this.hideAction();
25724                     if(c){
25725                         c();
25726                     }
25727                 }.createDelegate(this);
25728             }
25729             supr.setVisible.call(this, v, a, d, cb, e);
25730             if(v){
25731                 this.sync(true);
25732             }else if(!a){
25733                 this.hideAction();
25734             }
25735         }
25736     },
25737
25738     storeXY : function(xy){
25739         delete this.lastLT;
25740         this.lastXY = xy;
25741     },
25742
25743     storeLeftTop : function(left, top){
25744         delete this.lastXY;
25745         this.lastLT = [left, top];
25746     },
25747
25748     // private
25749     beforeFx : function(){
25750         this.beforeAction();
25751         return Roo.Layer.superclass.beforeFx.apply(this, arguments);
25752     },
25753
25754     // private
25755     afterFx : function(){
25756         Roo.Layer.superclass.afterFx.apply(this, arguments);
25757         this.sync(this.isVisible());
25758     },
25759
25760     // private
25761     beforeAction : function(){
25762         if(!this.updating && this.shadow){
25763             this.shadow.hide();
25764         }
25765     },
25766
25767     // overridden Element method
25768     setLeft : function(left){
25769         this.storeLeftTop(left, this.getTop(true));
25770         supr.setLeft.apply(this, arguments);
25771         this.sync();
25772     },
25773
25774     setTop : function(top){
25775         this.storeLeftTop(this.getLeft(true), top);
25776         supr.setTop.apply(this, arguments);
25777         this.sync();
25778     },
25779
25780     setLeftTop : function(left, top){
25781         this.storeLeftTop(left, top);
25782         supr.setLeftTop.apply(this, arguments);
25783         this.sync();
25784     },
25785
25786     setXY : function(xy, a, d, c, e){
25787         this.fixDisplay();
25788         this.beforeAction();
25789         this.storeXY(xy);
25790         var cb = this.createCB(c);
25791         supr.setXY.call(this, xy, a, d, cb, e);
25792         if(!a){
25793             cb();
25794         }
25795     },
25796
25797     // private
25798     createCB : function(c){
25799         var el = this;
25800         return function(){
25801             el.constrainXY();
25802             el.sync(true);
25803             if(c){
25804                 c();
25805             }
25806         };
25807     },
25808
25809     // overridden Element method
25810     setX : function(x, a, d, c, e){
25811         this.setXY([x, this.getY()], a, d, c, e);
25812     },
25813
25814     // overridden Element method
25815     setY : function(y, a, d, c, e){
25816         this.setXY([this.getX(), y], a, d, c, e);
25817     },
25818
25819     // overridden Element method
25820     setSize : function(w, h, a, d, c, e){
25821         this.beforeAction();
25822         var cb = this.createCB(c);
25823         supr.setSize.call(this, w, h, a, d, cb, e);
25824         if(!a){
25825             cb();
25826         }
25827     },
25828
25829     // overridden Element method
25830     setWidth : function(w, a, d, c, e){
25831         this.beforeAction();
25832         var cb = this.createCB(c);
25833         supr.setWidth.call(this, w, a, d, cb, e);
25834         if(!a){
25835             cb();
25836         }
25837     },
25838
25839     // overridden Element method
25840     setHeight : function(h, a, d, c, e){
25841         this.beforeAction();
25842         var cb = this.createCB(c);
25843         supr.setHeight.call(this, h, a, d, cb, e);
25844         if(!a){
25845             cb();
25846         }
25847     },
25848
25849     // overridden Element method
25850     setBounds : function(x, y, w, h, a, d, c, e){
25851         this.beforeAction();
25852         var cb = this.createCB(c);
25853         if(!a){
25854             this.storeXY([x, y]);
25855             supr.setXY.call(this, [x, y]);
25856             supr.setSize.call(this, w, h, a, d, cb, e);
25857             cb();
25858         }else{
25859             supr.setBounds.call(this, x, y, w, h, a, d, cb, e);
25860         }
25861         return this;
25862     },
25863     
25864     /**
25865      * Sets the z-index of this layer and adjusts any shadow and shim z-indexes. The layer z-index is automatically
25866      * incremented by two more than the value passed in so that it always shows above any shadow or shim (the shadow
25867      * element, if any, will be assigned z-index + 1, and the shim element, if any, will be assigned the unmodified z-index).
25868      * @param {Number} zindex The new z-index to set
25869      * @return {this} The Layer
25870      */
25871     setZIndex : function(zindex){
25872         this.zindex = zindex;
25873         this.setStyle("z-index", zindex + 2);
25874         if(this.shadow){
25875             this.shadow.setZIndex(zindex + 1);
25876         }
25877         if(this.shim){
25878             this.shim.setStyle("z-index", zindex);
25879         }
25880     }
25881 });
25882 })();/*
25883  * Based on:
25884  * Ext JS Library 1.1.1
25885  * Copyright(c) 2006-2007, Ext JS, LLC.
25886  *
25887  * Originally Released Under LGPL - original licence link has changed is not relivant.
25888  *
25889  * Fork - LGPL
25890  * <script type="text/javascript">
25891  */
25892
25893
25894 /**
25895  * @class Roo.Shadow
25896  * Simple class that can provide a shadow effect for any element.  Note that the element MUST be absolutely positioned,
25897  * and the shadow does not provide any shimming.  This should be used only in simple cases -- for more advanced
25898  * functionality that can also provide the same shadow effect, see the {@link Roo.Layer} class.
25899  * @constructor
25900  * Create a new Shadow
25901  * @param {Object} config The config object
25902  */
25903 Roo.Shadow = function(config){
25904     Roo.apply(this, config);
25905     if(typeof this.mode != "string"){
25906         this.mode = this.defaultMode;
25907     }
25908     var o = this.offset, a = {h: 0};
25909     var rad = Math.floor(this.offset/2);
25910     switch(this.mode.toLowerCase()){ // all this hideous nonsense calculates the various offsets for shadows
25911         case "drop":
25912             a.w = 0;
25913             a.l = a.t = o;
25914             a.t -= 1;
25915             if(Roo.isIE){
25916                 a.l -= this.offset + rad;
25917                 a.t -= this.offset + rad;
25918                 a.w -= rad;
25919                 a.h -= rad;
25920                 a.t += 1;
25921             }
25922         break;
25923         case "sides":
25924             a.w = (o*2);
25925             a.l = -o;
25926             a.t = o-1;
25927             if(Roo.isIE){
25928                 a.l -= (this.offset - rad);
25929                 a.t -= this.offset + rad;
25930                 a.l += 1;
25931                 a.w -= (this.offset - rad)*2;
25932                 a.w -= rad + 1;
25933                 a.h -= 1;
25934             }
25935         break;
25936         case "frame":
25937             a.w = a.h = (o*2);
25938             a.l = a.t = -o;
25939             a.t += 1;
25940             a.h -= 2;
25941             if(Roo.isIE){
25942                 a.l -= (this.offset - rad);
25943                 a.t -= (this.offset - rad);
25944                 a.l += 1;
25945                 a.w -= (this.offset + rad + 1);
25946                 a.h -= (this.offset + rad);
25947                 a.h += 1;
25948             }
25949         break;
25950     };
25951
25952     this.adjusts = a;
25953 };
25954
25955 Roo.Shadow.prototype = {
25956     /**
25957      * @cfg {String} mode
25958      * The shadow display mode.  Supports the following options:<br />
25959      * sides: Shadow displays on both sides and bottom only<br />
25960      * frame: Shadow displays equally on all four sides<br />
25961      * drop: Traditional bottom-right drop shadow (default)
25962      */
25963     /**
25964      * @cfg {String} offset
25965      * The number of pixels to offset the shadow from the element (defaults to 4)
25966      */
25967     offset: 4,
25968
25969     // private
25970     defaultMode: "drop",
25971
25972     /**
25973      * Displays the shadow under the target element
25974      * @param {String/HTMLElement/Element} targetEl The id or element under which the shadow should display
25975      */
25976     show : function(target){
25977         target = Roo.get(target);
25978         if(!this.el){
25979             this.el = Roo.Shadow.Pool.pull();
25980             if(this.el.dom.nextSibling != target.dom){
25981                 this.el.insertBefore(target);
25982             }
25983         }
25984         this.el.setStyle("z-index", this.zIndex || parseInt(target.getStyle("z-index"), 10)-1);
25985         if(Roo.isIE){
25986             this.el.dom.style.filter="progid:DXImageTransform.Microsoft.alpha(opacity=50) progid:DXImageTransform.Microsoft.Blur(pixelradius="+(this.offset)+")";
25987         }
25988         this.realign(
25989             target.getLeft(true),
25990             target.getTop(true),
25991             target.getWidth(),
25992             target.getHeight()
25993         );
25994         this.el.dom.style.display = "block";
25995     },
25996
25997     /**
25998      * Returns true if the shadow is visible, else false
25999      */
26000     isVisible : function(){
26001         return this.el ? true : false;  
26002     },
26003
26004     /**
26005      * Direct alignment when values are already available. Show must be called at least once before
26006      * calling this method to ensure it is initialized.
26007      * @param {Number} left The target element left position
26008      * @param {Number} top The target element top position
26009      * @param {Number} width The target element width
26010      * @param {Number} height The target element height
26011      */
26012     realign : function(l, t, w, h){
26013         if(!this.el){
26014             return;
26015         }
26016         var a = this.adjusts, d = this.el.dom, s = d.style;
26017         var iea = 0;
26018         s.left = (l+a.l)+"px";
26019         s.top = (t+a.t)+"px";
26020         var sw = (w+a.w), sh = (h+a.h), sws = sw +"px", shs = sh + "px";
26021  
26022         if(s.width != sws || s.height != shs){
26023             s.width = sws;
26024             s.height = shs;
26025             if(!Roo.isIE){
26026                 var cn = d.childNodes;
26027                 var sww = Math.max(0, (sw-12))+"px";
26028                 cn[0].childNodes[1].style.width = sww;
26029                 cn[1].childNodes[1].style.width = sww;
26030                 cn[2].childNodes[1].style.width = sww;
26031                 cn[1].style.height = Math.max(0, (sh-12))+"px";
26032             }
26033         }
26034     },
26035
26036     /**
26037      * Hides this shadow
26038      */
26039     hide : function(){
26040         if(this.el){
26041             this.el.dom.style.display = "none";
26042             Roo.Shadow.Pool.push(this.el);
26043             delete this.el;
26044         }
26045     },
26046
26047     /**
26048      * Adjust the z-index of this shadow
26049      * @param {Number} zindex The new z-index
26050      */
26051     setZIndex : function(z){
26052         this.zIndex = z;
26053         if(this.el){
26054             this.el.setStyle("z-index", z);
26055         }
26056     }
26057 };
26058
26059 // Private utility class that manages the internal Shadow cache
26060 Roo.Shadow.Pool = function(){
26061     var p = [];
26062     var markup = Roo.isIE ?
26063                  '<div class="x-ie-shadow"></div>' :
26064                  '<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>';
26065     return {
26066         pull : function(){
26067             var sh = p.shift();
26068             if(!sh){
26069                 sh = Roo.get(Roo.DomHelper.insertHtml("beforeBegin", document.body.firstChild, markup));
26070                 sh.autoBoxAdjust = false;
26071             }
26072             return sh;
26073         },
26074
26075         push : function(sh){
26076             p.push(sh);
26077         }
26078     };
26079 }();/*
26080  * Based on:
26081  * Ext JS Library 1.1.1
26082  * Copyright(c) 2006-2007, Ext JS, LLC.
26083  *
26084  * Originally Released Under LGPL - original licence link has changed is not relivant.
26085  *
26086  * Fork - LGPL
26087  * <script type="text/javascript">
26088  */
26089
26090
26091 /**
26092  * @class Roo.SplitBar
26093  * @extends Roo.util.Observable
26094  * Creates draggable splitter bar functionality from two elements (element to be dragged and element to be resized).
26095  * <br><br>
26096  * Usage:
26097  * <pre><code>
26098 var split = new Roo.SplitBar("elementToDrag", "elementToSize",
26099                    Roo.SplitBar.HORIZONTAL, Roo.SplitBar.LEFT);
26100 split.setAdapter(new Roo.SplitBar.AbsoluteLayoutAdapter("container"));
26101 split.minSize = 100;
26102 split.maxSize = 600;
26103 split.animate = true;
26104 split.on('moved', splitterMoved);
26105 </code></pre>
26106  * @constructor
26107  * Create a new SplitBar
26108  * @param {String/HTMLElement/Roo.Element} dragElement The element to be dragged and act as the SplitBar. 
26109  * @param {String/HTMLElement/Roo.Element} resizingElement The element to be resized based on where the SplitBar element is dragged 
26110  * @param {Number} orientation (optional) Either Roo.SplitBar.HORIZONTAL or Roo.SplitBar.VERTICAL. (Defaults to HORIZONTAL)
26111  * @param {Number} placement (optional) Either Roo.SplitBar.LEFT or Roo.SplitBar.RIGHT for horizontal or  
26112                         Roo.SplitBar.TOP or Roo.SplitBar.BOTTOM for vertical. (By default, this is determined automatically by the initial
26113                         position of the SplitBar).
26114  */
26115 Roo.SplitBar = function(dragElement, resizingElement, orientation, placement, existingProxy){
26116     
26117     /** @private */
26118     this.el = Roo.get(dragElement, true);
26119     this.el.dom.unselectable = "on";
26120     /** @private */
26121     this.resizingEl = Roo.get(resizingElement, true);
26122
26123     /**
26124      * @private
26125      * The orientation of the split. Either Roo.SplitBar.HORIZONTAL or Roo.SplitBar.VERTICAL. (Defaults to HORIZONTAL)
26126      * Note: If this is changed after creating the SplitBar, the placement property must be manually updated
26127      * @type Number
26128      */
26129     this.orientation = orientation || Roo.SplitBar.HORIZONTAL;
26130     
26131     /**
26132      * The minimum size of the resizing element. (Defaults to 0)
26133      * @type Number
26134      */
26135     this.minSize = 0;
26136     
26137     /**
26138      * The maximum size of the resizing element. (Defaults to 2000)
26139      * @type Number
26140      */
26141     this.maxSize = 2000;
26142     
26143     /**
26144      * Whether to animate the transition to the new size
26145      * @type Boolean
26146      */
26147     this.animate = false;
26148     
26149     /**
26150      * Whether to create a transparent shim that overlays the page when dragging, enables dragging across iframes.
26151      * @type Boolean
26152      */
26153     this.useShim = false;
26154     
26155     /** @private */
26156     this.shim = null;
26157     
26158     if(!existingProxy){
26159         /** @private */
26160         this.proxy = Roo.SplitBar.createProxy(this.orientation);
26161     }else{
26162         this.proxy = Roo.get(existingProxy).dom;
26163     }
26164     /** @private */
26165     this.dd = new Roo.dd.DDProxy(this.el.dom.id, "XSplitBars", {dragElId : this.proxy.id});
26166     
26167     /** @private */
26168     this.dd.b4StartDrag = this.onStartProxyDrag.createDelegate(this);
26169     
26170     /** @private */
26171     this.dd.endDrag = this.onEndProxyDrag.createDelegate(this);
26172     
26173     /** @private */
26174     this.dragSpecs = {};
26175     
26176     /**
26177      * @private The adapter to use to positon and resize elements
26178      */
26179     this.adapter = new Roo.SplitBar.BasicLayoutAdapter();
26180     this.adapter.init(this);
26181     
26182     if(this.orientation == Roo.SplitBar.HORIZONTAL){
26183         /** @private */
26184         this.placement = placement || (this.el.getX() > this.resizingEl.getX() ? Roo.SplitBar.LEFT : Roo.SplitBar.RIGHT);
26185         this.el.addClass("x-splitbar-h");
26186     }else{
26187         /** @private */
26188         this.placement = placement || (this.el.getY() > this.resizingEl.getY() ? Roo.SplitBar.TOP : Roo.SplitBar.BOTTOM);
26189         this.el.addClass("x-splitbar-v");
26190     }
26191     
26192     this.addEvents({
26193         /**
26194          * @event resize
26195          * Fires when the splitter is moved (alias for {@link #event-moved})
26196          * @param {Roo.SplitBar} this
26197          * @param {Number} newSize the new width or height
26198          */
26199         "resize" : true,
26200         /**
26201          * @event moved
26202          * Fires when the splitter is moved
26203          * @param {Roo.SplitBar} this
26204          * @param {Number} newSize the new width or height
26205          */
26206         "moved" : true,
26207         /**
26208          * @event beforeresize
26209          * Fires before the splitter is dragged
26210          * @param {Roo.SplitBar} this
26211          */
26212         "beforeresize" : true,
26213
26214         "beforeapply" : true
26215     });
26216
26217     Roo.util.Observable.call(this);
26218 };
26219
26220 Roo.extend(Roo.SplitBar, Roo.util.Observable, {
26221     onStartProxyDrag : function(x, y){
26222         this.fireEvent("beforeresize", this);
26223         if(!this.overlay){
26224             var o = Roo.DomHelper.insertFirst(document.body,  {cls: "x-drag-overlay", html: "&#160;"}, true);
26225             o.unselectable();
26226             o.enableDisplayMode("block");
26227             // all splitbars share the same overlay
26228             Roo.SplitBar.prototype.overlay = o;
26229         }
26230         this.overlay.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
26231         this.overlay.show();
26232         Roo.get(this.proxy).setDisplayed("block");
26233         var size = this.adapter.getElementSize(this);
26234         this.activeMinSize = this.getMinimumSize();;
26235         this.activeMaxSize = this.getMaximumSize();;
26236         var c1 = size - this.activeMinSize;
26237         var c2 = Math.max(this.activeMaxSize - size, 0);
26238         if(this.orientation == Roo.SplitBar.HORIZONTAL){
26239             this.dd.resetConstraints();
26240             this.dd.setXConstraint(
26241                 this.placement == Roo.SplitBar.LEFT ? c1 : c2, 
26242                 this.placement == Roo.SplitBar.LEFT ? c2 : c1
26243             );
26244             this.dd.setYConstraint(0, 0);
26245         }else{
26246             this.dd.resetConstraints();
26247             this.dd.setXConstraint(0, 0);
26248             this.dd.setYConstraint(
26249                 this.placement == Roo.SplitBar.TOP ? c1 : c2, 
26250                 this.placement == Roo.SplitBar.TOP ? c2 : c1
26251             );
26252          }
26253         this.dragSpecs.startSize = size;
26254         this.dragSpecs.startPoint = [x, y];
26255         Roo.dd.DDProxy.prototype.b4StartDrag.call(this.dd, x, y);
26256     },
26257     
26258     /** 
26259      * @private Called after the drag operation by the DDProxy
26260      */
26261     onEndProxyDrag : function(e){
26262         Roo.get(this.proxy).setDisplayed(false);
26263         var endPoint = Roo.lib.Event.getXY(e);
26264         if(this.overlay){
26265             this.overlay.hide();
26266         }
26267         var newSize;
26268         if(this.orientation == Roo.SplitBar.HORIZONTAL){
26269             newSize = this.dragSpecs.startSize + 
26270                 (this.placement == Roo.SplitBar.LEFT ?
26271                     endPoint[0] - this.dragSpecs.startPoint[0] :
26272                     this.dragSpecs.startPoint[0] - endPoint[0]
26273                 );
26274         }else{
26275             newSize = this.dragSpecs.startSize + 
26276                 (this.placement == Roo.SplitBar.TOP ?
26277                     endPoint[1] - this.dragSpecs.startPoint[1] :
26278                     this.dragSpecs.startPoint[1] - endPoint[1]
26279                 );
26280         }
26281         newSize = Math.min(Math.max(newSize, this.activeMinSize), this.activeMaxSize);
26282         if(newSize != this.dragSpecs.startSize){
26283             if(this.fireEvent('beforeapply', this, newSize) !== false){
26284                 this.adapter.setElementSize(this, newSize);
26285                 this.fireEvent("moved", this, newSize);
26286                 this.fireEvent("resize", this, newSize);
26287             }
26288         }
26289     },
26290     
26291     /**
26292      * Get the adapter this SplitBar uses
26293      * @return The adapter object
26294      */
26295     getAdapter : function(){
26296         return this.adapter;
26297     },
26298     
26299     /**
26300      * Set the adapter this SplitBar uses
26301      * @param {Object} adapter A SplitBar adapter object
26302      */
26303     setAdapter : function(adapter){
26304         this.adapter = adapter;
26305         this.adapter.init(this);
26306     },
26307     
26308     /**
26309      * Gets the minimum size for the resizing element
26310      * @return {Number} The minimum size
26311      */
26312     getMinimumSize : function(){
26313         return this.minSize;
26314     },
26315     
26316     /**
26317      * Sets the minimum size for the resizing element
26318      * @param {Number} minSize The minimum size
26319      */
26320     setMinimumSize : function(minSize){
26321         this.minSize = minSize;
26322     },
26323     
26324     /**
26325      * Gets the maximum size for the resizing element
26326      * @return {Number} The maximum size
26327      */
26328     getMaximumSize : function(){
26329         return this.maxSize;
26330     },
26331     
26332     /**
26333      * Sets the maximum size for the resizing element
26334      * @param {Number} maxSize The maximum size
26335      */
26336     setMaximumSize : function(maxSize){
26337         this.maxSize = maxSize;
26338     },
26339     
26340     /**
26341      * Sets the initialize size for the resizing element
26342      * @param {Number} size The initial size
26343      */
26344     setCurrentSize : function(size){
26345         var oldAnimate = this.animate;
26346         this.animate = false;
26347         this.adapter.setElementSize(this, size);
26348         this.animate = oldAnimate;
26349     },
26350     
26351     /**
26352      * Destroy this splitbar. 
26353      * @param {Boolean} removeEl True to remove the element
26354      */
26355     destroy : function(removeEl){
26356         if(this.shim){
26357             this.shim.remove();
26358         }
26359         this.dd.unreg();
26360         this.proxy.parentNode.removeChild(this.proxy);
26361         if(removeEl){
26362             this.el.remove();
26363         }
26364     }
26365 });
26366
26367 /**
26368  * @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.
26369  */
26370 Roo.SplitBar.createProxy = function(dir){
26371     var proxy = new Roo.Element(document.createElement("div"));
26372     proxy.unselectable();
26373     var cls = 'x-splitbar-proxy';
26374     proxy.addClass(cls + ' ' + (dir == Roo.SplitBar.HORIZONTAL ? cls +'-h' : cls + '-v'));
26375     document.body.appendChild(proxy.dom);
26376     return proxy.dom;
26377 };
26378
26379 /** 
26380  * @class Roo.SplitBar.BasicLayoutAdapter
26381  * Default Adapter. It assumes the splitter and resizing element are not positioned
26382  * elements and only gets/sets the width of the element. Generally used for table based layouts.
26383  */
26384 Roo.SplitBar.BasicLayoutAdapter = function(){
26385 };
26386
26387 Roo.SplitBar.BasicLayoutAdapter.prototype = {
26388     // do nothing for now
26389     init : function(s){
26390     
26391     },
26392     /**
26393      * Called before drag operations to get the current size of the resizing element. 
26394      * @param {Roo.SplitBar} s The SplitBar using this adapter
26395      */
26396      getElementSize : function(s){
26397         if(s.orientation == Roo.SplitBar.HORIZONTAL){
26398             return s.resizingEl.getWidth();
26399         }else{
26400             return s.resizingEl.getHeight();
26401         }
26402     },
26403     
26404     /**
26405      * Called after drag operations to set the size of the resizing element.
26406      * @param {Roo.SplitBar} s The SplitBar using this adapter
26407      * @param {Number} newSize The new size to set
26408      * @param {Function} onComplete A function to be invoked when resizing is complete
26409      */
26410     setElementSize : function(s, newSize, onComplete){
26411         if(s.orientation == Roo.SplitBar.HORIZONTAL){
26412             if(!s.animate){
26413                 s.resizingEl.setWidth(newSize);
26414                 if(onComplete){
26415                     onComplete(s, newSize);
26416                 }
26417             }else{
26418                 s.resizingEl.setWidth(newSize, true, .1, onComplete, 'easeOut');
26419             }
26420         }else{
26421             
26422             if(!s.animate){
26423                 s.resizingEl.setHeight(newSize);
26424                 if(onComplete){
26425                     onComplete(s, newSize);
26426                 }
26427             }else{
26428                 s.resizingEl.setHeight(newSize, true, .1, onComplete, 'easeOut');
26429             }
26430         }
26431     }
26432 };
26433
26434 /** 
26435  *@class Roo.SplitBar.AbsoluteLayoutAdapter
26436  * @extends Roo.SplitBar.BasicLayoutAdapter
26437  * Adapter that  moves the splitter element to align with the resized sizing element. 
26438  * Used with an absolute positioned SplitBar.
26439  * @param {String/HTMLElement/Roo.Element} container The container that wraps around the absolute positioned content. If it's
26440  * document.body, make sure you assign an id to the body element.
26441  */
26442 Roo.SplitBar.AbsoluteLayoutAdapter = function(container){
26443     this.basic = new Roo.SplitBar.BasicLayoutAdapter();
26444     this.container = Roo.get(container);
26445 };
26446
26447 Roo.SplitBar.AbsoluteLayoutAdapter.prototype = {
26448     init : function(s){
26449         this.basic.init(s);
26450     },
26451     
26452     getElementSize : function(s){
26453         return this.basic.getElementSize(s);
26454     },
26455     
26456     setElementSize : function(s, newSize, onComplete){
26457         this.basic.setElementSize(s, newSize, this.moveSplitter.createDelegate(this, [s]));
26458     },
26459     
26460     moveSplitter : function(s){
26461         var yes = Roo.SplitBar;
26462         switch(s.placement){
26463             case yes.LEFT:
26464                 s.el.setX(s.resizingEl.getRight());
26465                 break;
26466             case yes.RIGHT:
26467                 s.el.setStyle("right", (this.container.getWidth() - s.resizingEl.getLeft()) + "px");
26468                 break;
26469             case yes.TOP:
26470                 s.el.setY(s.resizingEl.getBottom());
26471                 break;
26472             case yes.BOTTOM:
26473                 s.el.setY(s.resizingEl.getTop() - s.el.getHeight());
26474                 break;
26475         }
26476     }
26477 };
26478
26479 /**
26480  * Orientation constant - Create a vertical SplitBar
26481  * @static
26482  * @type Number
26483  */
26484 Roo.SplitBar.VERTICAL = 1;
26485
26486 /**
26487  * Orientation constant - Create a horizontal SplitBar
26488  * @static
26489  * @type Number
26490  */
26491 Roo.SplitBar.HORIZONTAL = 2;
26492
26493 /**
26494  * Placement constant - The resizing element is to the left of the splitter element
26495  * @static
26496  * @type Number
26497  */
26498 Roo.SplitBar.LEFT = 1;
26499
26500 /**
26501  * Placement constant - The resizing element is to the right of the splitter element
26502  * @static
26503  * @type Number
26504  */
26505 Roo.SplitBar.RIGHT = 2;
26506
26507 /**
26508  * Placement constant - The resizing element is positioned above the splitter element
26509  * @static
26510  * @type Number
26511  */
26512 Roo.SplitBar.TOP = 3;
26513
26514 /**
26515  * Placement constant - The resizing element is positioned under splitter element
26516  * @static
26517  * @type Number
26518  */
26519 Roo.SplitBar.BOTTOM = 4;
26520 /*
26521  * Based on:
26522  * Ext JS Library 1.1.1
26523  * Copyright(c) 2006-2007, Ext JS, LLC.
26524  *
26525  * Originally Released Under LGPL - original licence link has changed is not relivant.
26526  *
26527  * Fork - LGPL
26528  * <script type="text/javascript">
26529  */
26530
26531 /**
26532  * @class Roo.View
26533  * @extends Roo.util.Observable
26534  * Create a "View" for an element based on a data model or UpdateManager and the supplied DomHelper template. 
26535  * This class also supports single and multi selection modes. <br>
26536  * Create a data model bound view:
26537  <pre><code>
26538  var store = new Roo.data.Store(...);
26539
26540  var view = new Roo.View({
26541     el : "my-element",
26542     tpl : '&lt;div id="{0}"&gt;{2} - {1}&lt;/div&gt;', // auto create template
26543  
26544     singleSelect: true,
26545     selectedClass: "ydataview-selected",
26546     store: store
26547  });
26548
26549  // listen for node click?
26550  view.on("click", function(vw, index, node, e){
26551  alert('Node "' + node.id + '" at index: ' + index + " was clicked.");
26552  });
26553
26554  // load XML data
26555  dataModel.load("foobar.xml");
26556  </code></pre>
26557  For an example of creating a JSON/UpdateManager view, see {@link Roo.JsonView}.
26558  * <br><br>
26559  * <b>Note: The root of your template must be a single node. Table/row implementations may work but are not supported due to
26560  * IE"s limited insertion support with tables and Opera"s faulty event bubbling.</b>
26561  * 
26562  * Note: old style constructor is still suported (container, template, config)
26563  * 
26564  * @constructor
26565  * Create a new View
26566  * @param {Object} config The config object
26567  * 
26568  */
26569 Roo.View = function(config, depreciated_tpl, depreciated_config){
26570     
26571     this.parent = false;
26572     
26573     if (typeof(depreciated_tpl) == 'undefined') {
26574         // new way.. - universal constructor.
26575         Roo.apply(this, config);
26576         this.el  = Roo.get(this.el);
26577     } else {
26578         // old format..
26579         this.el  = Roo.get(config);
26580         this.tpl = depreciated_tpl;
26581         Roo.apply(this, depreciated_config);
26582     }
26583     this.wrapEl  = this.el.wrap().wrap();
26584     ///this.el = this.wrapEla.appendChild(document.createElement("div"));
26585     
26586     
26587     if(typeof(this.tpl) == "string"){
26588         this.tpl = new Roo.Template(this.tpl);
26589     } else {
26590         // support xtype ctors..
26591         this.tpl = new Roo.factory(this.tpl, Roo);
26592     }
26593     
26594     
26595     this.tpl.compile();
26596     
26597     /** @private */
26598     this.addEvents({
26599         /**
26600          * @event beforeclick
26601          * Fires before a click is processed. Returns false to cancel the default action.
26602          * @param {Roo.View} this
26603          * @param {Number} index The index of the target node
26604          * @param {HTMLElement} node The target node
26605          * @param {Roo.EventObject} e The raw event object
26606          */
26607             "beforeclick" : true,
26608         /**
26609          * @event click
26610          * Fires when a template node is clicked.
26611          * @param {Roo.View} this
26612          * @param {Number} index The index of the target node
26613          * @param {HTMLElement} node The target node
26614          * @param {Roo.EventObject} e The raw event object
26615          */
26616             "click" : true,
26617         /**
26618          * @event dblclick
26619          * Fires when a template node is double clicked.
26620          * @param {Roo.View} this
26621          * @param {Number} index The index of the target node
26622          * @param {HTMLElement} node The target node
26623          * @param {Roo.EventObject} e The raw event object
26624          */
26625             "dblclick" : true,
26626         /**
26627          * @event contextmenu
26628          * Fires when a template node is right clicked.
26629          * @param {Roo.View} this
26630          * @param {Number} index The index of the target node
26631          * @param {HTMLElement} node The target node
26632          * @param {Roo.EventObject} e The raw event object
26633          */
26634             "contextmenu" : true,
26635         /**
26636          * @event selectionchange
26637          * Fires when the selected nodes change.
26638          * @param {Roo.View} this
26639          * @param {Array} selections Array of the selected nodes
26640          */
26641             "selectionchange" : true,
26642     
26643         /**
26644          * @event beforeselect
26645          * Fires before a selection is made. If any handlers return false, the selection is cancelled.
26646          * @param {Roo.View} this
26647          * @param {HTMLElement} node The node to be selected
26648          * @param {Array} selections Array of currently selected nodes
26649          */
26650             "beforeselect" : true,
26651         /**
26652          * @event preparedata
26653          * Fires on every row to render, to allow you to change the data.
26654          * @param {Roo.View} this
26655          * @param {Object} data to be rendered (change this)
26656          */
26657           "preparedata" : true
26658           
26659           
26660         });
26661
26662
26663
26664     this.el.on({
26665         "click": this.onClick,
26666         "dblclick": this.onDblClick,
26667         "contextmenu": this.onContextMenu,
26668         scope:this
26669     });
26670
26671     this.selections = [];
26672     this.nodes = [];
26673     this.cmp = new Roo.CompositeElementLite([]);
26674     if(this.store){
26675         this.store = Roo.factory(this.store, Roo.data);
26676         this.setStore(this.store, true);
26677     }
26678     
26679     if ( this.footer && this.footer.xtype) {
26680            
26681          var fctr = this.wrapEl.appendChild(document.createElement("div"));
26682         
26683         this.footer.dataSource = this.store;
26684         this.footer.container = fctr;
26685         this.footer = Roo.factory(this.footer, Roo);
26686         fctr.insertFirst(this.el);
26687         
26688         // this is a bit insane - as the paging toolbar seems to detach the el..
26689 //        dom.parentNode.parentNode.parentNode
26690          // they get detached?
26691     }
26692     
26693     
26694     Roo.View.superclass.constructor.call(this);
26695     
26696     
26697 };
26698
26699 Roo.extend(Roo.View, Roo.util.Observable, {
26700     
26701      /**
26702      * @cfg {Roo.data.Store} store Data store to load data from.
26703      */
26704     store : false,
26705     
26706     /**
26707      * @cfg {String|Roo.Element} el The container element.
26708      */
26709     el : '',
26710     
26711     /**
26712      * @cfg {String|Roo.Template} tpl The template used by this View 
26713      */
26714     tpl : false,
26715     /**
26716      * @cfg {String} dataName the named area of the template to use as the data area
26717      *                          Works with domtemplates roo-name="name"
26718      */
26719     dataName: false,
26720     /**
26721      * @cfg {String} selectedClass The css class to add to selected nodes
26722      */
26723     selectedClass : "x-view-selected",
26724      /**
26725      * @cfg {String} emptyText The empty text to show when nothing is loaded.
26726      */
26727     emptyText : "",
26728     
26729     /**
26730      * @cfg {String} text to display on mask (default Loading)
26731      */
26732     mask : false,
26733     /**
26734      * @cfg {Boolean} multiSelect Allow multiple selection
26735      */
26736     multiSelect : false,
26737     /**
26738      * @cfg {Boolean} singleSelect Allow single selection
26739      */
26740     singleSelect:  false,
26741     
26742     /**
26743      * @cfg {Boolean} toggleSelect - selecting 
26744      */
26745     toggleSelect : false,
26746     
26747     /**
26748      * @cfg {Boolean} tickable - selecting 
26749      */
26750     tickable : false,
26751     
26752     /**
26753      * Returns the element this view is bound to.
26754      * @return {Roo.Element}
26755      */
26756     getEl : function(){
26757         return this.wrapEl;
26758     },
26759     
26760     
26761
26762     /**
26763      * Refreshes the view. - called by datachanged on the store. - do not call directly.
26764      */
26765     refresh : function(){
26766         //Roo.log('refresh');
26767         var t = this.tpl;
26768         
26769         // if we are using something like 'domtemplate', then
26770         // the what gets used is:
26771         // t.applySubtemplate(NAME, data, wrapping data..)
26772         // the outer template then get' applied with
26773         //     the store 'extra data'
26774         // and the body get's added to the
26775         //      roo-name="data" node?
26776         //      <span class='roo-tpl-{name}'></span> ?????
26777         
26778         
26779         
26780         this.clearSelections();
26781         this.el.update("");
26782         var html = [];
26783         var records = this.store.getRange();
26784         if(records.length < 1) {
26785             
26786             // is this valid??  = should it render a template??
26787             
26788             this.el.update(this.emptyText);
26789             return;
26790         }
26791         var el = this.el;
26792         if (this.dataName) {
26793             this.el.update(t.apply(this.store.meta)); //????
26794             el = this.el.child('.roo-tpl-' + this.dataName);
26795         }
26796         
26797         for(var i = 0, len = records.length; i < len; i++){
26798             var data = this.prepareData(records[i].data, i, records[i]);
26799             this.fireEvent("preparedata", this, data, i, records[i]);
26800             
26801             var d = Roo.apply({}, data);
26802             
26803             if(this.tickable){
26804                 Roo.apply(d, {'roo-id' : Roo.id()});
26805                 
26806                 var _this = this;
26807             
26808                 Roo.each(this.parent.item, function(item){
26809                     if(item[_this.parent.valueField] != data[_this.parent.valueField]){
26810                         return;
26811                     }
26812                     Roo.apply(d, {'roo-data-checked' : 'checked'});
26813                 });
26814             }
26815             
26816             html[html.length] = Roo.util.Format.trim(
26817                 this.dataName ?
26818                     t.applySubtemplate(this.dataName, d, this.store.meta) :
26819                     t.apply(d)
26820             );
26821         }
26822         
26823         
26824         
26825         el.update(html.join(""));
26826         this.nodes = el.dom.childNodes;
26827         this.updateIndexes(0);
26828     },
26829     
26830
26831     /**
26832      * Function to override to reformat the data that is sent to
26833      * the template for each node.
26834      * DEPRICATED - use the preparedata event handler.
26835      * @param {Array/Object} data The raw data (array of colData for a data model bound view or
26836      * a JSON object for an UpdateManager bound view).
26837      */
26838     prepareData : function(data, index, record)
26839     {
26840         this.fireEvent("preparedata", this, data, index, record);
26841         return data;
26842     },
26843
26844     onUpdate : function(ds, record){
26845         // Roo.log('on update');   
26846         this.clearSelections();
26847         var index = this.store.indexOf(record);
26848         var n = this.nodes[index];
26849         this.tpl.insertBefore(n, this.prepareData(record.data, index, record));
26850         n.parentNode.removeChild(n);
26851         this.updateIndexes(index, index);
26852     },
26853
26854     
26855     
26856 // --------- FIXME     
26857     onAdd : function(ds, records, index)
26858     {
26859         //Roo.log(['on Add', ds, records, index] );        
26860         this.clearSelections();
26861         if(this.nodes.length == 0){
26862             this.refresh();
26863             return;
26864         }
26865         var n = this.nodes[index];
26866         for(var i = 0, len = records.length; i < len; i++){
26867             var d = this.prepareData(records[i].data, i, records[i]);
26868             if(n){
26869                 this.tpl.insertBefore(n, d);
26870             }else{
26871                 
26872                 this.tpl.append(this.el, d);
26873             }
26874         }
26875         this.updateIndexes(index);
26876     },
26877
26878     onRemove : function(ds, record, index){
26879        // Roo.log('onRemove');
26880         this.clearSelections();
26881         var el = this.dataName  ?
26882             this.el.child('.roo-tpl-' + this.dataName) :
26883             this.el; 
26884         
26885         el.dom.removeChild(this.nodes[index]);
26886         this.updateIndexes(index);
26887     },
26888
26889     /**
26890      * Refresh an individual node.
26891      * @param {Number} index
26892      */
26893     refreshNode : function(index){
26894         this.onUpdate(this.store, this.store.getAt(index));
26895     },
26896
26897     updateIndexes : function(startIndex, endIndex){
26898         var ns = this.nodes;
26899         startIndex = startIndex || 0;
26900         endIndex = endIndex || ns.length - 1;
26901         for(var i = startIndex; i <= endIndex; i++){
26902             ns[i].nodeIndex = i;
26903         }
26904     },
26905
26906     /**
26907      * Changes the data store this view uses and refresh the view.
26908      * @param {Store} store
26909      */
26910     setStore : function(store, initial){
26911         if(!initial && this.store){
26912             this.store.un("datachanged", this.refresh);
26913             this.store.un("add", this.onAdd);
26914             this.store.un("remove", this.onRemove);
26915             this.store.un("update", this.onUpdate);
26916             this.store.un("clear", this.refresh);
26917             this.store.un("beforeload", this.onBeforeLoad);
26918             this.store.un("load", this.onLoad);
26919             this.store.un("loadexception", this.onLoad);
26920         }
26921         if(store){
26922           
26923             store.on("datachanged", this.refresh, this);
26924             store.on("add", this.onAdd, this);
26925             store.on("remove", this.onRemove, this);
26926             store.on("update", this.onUpdate, this);
26927             store.on("clear", this.refresh, this);
26928             store.on("beforeload", this.onBeforeLoad, this);
26929             store.on("load", this.onLoad, this);
26930             store.on("loadexception", this.onLoad, this);
26931         }
26932         
26933         if(store){
26934             this.refresh();
26935         }
26936     },
26937     /**
26938      * onbeforeLoad - masks the loading area.
26939      *
26940      */
26941     onBeforeLoad : function(store,opts)
26942     {
26943          //Roo.log('onBeforeLoad');   
26944         if (!opts.add) {
26945             this.el.update("");
26946         }
26947         this.el.mask(this.mask ? this.mask : "Loading" ); 
26948     },
26949     onLoad : function ()
26950     {
26951         this.el.unmask();
26952     },
26953     
26954
26955     /**
26956      * Returns the template node the passed child belongs to or null if it doesn't belong to one.
26957      * @param {HTMLElement} node
26958      * @return {HTMLElement} The template node
26959      */
26960     findItemFromChild : function(node){
26961         var el = this.dataName  ?
26962             this.el.child('.roo-tpl-' + this.dataName,true) :
26963             this.el.dom; 
26964         
26965         if(!node || node.parentNode == el){
26966                     return node;
26967             }
26968             var p = node.parentNode;
26969             while(p && p != el){
26970             if(p.parentNode == el){
26971                 return p;
26972             }
26973             p = p.parentNode;
26974         }
26975             return null;
26976     },
26977
26978     /** @ignore */
26979     onClick : function(e){
26980         var item = this.findItemFromChild(e.getTarget());
26981         if(item){
26982             var index = this.indexOf(item);
26983             if(this.onItemClick(item, index, e) !== false){
26984                 this.fireEvent("click", this, index, item, e);
26985             }
26986         }else{
26987             this.clearSelections();
26988         }
26989     },
26990
26991     /** @ignore */
26992     onContextMenu : function(e){
26993         var item = this.findItemFromChild(e.getTarget());
26994         if(item){
26995             this.fireEvent("contextmenu", this, this.indexOf(item), item, e);
26996         }
26997     },
26998
26999     /** @ignore */
27000     onDblClick : function(e){
27001         var item = this.findItemFromChild(e.getTarget());
27002         if(item){
27003             this.fireEvent("dblclick", this, this.indexOf(item), item, e);
27004         }
27005     },
27006
27007     onItemClick : function(item, index, e)
27008     {
27009         if(this.fireEvent("beforeclick", this, index, item, e) === false){
27010             return false;
27011         }
27012         if (this.toggleSelect) {
27013             var m = this.isSelected(item) ? 'unselect' : 'select';
27014             //Roo.log(m);
27015             var _t = this;
27016             _t[m](item, true, false);
27017             return true;
27018         }
27019         if(this.multiSelect || this.singleSelect){
27020             if(this.multiSelect && e.shiftKey && this.lastSelection){
27021                 this.select(this.getNodes(this.indexOf(this.lastSelection), index), false);
27022             }else{
27023                 this.select(item, this.multiSelect && e.ctrlKey);
27024                 this.lastSelection = item;
27025             }
27026             
27027             if(!this.tickable){
27028                 e.preventDefault();
27029             }
27030             
27031         }
27032         return true;
27033     },
27034
27035     /**
27036      * Get the number of selected nodes.
27037      * @return {Number}
27038      */
27039     getSelectionCount : function(){
27040         return this.selections.length;
27041     },
27042
27043     /**
27044      * Get the currently selected nodes.
27045      * @return {Array} An array of HTMLElements
27046      */
27047     getSelectedNodes : function(){
27048         return this.selections;
27049     },
27050
27051     /**
27052      * Get the indexes of the selected nodes.
27053      * @return {Array}
27054      */
27055     getSelectedIndexes : function(){
27056         var indexes = [], s = this.selections;
27057         for(var i = 0, len = s.length; i < len; i++){
27058             indexes.push(s[i].nodeIndex);
27059         }
27060         return indexes;
27061     },
27062
27063     /**
27064      * Clear all selections
27065      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange event
27066      */
27067     clearSelections : function(suppressEvent){
27068         if(this.nodes && (this.multiSelect || this.singleSelect) && this.selections.length > 0){
27069             this.cmp.elements = this.selections;
27070             this.cmp.removeClass(this.selectedClass);
27071             this.selections = [];
27072             if(!suppressEvent){
27073                 this.fireEvent("selectionchange", this, this.selections);
27074             }
27075         }
27076     },
27077
27078     /**
27079      * Returns true if the passed node is selected
27080      * @param {HTMLElement/Number} node The node or node index
27081      * @return {Boolean}
27082      */
27083     isSelected : function(node){
27084         var s = this.selections;
27085         if(s.length < 1){
27086             return false;
27087         }
27088         node = this.getNode(node);
27089         return s.indexOf(node) !== -1;
27090     },
27091
27092     /**
27093      * Selects nodes.
27094      * @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
27095      * @param {Boolean} keepExisting (optional) true to keep existing selections
27096      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange vent
27097      */
27098     select : function(nodeInfo, keepExisting, suppressEvent){
27099         if(nodeInfo instanceof Array){
27100             if(!keepExisting){
27101                 this.clearSelections(true);
27102             }
27103             for(var i = 0, len = nodeInfo.length; i < len; i++){
27104                 this.select(nodeInfo[i], true, true);
27105             }
27106             return;
27107         } 
27108         var node = this.getNode(nodeInfo);
27109         if(!node || this.isSelected(node)){
27110             return; // already selected.
27111         }
27112         if(!keepExisting){
27113             this.clearSelections(true);
27114         }
27115         
27116         if(this.fireEvent("beforeselect", this, node, this.selections) !== false){
27117             Roo.fly(node).addClass(this.selectedClass);
27118             this.selections.push(node);
27119             if(!suppressEvent){
27120                 this.fireEvent("selectionchange", this, this.selections);
27121             }
27122         }
27123         
27124         
27125     },
27126       /**
27127      * Unselects nodes.
27128      * @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
27129      * @param {Boolean} keepExisting (optional) true IGNORED (for campatibility with select)
27130      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange vent
27131      */
27132     unselect : function(nodeInfo, keepExisting, suppressEvent)
27133     {
27134         if(nodeInfo instanceof Array){
27135             Roo.each(this.selections, function(s) {
27136                 this.unselect(s, nodeInfo);
27137             }, this);
27138             return;
27139         }
27140         var node = this.getNode(nodeInfo);
27141         if(!node || !this.isSelected(node)){
27142             //Roo.log("not selected");
27143             return; // not selected.
27144         }
27145         // fireevent???
27146         var ns = [];
27147         Roo.each(this.selections, function(s) {
27148             if (s == node ) {
27149                 Roo.fly(node).removeClass(this.selectedClass);
27150
27151                 return;
27152             }
27153             ns.push(s);
27154         },this);
27155         
27156         this.selections= ns;
27157         this.fireEvent("selectionchange", this, this.selections);
27158     },
27159
27160     /**
27161      * Gets a template node.
27162      * @param {HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node or the id of a template node
27163      * @return {HTMLElement} The node or null if it wasn't found
27164      */
27165     getNode : function(nodeInfo){
27166         if(typeof nodeInfo == "string"){
27167             return document.getElementById(nodeInfo);
27168         }else if(typeof nodeInfo == "number"){
27169             return this.nodes[nodeInfo];
27170         }
27171         return nodeInfo;
27172     },
27173
27174     /**
27175      * Gets a range template nodes.
27176      * @param {Number} startIndex
27177      * @param {Number} endIndex
27178      * @return {Array} An array of nodes
27179      */
27180     getNodes : function(start, end){
27181         var ns = this.nodes;
27182         start = start || 0;
27183         end = typeof end == "undefined" ? ns.length - 1 : end;
27184         var nodes = [];
27185         if(start <= end){
27186             for(var i = start; i <= end; i++){
27187                 nodes.push(ns[i]);
27188             }
27189         } else{
27190             for(var i = start; i >= end; i--){
27191                 nodes.push(ns[i]);
27192             }
27193         }
27194         return nodes;
27195     },
27196
27197     /**
27198      * Finds the index of the passed node
27199      * @param {HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node or the id of a template node
27200      * @return {Number} The index of the node or -1
27201      */
27202     indexOf : function(node){
27203         node = this.getNode(node);
27204         if(typeof node.nodeIndex == "number"){
27205             return node.nodeIndex;
27206         }
27207         var ns = this.nodes;
27208         for(var i = 0, len = ns.length; i < len; i++){
27209             if(ns[i] == node){
27210                 return i;
27211             }
27212         }
27213         return -1;
27214     }
27215 });
27216 /*
27217  * Based on:
27218  * Ext JS Library 1.1.1
27219  * Copyright(c) 2006-2007, Ext JS, LLC.
27220  *
27221  * Originally Released Under LGPL - original licence link has changed is not relivant.
27222  *
27223  * Fork - LGPL
27224  * <script type="text/javascript">
27225  */
27226
27227 /**
27228  * @class Roo.JsonView
27229  * @extends Roo.View
27230  * Shortcut class to create a JSON + {@link Roo.UpdateManager} template view. Usage:
27231 <pre><code>
27232 var view = new Roo.JsonView({
27233     container: "my-element",
27234     tpl: '&lt;div id="{id}"&gt;{foo} - {bar}&lt;/div&gt;', // auto create template
27235     multiSelect: true, 
27236     jsonRoot: "data" 
27237 });
27238
27239 // listen for node click?
27240 view.on("click", function(vw, index, node, e){
27241     alert('Node "' + node.id + '" at index: ' + index + " was clicked.");
27242 });
27243
27244 // direct load of JSON data
27245 view.load("foobar.php");
27246
27247 // Example from my blog list
27248 var tpl = new Roo.Template(
27249     '&lt;div class="entry"&gt;' +
27250     '&lt;a class="entry-title" href="{link}"&gt;{title}&lt;/a&gt;' +
27251     "&lt;h4&gt;{date} by {author} | {comments} Comments&lt;/h4&gt;{description}" +
27252     "&lt;/div&gt;&lt;hr /&gt;"
27253 );
27254
27255 var moreView = new Roo.JsonView({
27256     container :  "entry-list", 
27257     template : tpl,
27258     jsonRoot: "posts"
27259 });
27260 moreView.on("beforerender", this.sortEntries, this);
27261 moreView.load({
27262     url: "/blog/get-posts.php",
27263     params: "allposts=true",
27264     text: "Loading Blog Entries..."
27265 });
27266 </code></pre>
27267
27268 * Note: old code is supported with arguments : (container, template, config)
27269
27270
27271  * @constructor
27272  * Create a new JsonView
27273  * 
27274  * @param {Object} config The config object
27275  * 
27276  */
27277 Roo.JsonView = function(config, depreciated_tpl, depreciated_config){
27278     
27279     
27280     Roo.JsonView.superclass.constructor.call(this, config, depreciated_tpl, depreciated_config);
27281
27282     var um = this.el.getUpdateManager();
27283     um.setRenderer(this);
27284     um.on("update", this.onLoad, this);
27285     um.on("failure", this.onLoadException, this);
27286
27287     /**
27288      * @event beforerender
27289      * Fires before rendering of the downloaded JSON data.
27290      * @param {Roo.JsonView} this
27291      * @param {Object} data The JSON data loaded
27292      */
27293     /**
27294      * @event load
27295      * Fires when data is loaded.
27296      * @param {Roo.JsonView} this
27297      * @param {Object} data The JSON data loaded
27298      * @param {Object} response The raw Connect response object
27299      */
27300     /**
27301      * @event loadexception
27302      * Fires when loading fails.
27303      * @param {Roo.JsonView} this
27304      * @param {Object} response The raw Connect response object
27305      */
27306     this.addEvents({
27307         'beforerender' : true,
27308         'load' : true,
27309         'loadexception' : true
27310     });
27311 };
27312 Roo.extend(Roo.JsonView, Roo.View, {
27313     /**
27314      * @type {String} The root property in the loaded JSON object that contains the data
27315      */
27316     jsonRoot : "",
27317
27318     /**
27319      * Refreshes the view.
27320      */
27321     refresh : function(){
27322         this.clearSelections();
27323         this.el.update("");
27324         var html = [];
27325         var o = this.jsonData;
27326         if(o && o.length > 0){
27327             for(var i = 0, len = o.length; i < len; i++){
27328                 var data = this.prepareData(o[i], i, o);
27329                 html[html.length] = this.tpl.apply(data);
27330             }
27331         }else{
27332             html.push(this.emptyText);
27333         }
27334         this.el.update(html.join(""));
27335         this.nodes = this.el.dom.childNodes;
27336         this.updateIndexes(0);
27337     },
27338
27339     /**
27340      * 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.
27341      * @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:
27342      <pre><code>
27343      view.load({
27344          url: "your-url.php",
27345          params: {param1: "foo", param2: "bar"}, // or a URL encoded string
27346          callback: yourFunction,
27347          scope: yourObject, //(optional scope)
27348          discardUrl: false,
27349          nocache: false,
27350          text: "Loading...",
27351          timeout: 30,
27352          scripts: false
27353      });
27354      </code></pre>
27355      * The only required property is <i>url</i>. The optional properties <i>nocache</i>, <i>text</i> and <i>scripts</i>
27356      * 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.
27357      * @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}
27358      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
27359      * @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.
27360      */
27361     load : function(){
27362         var um = this.el.getUpdateManager();
27363         um.update.apply(um, arguments);
27364     },
27365
27366     // note - render is a standard framework call...
27367     // using it for the response is really flaky... - it's called by UpdateManager normally, except when called by the XComponent/addXtype.
27368     render : function(el, response){
27369         
27370         this.clearSelections();
27371         this.el.update("");
27372         var o;
27373         try{
27374             if (response != '') {
27375                 o = Roo.util.JSON.decode(response.responseText);
27376                 if(this.jsonRoot){
27377                     
27378                     o = o[this.jsonRoot];
27379                 }
27380             }
27381         } catch(e){
27382         }
27383         /**
27384          * The current JSON data or null
27385          */
27386         this.jsonData = o;
27387         this.beforeRender();
27388         this.refresh();
27389     },
27390
27391 /**
27392  * Get the number of records in the current JSON dataset
27393  * @return {Number}
27394  */
27395     getCount : function(){
27396         return this.jsonData ? this.jsonData.length : 0;
27397     },
27398
27399 /**
27400  * Returns the JSON object for the specified node(s)
27401  * @param {HTMLElement/Array} node The node or an array of nodes
27402  * @return {Object/Array} If you pass in an array, you get an array back, otherwise
27403  * you get the JSON object for the node
27404  */
27405     getNodeData : function(node){
27406         if(node instanceof Array){
27407             var data = [];
27408             for(var i = 0, len = node.length; i < len; i++){
27409                 data.push(this.getNodeData(node[i]));
27410             }
27411             return data;
27412         }
27413         return this.jsonData[this.indexOf(node)] || null;
27414     },
27415
27416     beforeRender : function(){
27417         this.snapshot = this.jsonData;
27418         if(this.sortInfo){
27419             this.sort.apply(this, this.sortInfo);
27420         }
27421         this.fireEvent("beforerender", this, this.jsonData);
27422     },
27423
27424     onLoad : function(el, o){
27425         this.fireEvent("load", this, this.jsonData, o);
27426     },
27427
27428     onLoadException : function(el, o){
27429         this.fireEvent("loadexception", this, o);
27430     },
27431
27432 /**
27433  * Filter the data by a specific property.
27434  * @param {String} property A property on your JSON objects
27435  * @param {String/RegExp} value Either string that the property values
27436  * should start with, or a RegExp to test against the property
27437  */
27438     filter : function(property, value){
27439         if(this.jsonData){
27440             var data = [];
27441             var ss = this.snapshot;
27442             if(typeof value == "string"){
27443                 var vlen = value.length;
27444                 if(vlen == 0){
27445                     this.clearFilter();
27446                     return;
27447                 }
27448                 value = value.toLowerCase();
27449                 for(var i = 0, len = ss.length; i < len; i++){
27450                     var o = ss[i];
27451                     if(o[property].substr(0, vlen).toLowerCase() == value){
27452                         data.push(o);
27453                     }
27454                 }
27455             } else if(value.exec){ // regex?
27456                 for(var i = 0, len = ss.length; i < len; i++){
27457                     var o = ss[i];
27458                     if(value.test(o[property])){
27459                         data.push(o);
27460                     }
27461                 }
27462             } else{
27463                 return;
27464             }
27465             this.jsonData = data;
27466             this.refresh();
27467         }
27468     },
27469
27470 /**
27471  * Filter by a function. The passed function will be called with each
27472  * object in the current dataset. If the function returns true the value is kept,
27473  * otherwise it is filtered.
27474  * @param {Function} fn
27475  * @param {Object} scope (optional) The scope of the function (defaults to this JsonView)
27476  */
27477     filterBy : function(fn, scope){
27478         if(this.jsonData){
27479             var data = [];
27480             var ss = this.snapshot;
27481             for(var i = 0, len = ss.length; i < len; i++){
27482                 var o = ss[i];
27483                 if(fn.call(scope || this, o)){
27484                     data.push(o);
27485                 }
27486             }
27487             this.jsonData = data;
27488             this.refresh();
27489         }
27490     },
27491
27492 /**
27493  * Clears the current filter.
27494  */
27495     clearFilter : function(){
27496         if(this.snapshot && this.jsonData != this.snapshot){
27497             this.jsonData = this.snapshot;
27498             this.refresh();
27499         }
27500     },
27501
27502
27503 /**
27504  * Sorts the data for this view and refreshes it.
27505  * @param {String} property A property on your JSON objects to sort on
27506  * @param {String} direction (optional) "desc" or "asc" (defaults to "asc")
27507  * @param {Function} sortType (optional) A function to call to convert the data to a sortable value.
27508  */
27509     sort : function(property, dir, sortType){
27510         this.sortInfo = Array.prototype.slice.call(arguments, 0);
27511         if(this.jsonData){
27512             var p = property;
27513             var dsc = dir && dir.toLowerCase() == "desc";
27514             var f = function(o1, o2){
27515                 var v1 = sortType ? sortType(o1[p]) : o1[p];
27516                 var v2 = sortType ? sortType(o2[p]) : o2[p];
27517                 ;
27518                 if(v1 < v2){
27519                     return dsc ? +1 : -1;
27520                 } else if(v1 > v2){
27521                     return dsc ? -1 : +1;
27522                 } else{
27523                     return 0;
27524                 }
27525             };
27526             this.jsonData.sort(f);
27527             this.refresh();
27528             if(this.jsonData != this.snapshot){
27529                 this.snapshot.sort(f);
27530             }
27531         }
27532     }
27533 });/*
27534  * Based on:
27535  * Ext JS Library 1.1.1
27536  * Copyright(c) 2006-2007, Ext JS, LLC.
27537  *
27538  * Originally Released Under LGPL - original licence link has changed is not relivant.
27539  *
27540  * Fork - LGPL
27541  * <script type="text/javascript">
27542  */
27543  
27544
27545 /**
27546  * @class Roo.ColorPalette
27547  * @extends Roo.Component
27548  * Simple color palette class for choosing colors.  The palette can be rendered to any container.<br />
27549  * Here's an example of typical usage:
27550  * <pre><code>
27551 var cp = new Roo.ColorPalette({value:'993300'});  // initial selected color
27552 cp.render('my-div');
27553
27554 cp.on('select', function(palette, selColor){
27555     // do something with selColor
27556 });
27557 </code></pre>
27558  * @constructor
27559  * Create a new ColorPalette
27560  * @param {Object} config The config object
27561  */
27562 Roo.ColorPalette = function(config){
27563     Roo.ColorPalette.superclass.constructor.call(this, config);
27564     this.addEvents({
27565         /**
27566              * @event select
27567              * Fires when a color is selected
27568              * @param {ColorPalette} this
27569              * @param {String} color The 6-digit color hex code (without the # symbol)
27570              */
27571         select: true
27572     });
27573
27574     if(this.handler){
27575         this.on("select", this.handler, this.scope, true);
27576     }
27577 };
27578 Roo.extend(Roo.ColorPalette, Roo.Component, {
27579     /**
27580      * @cfg {String} itemCls
27581      * The CSS class to apply to the containing element (defaults to "x-color-palette")
27582      */
27583     itemCls : "x-color-palette",
27584     /**
27585      * @cfg {String} value
27586      * The initial color to highlight (should be a valid 6-digit color hex code without the # symbol).  Note that
27587      * the hex codes are case-sensitive.
27588      */
27589     value : null,
27590     clickEvent:'click',
27591     // private
27592     ctype: "Roo.ColorPalette",
27593
27594     /**
27595      * @cfg {Boolean} allowReselect If set to true then reselecting a color that is already selected fires the selection event
27596      */
27597     allowReselect : false,
27598
27599     /**
27600      * <p>An array of 6-digit color hex code strings (without the # symbol).  This array can contain any number
27601      * of colors, and each hex code should be unique.  The width of the palette is controlled via CSS by adjusting
27602      * the width property of the 'x-color-palette' class (or assigning a custom class), so you can balance the number
27603      * of colors with the width setting until the box is symmetrical.</p>
27604      * <p>You can override individual colors if needed:</p>
27605      * <pre><code>
27606 var cp = new Roo.ColorPalette();
27607 cp.colors[0] = "FF0000";  // change the first box to red
27608 </code></pre>
27609
27610 Or you can provide a custom array of your own for complete control:
27611 <pre><code>
27612 var cp = new Roo.ColorPalette();
27613 cp.colors = ["000000", "993300", "333300"];
27614 </code></pre>
27615      * @type Array
27616      */
27617     colors : [
27618         "000000", "993300", "333300", "003300", "003366", "000080", "333399", "333333",
27619         "800000", "FF6600", "808000", "008000", "008080", "0000FF", "666699", "808080",
27620         "FF0000", "FF9900", "99CC00", "339966", "33CCCC", "3366FF", "800080", "969696",
27621         "FF00FF", "FFCC00", "FFFF00", "00FF00", "00FFFF", "00CCFF", "993366", "C0C0C0",
27622         "FF99CC", "FFCC99", "FFFF99", "CCFFCC", "CCFFFF", "99CCFF", "CC99FF", "FFFFFF"
27623     ],
27624
27625     // private
27626     onRender : function(container, position){
27627         var t = new Roo.MasterTemplate(
27628             '<tpl><a href="#" class="color-{0}" hidefocus="on"><em><span style="background:#{0}" unselectable="on">&#160;</span></em></a></tpl>'
27629         );
27630         var c = this.colors;
27631         for(var i = 0, len = c.length; i < len; i++){
27632             t.add([c[i]]);
27633         }
27634         var el = document.createElement("div");
27635         el.className = this.itemCls;
27636         t.overwrite(el);
27637         container.dom.insertBefore(el, position);
27638         this.el = Roo.get(el);
27639         this.el.on(this.clickEvent, this.handleClick,  this, {delegate: "a"});
27640         if(this.clickEvent != 'click'){
27641             this.el.on('click', Roo.emptyFn,  this, {delegate: "a", preventDefault:true});
27642         }
27643     },
27644
27645     // private
27646     afterRender : function(){
27647         Roo.ColorPalette.superclass.afterRender.call(this);
27648         if(this.value){
27649             var s = this.value;
27650             this.value = null;
27651             this.select(s);
27652         }
27653     },
27654
27655     // private
27656     handleClick : function(e, t){
27657         e.preventDefault();
27658         if(!this.disabled){
27659             var c = t.className.match(/(?:^|\s)color-(.{6})(?:\s|$)/)[1];
27660             this.select(c.toUpperCase());
27661         }
27662     },
27663
27664     /**
27665      * Selects the specified color in the palette (fires the select event)
27666      * @param {String} color A valid 6-digit color hex code (# will be stripped if included)
27667      */
27668     select : function(color){
27669         color = color.replace("#", "");
27670         if(color != this.value || this.allowReselect){
27671             var el = this.el;
27672             if(this.value){
27673                 el.child("a.color-"+this.value).removeClass("x-color-palette-sel");
27674             }
27675             el.child("a.color-"+color).addClass("x-color-palette-sel");
27676             this.value = color;
27677             this.fireEvent("select", this, color);
27678         }
27679     }
27680 });/*
27681  * Based on:
27682  * Ext JS Library 1.1.1
27683  * Copyright(c) 2006-2007, Ext JS, LLC.
27684  *
27685  * Originally Released Under LGPL - original licence link has changed is not relivant.
27686  *
27687  * Fork - LGPL
27688  * <script type="text/javascript">
27689  */
27690  
27691 /**
27692  * @class Roo.DatePicker
27693  * @extends Roo.Component
27694  * Simple date picker class.
27695  * @constructor
27696  * Create a new DatePicker
27697  * @param {Object} config The config object
27698  */
27699 Roo.DatePicker = function(config){
27700     Roo.DatePicker.superclass.constructor.call(this, config);
27701
27702     this.value = config && config.value ?
27703                  config.value.clearTime() : new Date().clearTime();
27704
27705     this.addEvents({
27706         /**
27707              * @event select
27708              * Fires when a date is selected
27709              * @param {DatePicker} this
27710              * @param {Date} date The selected date
27711              */
27712         'select': true,
27713         /**
27714              * @event monthchange
27715              * Fires when the displayed month changes 
27716              * @param {DatePicker} this
27717              * @param {Date} date The selected month
27718              */
27719         'monthchange': true
27720     });
27721
27722     if(this.handler){
27723         this.on("select", this.handler,  this.scope || this);
27724     }
27725     // build the disabledDatesRE
27726     if(!this.disabledDatesRE && this.disabledDates){
27727         var dd = this.disabledDates;
27728         var re = "(?:";
27729         for(var i = 0; i < dd.length; i++){
27730             re += dd[i];
27731             if(i != dd.length-1) {
27732                 re += "|";
27733             }
27734         }
27735         this.disabledDatesRE = new RegExp(re + ")");
27736     }
27737 };
27738
27739 Roo.extend(Roo.DatePicker, Roo.Component, {
27740     /**
27741      * @cfg {String} todayText
27742      * The text to display on the button that selects the current date (defaults to "Today")
27743      */
27744     todayText : "Today",
27745     /**
27746      * @cfg {String} okText
27747      * The text to display on the ok button
27748      */
27749     okText : "&#160;OK&#160;", // &#160; to give the user extra clicking room
27750     /**
27751      * @cfg {String} cancelText
27752      * The text to display on the cancel button
27753      */
27754     cancelText : "Cancel",
27755     /**
27756      * @cfg {String} todayTip
27757      * The tooltip to display for the button that selects the current date (defaults to "{current date} (Spacebar)")
27758      */
27759     todayTip : "{0} (Spacebar)",
27760     /**
27761      * @cfg {Date} minDate
27762      * Minimum allowable date (JavaScript date object, defaults to null)
27763      */
27764     minDate : null,
27765     /**
27766      * @cfg {Date} maxDate
27767      * Maximum allowable date (JavaScript date object, defaults to null)
27768      */
27769     maxDate : null,
27770     /**
27771      * @cfg {String} minText
27772      * The error text to display if the minDate validation fails (defaults to "This date is before the minimum date")
27773      */
27774     minText : "This date is before the minimum date",
27775     /**
27776      * @cfg {String} maxText
27777      * The error text to display if the maxDate validation fails (defaults to "This date is after the maximum date")
27778      */
27779     maxText : "This date is after the maximum date",
27780     /**
27781      * @cfg {String} format
27782      * The default date format string which can be overriden for localization support.  The format must be
27783      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
27784      */
27785     format : "m/d/y",
27786     /**
27787      * @cfg {Array} disabledDays
27788      * An array of days to disable, 0-based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
27789      */
27790     disabledDays : null,
27791     /**
27792      * @cfg {String} disabledDaysText
27793      * The tooltip to display when the date falls on a disabled day (defaults to "")
27794      */
27795     disabledDaysText : "",
27796     /**
27797      * @cfg {RegExp} disabledDatesRE
27798      * JavaScript regular expression used to disable a pattern of dates (defaults to null)
27799      */
27800     disabledDatesRE : null,
27801     /**
27802      * @cfg {String} disabledDatesText
27803      * The tooltip text to display when the date falls on a disabled date (defaults to "")
27804      */
27805     disabledDatesText : "",
27806     /**
27807      * @cfg {Boolean} constrainToViewport
27808      * True to constrain the date picker to the viewport (defaults to true)
27809      */
27810     constrainToViewport : true,
27811     /**
27812      * @cfg {Array} monthNames
27813      * An array of textual month names which can be overriden for localization support (defaults to Date.monthNames)
27814      */
27815     monthNames : Date.monthNames,
27816     /**
27817      * @cfg {Array} dayNames
27818      * An array of textual day names which can be overriden for localization support (defaults to Date.dayNames)
27819      */
27820     dayNames : Date.dayNames,
27821     /**
27822      * @cfg {String} nextText
27823      * The next month navigation button tooltip (defaults to 'Next Month (Control+Right)')
27824      */
27825     nextText: 'Next Month (Control+Right)',
27826     /**
27827      * @cfg {String} prevText
27828      * The previous month navigation button tooltip (defaults to 'Previous Month (Control+Left)')
27829      */
27830     prevText: 'Previous Month (Control+Left)',
27831     /**
27832      * @cfg {String} monthYearText
27833      * The header month selector tooltip (defaults to 'Choose a month (Control+Up/Down to move years)')
27834      */
27835     monthYearText: 'Choose a month (Control+Up/Down to move years)',
27836     /**
27837      * @cfg {Number} startDay
27838      * Day index at which the week should begin, 0-based (defaults to 0, which is Sunday)
27839      */
27840     startDay : 0,
27841     /**
27842      * @cfg {Bool} showClear
27843      * Show a clear button (usefull for date form elements that can be blank.)
27844      */
27845     
27846     showClear: false,
27847     
27848     /**
27849      * Sets the value of the date field
27850      * @param {Date} value The date to set
27851      */
27852     setValue : function(value){
27853         var old = this.value;
27854         
27855         if (typeof(value) == 'string') {
27856          
27857             value = Date.parseDate(value, this.format);
27858         }
27859         if (!value) {
27860             value = new Date();
27861         }
27862         
27863         this.value = value.clearTime(true);
27864         if(this.el){
27865             this.update(this.value);
27866         }
27867     },
27868
27869     /**
27870      * Gets the current selected value of the date field
27871      * @return {Date} The selected date
27872      */
27873     getValue : function(){
27874         return this.value;
27875     },
27876
27877     // private
27878     focus : function(){
27879         if(this.el){
27880             this.update(this.activeDate);
27881         }
27882     },
27883
27884     // privateval
27885     onRender : function(container, position){
27886         
27887         var m = [
27888              '<table cellspacing="0">',
27889                 '<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>',
27890                 '<tr><td colspan="3"><table class="x-date-inner" cellspacing="0"><thead><tr>'];
27891         var dn = this.dayNames;
27892         for(var i = 0; i < 7; i++){
27893             var d = this.startDay+i;
27894             if(d > 6){
27895                 d = d-7;
27896             }
27897             m.push("<th><span>", dn[d].substr(0,1), "</span></th>");
27898         }
27899         m[m.length] = "</tr></thead><tbody><tr>";
27900         for(var i = 0; i < 42; i++) {
27901             if(i % 7 == 0 && i != 0){
27902                 m[m.length] = "</tr><tr>";
27903             }
27904             m[m.length] = '<td><a href="#" hidefocus="on" class="x-date-date" tabIndex="1"><em><span></span></em></a></td>';
27905         }
27906         m[m.length] = '</tr></tbody></table></td></tr><tr>'+
27907             '<td colspan="3" class="x-date-bottom" align="center"></td></tr></table><div class="x-date-mp"></div>';
27908
27909         var el = document.createElement("div");
27910         el.className = "x-date-picker";
27911         el.innerHTML = m.join("");
27912
27913         container.dom.insertBefore(el, position);
27914
27915         this.el = Roo.get(el);
27916         this.eventEl = Roo.get(el.firstChild);
27917
27918         new Roo.util.ClickRepeater(this.el.child("td.x-date-left a"), {
27919             handler: this.showPrevMonth,
27920             scope: this,
27921             preventDefault:true,
27922             stopDefault:true
27923         });
27924
27925         new Roo.util.ClickRepeater(this.el.child("td.x-date-right a"), {
27926             handler: this.showNextMonth,
27927             scope: this,
27928             preventDefault:true,
27929             stopDefault:true
27930         });
27931
27932         this.eventEl.on("mousewheel", this.handleMouseWheel,  this);
27933
27934         this.monthPicker = this.el.down('div.x-date-mp');
27935         this.monthPicker.enableDisplayMode('block');
27936         
27937         var kn = new Roo.KeyNav(this.eventEl, {
27938             "left" : function(e){
27939                 e.ctrlKey ?
27940                     this.showPrevMonth() :
27941                     this.update(this.activeDate.add("d", -1));
27942             },
27943
27944             "right" : function(e){
27945                 e.ctrlKey ?
27946                     this.showNextMonth() :
27947                     this.update(this.activeDate.add("d", 1));
27948             },
27949
27950             "up" : function(e){
27951                 e.ctrlKey ?
27952                     this.showNextYear() :
27953                     this.update(this.activeDate.add("d", -7));
27954             },
27955
27956             "down" : function(e){
27957                 e.ctrlKey ?
27958                     this.showPrevYear() :
27959                     this.update(this.activeDate.add("d", 7));
27960             },
27961
27962             "pageUp" : function(e){
27963                 this.showNextMonth();
27964             },
27965
27966             "pageDown" : function(e){
27967                 this.showPrevMonth();
27968             },
27969
27970             "enter" : function(e){
27971                 e.stopPropagation();
27972                 return true;
27973             },
27974
27975             scope : this
27976         });
27977
27978         this.eventEl.on("click", this.handleDateClick,  this, {delegate: "a.x-date-date"});
27979
27980         this.eventEl.addKeyListener(Roo.EventObject.SPACE, this.selectToday,  this);
27981
27982         this.el.unselectable();
27983         
27984         this.cells = this.el.select("table.x-date-inner tbody td");
27985         this.textNodes = this.el.query("table.x-date-inner tbody span");
27986
27987         this.mbtn = new Roo.Button(this.el.child("td.x-date-middle", true), {
27988             text: "&#160;",
27989             tooltip: this.monthYearText
27990         });
27991
27992         this.mbtn.on('click', this.showMonthPicker, this);
27993         this.mbtn.el.child(this.mbtn.menuClassTarget).addClass("x-btn-with-menu");
27994
27995
27996         var today = (new Date()).dateFormat(this.format);
27997         
27998         var baseTb = new Roo.Toolbar(this.el.child("td.x-date-bottom", true));
27999         if (this.showClear) {
28000             baseTb.add( new Roo.Toolbar.Fill());
28001         }
28002         baseTb.add({
28003             text: String.format(this.todayText, today),
28004             tooltip: String.format(this.todayTip, today),
28005             handler: this.selectToday,
28006             scope: this
28007         });
28008         
28009         //var todayBtn = new Roo.Button(this.el.child("td.x-date-bottom", true), {
28010             
28011         //});
28012         if (this.showClear) {
28013             
28014             baseTb.add( new Roo.Toolbar.Fill());
28015             baseTb.add({
28016                 text: '&#160;',
28017                 cls: 'x-btn-icon x-btn-clear',
28018                 handler: function() {
28019                     //this.value = '';
28020                     this.fireEvent("select", this, '');
28021                 },
28022                 scope: this
28023             });
28024         }
28025         
28026         
28027         if(Roo.isIE){
28028             this.el.repaint();
28029         }
28030         this.update(this.value);
28031     },
28032
28033     createMonthPicker : function(){
28034         if(!this.monthPicker.dom.firstChild){
28035             var buf = ['<table border="0" cellspacing="0">'];
28036             for(var i = 0; i < 6; i++){
28037                 buf.push(
28038                     '<tr><td class="x-date-mp-month"><a href="#">', this.monthNames[i].substr(0, 3), '</a></td>',
28039                     '<td class="x-date-mp-month x-date-mp-sep"><a href="#">', this.monthNames[i+6].substr(0, 3), '</a></td>',
28040                     i == 0 ?
28041                     '<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>' :
28042                     '<td class="x-date-mp-year"><a href="#"></a></td><td class="x-date-mp-year"><a href="#"></a></td></tr>'
28043                 );
28044             }
28045             buf.push(
28046                 '<tr class="x-date-mp-btns"><td colspan="4"><button type="button" class="x-date-mp-ok">',
28047                     this.okText,
28048                     '</button><button type="button" class="x-date-mp-cancel">',
28049                     this.cancelText,
28050                     '</button></td></tr>',
28051                 '</table>'
28052             );
28053             this.monthPicker.update(buf.join(''));
28054             this.monthPicker.on('click', this.onMonthClick, this);
28055             this.monthPicker.on('dblclick', this.onMonthDblClick, this);
28056
28057             this.mpMonths = this.monthPicker.select('td.x-date-mp-month');
28058             this.mpYears = this.monthPicker.select('td.x-date-mp-year');
28059
28060             this.mpMonths.each(function(m, a, i){
28061                 i += 1;
28062                 if((i%2) == 0){
28063                     m.dom.xmonth = 5 + Math.round(i * .5);
28064                 }else{
28065                     m.dom.xmonth = Math.round((i-1) * .5);
28066                 }
28067             });
28068         }
28069     },
28070
28071     showMonthPicker : function(){
28072         this.createMonthPicker();
28073         var size = this.el.getSize();
28074         this.monthPicker.setSize(size);
28075         this.monthPicker.child('table').setSize(size);
28076
28077         this.mpSelMonth = (this.activeDate || this.value).getMonth();
28078         this.updateMPMonth(this.mpSelMonth);
28079         this.mpSelYear = (this.activeDate || this.value).getFullYear();
28080         this.updateMPYear(this.mpSelYear);
28081
28082         this.monthPicker.slideIn('t', {duration:.2});
28083     },
28084
28085     updateMPYear : function(y){
28086         this.mpyear = y;
28087         var ys = this.mpYears.elements;
28088         for(var i = 1; i <= 10; i++){
28089             var td = ys[i-1], y2;
28090             if((i%2) == 0){
28091                 y2 = y + Math.round(i * .5);
28092                 td.firstChild.innerHTML = y2;
28093                 td.xyear = y2;
28094             }else{
28095                 y2 = y - (5-Math.round(i * .5));
28096                 td.firstChild.innerHTML = y2;
28097                 td.xyear = y2;
28098             }
28099             this.mpYears.item(i-1)[y2 == this.mpSelYear ? 'addClass' : 'removeClass']('x-date-mp-sel');
28100         }
28101     },
28102
28103     updateMPMonth : function(sm){
28104         this.mpMonths.each(function(m, a, i){
28105             m[m.dom.xmonth == sm ? 'addClass' : 'removeClass']('x-date-mp-sel');
28106         });
28107     },
28108
28109     selectMPMonth: function(m){
28110         
28111     },
28112
28113     onMonthClick : function(e, t){
28114         e.stopEvent();
28115         var el = new Roo.Element(t), pn;
28116         if(el.is('button.x-date-mp-cancel')){
28117             this.hideMonthPicker();
28118         }
28119         else if(el.is('button.x-date-mp-ok')){
28120             this.update(new Date(this.mpSelYear, this.mpSelMonth, (this.activeDate || this.value).getDate()));
28121             this.hideMonthPicker();
28122         }
28123         else if(pn = el.up('td.x-date-mp-month', 2)){
28124             this.mpMonths.removeClass('x-date-mp-sel');
28125             pn.addClass('x-date-mp-sel');
28126             this.mpSelMonth = pn.dom.xmonth;
28127         }
28128         else if(pn = el.up('td.x-date-mp-year', 2)){
28129             this.mpYears.removeClass('x-date-mp-sel');
28130             pn.addClass('x-date-mp-sel');
28131             this.mpSelYear = pn.dom.xyear;
28132         }
28133         else if(el.is('a.x-date-mp-prev')){
28134             this.updateMPYear(this.mpyear-10);
28135         }
28136         else if(el.is('a.x-date-mp-next')){
28137             this.updateMPYear(this.mpyear+10);
28138         }
28139     },
28140
28141     onMonthDblClick : function(e, t){
28142         e.stopEvent();
28143         var el = new Roo.Element(t), pn;
28144         if(pn = el.up('td.x-date-mp-month', 2)){
28145             this.update(new Date(this.mpSelYear, pn.dom.xmonth, (this.activeDate || this.value).getDate()));
28146             this.hideMonthPicker();
28147         }
28148         else if(pn = el.up('td.x-date-mp-year', 2)){
28149             this.update(new Date(pn.dom.xyear, this.mpSelMonth, (this.activeDate || this.value).getDate()));
28150             this.hideMonthPicker();
28151         }
28152     },
28153
28154     hideMonthPicker : function(disableAnim){
28155         if(this.monthPicker){
28156             if(disableAnim === true){
28157                 this.monthPicker.hide();
28158             }else{
28159                 this.monthPicker.slideOut('t', {duration:.2});
28160             }
28161         }
28162     },
28163
28164     // private
28165     showPrevMonth : function(e){
28166         this.update(this.activeDate.add("mo", -1));
28167     },
28168
28169     // private
28170     showNextMonth : function(e){
28171         this.update(this.activeDate.add("mo", 1));
28172     },
28173
28174     // private
28175     showPrevYear : function(){
28176         this.update(this.activeDate.add("y", -1));
28177     },
28178
28179     // private
28180     showNextYear : function(){
28181         this.update(this.activeDate.add("y", 1));
28182     },
28183
28184     // private
28185     handleMouseWheel : function(e){
28186         var delta = e.getWheelDelta();
28187         if(delta > 0){
28188             this.showPrevMonth();
28189             e.stopEvent();
28190         } else if(delta < 0){
28191             this.showNextMonth();
28192             e.stopEvent();
28193         }
28194     },
28195
28196     // private
28197     handleDateClick : function(e, t){
28198         e.stopEvent();
28199         if(t.dateValue && !Roo.fly(t.parentNode).hasClass("x-date-disabled")){
28200             this.setValue(new Date(t.dateValue));
28201             this.fireEvent("select", this, this.value);
28202         }
28203     },
28204
28205     // private
28206     selectToday : function(){
28207         this.setValue(new Date().clearTime());
28208         this.fireEvent("select", this, this.value);
28209     },
28210
28211     // private
28212     update : function(date)
28213     {
28214         var vd = this.activeDate;
28215         this.activeDate = date;
28216         if(vd && this.el){
28217             var t = date.getTime();
28218             if(vd.getMonth() == date.getMonth() && vd.getFullYear() == date.getFullYear()){
28219                 this.cells.removeClass("x-date-selected");
28220                 this.cells.each(function(c){
28221                    if(c.dom.firstChild.dateValue == t){
28222                        c.addClass("x-date-selected");
28223                        setTimeout(function(){
28224                             try{c.dom.firstChild.focus();}catch(e){}
28225                        }, 50);
28226                        return false;
28227                    }
28228                 });
28229                 return;
28230             }
28231         }
28232         
28233         var days = date.getDaysInMonth();
28234         var firstOfMonth = date.getFirstDateOfMonth();
28235         var startingPos = firstOfMonth.getDay()-this.startDay;
28236
28237         if(startingPos <= this.startDay){
28238             startingPos += 7;
28239         }
28240
28241         var pm = date.add("mo", -1);
28242         var prevStart = pm.getDaysInMonth()-startingPos;
28243
28244         var cells = this.cells.elements;
28245         var textEls = this.textNodes;
28246         days += startingPos;
28247
28248         // convert everything to numbers so it's fast
28249         var day = 86400000;
28250         var d = (new Date(pm.getFullYear(), pm.getMonth(), prevStart)).clearTime();
28251         var today = new Date().clearTime().getTime();
28252         var sel = date.clearTime().getTime();
28253         var min = this.minDate ? this.minDate.clearTime() : Number.NEGATIVE_INFINITY;
28254         var max = this.maxDate ? this.maxDate.clearTime() : Number.POSITIVE_INFINITY;
28255         var ddMatch = this.disabledDatesRE;
28256         var ddText = this.disabledDatesText;
28257         var ddays = this.disabledDays ? this.disabledDays.join("") : false;
28258         var ddaysText = this.disabledDaysText;
28259         var format = this.format;
28260
28261         var setCellClass = function(cal, cell){
28262             cell.title = "";
28263             var t = d.getTime();
28264             cell.firstChild.dateValue = t;
28265             if(t == today){
28266                 cell.className += " x-date-today";
28267                 cell.title = cal.todayText;
28268             }
28269             if(t == sel){
28270                 cell.className += " x-date-selected";
28271                 setTimeout(function(){
28272                     try{cell.firstChild.focus();}catch(e){}
28273                 }, 50);
28274             }
28275             // disabling
28276             if(t < min) {
28277                 cell.className = " x-date-disabled";
28278                 cell.title = cal.minText;
28279                 return;
28280             }
28281             if(t > max) {
28282                 cell.className = " x-date-disabled";
28283                 cell.title = cal.maxText;
28284                 return;
28285             }
28286             if(ddays){
28287                 if(ddays.indexOf(d.getDay()) != -1){
28288                     cell.title = ddaysText;
28289                     cell.className = " x-date-disabled";
28290                 }
28291             }
28292             if(ddMatch && format){
28293                 var fvalue = d.dateFormat(format);
28294                 if(ddMatch.test(fvalue)){
28295                     cell.title = ddText.replace("%0", fvalue);
28296                     cell.className = " x-date-disabled";
28297                 }
28298             }
28299         };
28300
28301         var i = 0;
28302         for(; i < startingPos; i++) {
28303             textEls[i].innerHTML = (++prevStart);
28304             d.setDate(d.getDate()+1);
28305             cells[i].className = "x-date-prevday";
28306             setCellClass(this, cells[i]);
28307         }
28308         for(; i < days; i++){
28309             intDay = i - startingPos + 1;
28310             textEls[i].innerHTML = (intDay);
28311             d.setDate(d.getDate()+1);
28312             cells[i].className = "x-date-active";
28313             setCellClass(this, cells[i]);
28314         }
28315         var extraDays = 0;
28316         for(; i < 42; i++) {
28317              textEls[i].innerHTML = (++extraDays);
28318              d.setDate(d.getDate()+1);
28319              cells[i].className = "x-date-nextday";
28320              setCellClass(this, cells[i]);
28321         }
28322
28323         this.mbtn.setText(this.monthNames[date.getMonth()] + " " + date.getFullYear());
28324         this.fireEvent('monthchange', this, date);
28325         
28326         if(!this.internalRender){
28327             var main = this.el.dom.firstChild;
28328             var w = main.offsetWidth;
28329             this.el.setWidth(w + this.el.getBorderWidth("lr"));
28330             Roo.fly(main).setWidth(w);
28331             this.internalRender = true;
28332             // opera does not respect the auto grow header center column
28333             // then, after it gets a width opera refuses to recalculate
28334             // without a second pass
28335             if(Roo.isOpera && !this.secondPass){
28336                 main.rows[0].cells[1].style.width = (w - (main.rows[0].cells[0].offsetWidth+main.rows[0].cells[2].offsetWidth)) + "px";
28337                 this.secondPass = true;
28338                 this.update.defer(10, this, [date]);
28339             }
28340         }
28341         
28342         
28343     }
28344 });        /*
28345  * Based on:
28346  * Ext JS Library 1.1.1
28347  * Copyright(c) 2006-2007, Ext JS, LLC.
28348  *
28349  * Originally Released Under LGPL - original licence link has changed is not relivant.
28350  *
28351  * Fork - LGPL
28352  * <script type="text/javascript">
28353  */
28354 /**
28355  * @class Roo.TabPanel
28356  * @extends Roo.util.Observable
28357  * A lightweight tab container.
28358  * <br><br>
28359  * Usage:
28360  * <pre><code>
28361 // basic tabs 1, built from existing content
28362 var tabs = new Roo.TabPanel("tabs1");
28363 tabs.addTab("script", "View Script");
28364 tabs.addTab("markup", "View Markup");
28365 tabs.activate("script");
28366
28367 // more advanced tabs, built from javascript
28368 var jtabs = new Roo.TabPanel("jtabs");
28369 jtabs.addTab("jtabs-1", "Normal Tab", "My content was added during construction.");
28370
28371 // set up the UpdateManager
28372 var tab2 = jtabs.addTab("jtabs-2", "Ajax Tab 1");
28373 var updater = tab2.getUpdateManager();
28374 updater.setDefaultUrl("ajax1.htm");
28375 tab2.on('activate', updater.refresh, updater, true);
28376
28377 // Use setUrl for Ajax loading
28378 var tab3 = jtabs.addTab("jtabs-3", "Ajax Tab 2");
28379 tab3.setUrl("ajax2.htm", null, true);
28380
28381 // Disabled tab
28382 var tab4 = jtabs.addTab("tabs1-5", "Disabled Tab", "Can't see me cause I'm disabled");
28383 tab4.disable();
28384
28385 jtabs.activate("jtabs-1");
28386  * </code></pre>
28387  * @constructor
28388  * Create a new TabPanel.
28389  * @param {String/HTMLElement/Roo.Element} container The id, DOM element or Roo.Element container where this TabPanel is to be rendered.
28390  * @param {Object/Boolean} config Config object to set any properties for this TabPanel, or true to render the tabs on the bottom.
28391  */
28392 Roo.TabPanel = function(container, config){
28393     /**
28394     * The container element for this TabPanel.
28395     * @type Roo.Element
28396     */
28397     this.el = Roo.get(container, true);
28398     if(config){
28399         if(typeof config == "boolean"){
28400             this.tabPosition = config ? "bottom" : "top";
28401         }else{
28402             Roo.apply(this, config);
28403         }
28404     }
28405     if(this.tabPosition == "bottom"){
28406         this.bodyEl = Roo.get(this.createBody(this.el.dom));
28407         this.el.addClass("x-tabs-bottom");
28408     }
28409     this.stripWrap = Roo.get(this.createStrip(this.el.dom), true);
28410     this.stripEl = Roo.get(this.createStripList(this.stripWrap.dom), true);
28411     this.stripBody = Roo.get(this.stripWrap.dom.firstChild.firstChild, true);
28412     if(Roo.isIE){
28413         Roo.fly(this.stripWrap.dom.firstChild).setStyle("overflow-x", "hidden");
28414     }
28415     if(this.tabPosition != "bottom"){
28416         /** The body element that contains {@link Roo.TabPanelItem} bodies. +
28417          * @type Roo.Element
28418          */
28419         this.bodyEl = Roo.get(this.createBody(this.el.dom));
28420         this.el.addClass("x-tabs-top");
28421     }
28422     this.items = [];
28423
28424     this.bodyEl.setStyle("position", "relative");
28425
28426     this.active = null;
28427     this.activateDelegate = this.activate.createDelegate(this);
28428
28429     this.addEvents({
28430         /**
28431          * @event tabchange
28432          * Fires when the active tab changes
28433          * @param {Roo.TabPanel} this
28434          * @param {Roo.TabPanelItem} activePanel The new active tab
28435          */
28436         "tabchange": true,
28437         /**
28438          * @event beforetabchange
28439          * Fires before the active tab changes, set cancel to true on the "e" parameter to cancel the change
28440          * @param {Roo.TabPanel} this
28441          * @param {Object} e Set cancel to true on this object to cancel the tab change
28442          * @param {Roo.TabPanelItem} tab The tab being changed to
28443          */
28444         "beforetabchange" : true
28445     });
28446
28447     Roo.EventManager.onWindowResize(this.onResize, this);
28448     this.cpad = this.el.getPadding("lr");
28449     this.hiddenCount = 0;
28450
28451
28452     // toolbar on the tabbar support...
28453     if (this.toolbar) {
28454         var tcfg = this.toolbar;
28455         tcfg.container = this.stripEl.child('td.x-tab-strip-toolbar');  
28456         this.toolbar = new Roo.Toolbar(tcfg);
28457         if (Roo.isSafari) {
28458             var tbl = tcfg.container.child('table', true);
28459             tbl.setAttribute('width', '100%');
28460         }
28461         
28462     }
28463    
28464
28465
28466     Roo.TabPanel.superclass.constructor.call(this);
28467 };
28468
28469 Roo.extend(Roo.TabPanel, Roo.util.Observable, {
28470     /*
28471      *@cfg {String} tabPosition "top" or "bottom" (defaults to "top")
28472      */
28473     tabPosition : "top",
28474     /*
28475      *@cfg {Number} currentTabWidth The width of the current tab (defaults to 0)
28476      */
28477     currentTabWidth : 0,
28478     /*
28479      *@cfg {Number} minTabWidth The minimum width of a tab (defaults to 40) (ignored if {@link #resizeTabs} is not true)
28480      */
28481     minTabWidth : 40,
28482     /*
28483      *@cfg {Number} maxTabWidth The maximum width of a tab (defaults to 250) (ignored if {@link #resizeTabs} is not true)
28484      */
28485     maxTabWidth : 250,
28486     /*
28487      *@cfg {Number} preferredTabWidth The preferred (default) width of a tab (defaults to 175) (ignored if {@link #resizeTabs} is not true)
28488      */
28489     preferredTabWidth : 175,
28490     /*
28491      *@cfg {Boolean} resizeTabs True to enable dynamic tab resizing (defaults to false)
28492      */
28493     resizeTabs : false,
28494     /*
28495      *@cfg {Boolean} monitorResize Set this to true to turn on window resize monitoring (ignored if {@link #resizeTabs} is not true) (defaults to true)
28496      */
28497     monitorResize : true,
28498     /*
28499      *@cfg {Object} toolbar xtype description of toolbar to show at the right of the tab bar. 
28500      */
28501     toolbar : false,
28502
28503     /**
28504      * Creates a new {@link Roo.TabPanelItem} by looking for an existing element with the provided id -- if it's not found it creates one.
28505      * @param {String} id The id of the div to use <b>or create</b>
28506      * @param {String} text The text for the tab
28507      * @param {String} content (optional) Content to put in the TabPanelItem body
28508      * @param {Boolean} closable (optional) True to create a close icon on the tab
28509      * @return {Roo.TabPanelItem} The created TabPanelItem
28510      */
28511     addTab : function(id, text, content, closable){
28512         var item = new Roo.TabPanelItem(this, id, text, closable);
28513         this.addTabItem(item);
28514         if(content){
28515             item.setContent(content);
28516         }
28517         return item;
28518     },
28519
28520     /**
28521      * Returns the {@link Roo.TabPanelItem} with the specified id/index
28522      * @param {String/Number} id The id or index of the TabPanelItem to fetch.
28523      * @return {Roo.TabPanelItem}
28524      */
28525     getTab : function(id){
28526         return this.items[id];
28527     },
28528
28529     /**
28530      * Hides the {@link Roo.TabPanelItem} with the specified id/index
28531      * @param {String/Number} id The id or index of the TabPanelItem to hide.
28532      */
28533     hideTab : function(id){
28534         var t = this.items[id];
28535         if(!t.isHidden()){
28536            t.setHidden(true);
28537            this.hiddenCount++;
28538            this.autoSizeTabs();
28539         }
28540     },
28541
28542     /**
28543      * "Unhides" the {@link Roo.TabPanelItem} with the specified id/index.
28544      * @param {String/Number} id The id or index of the TabPanelItem to unhide.
28545      */
28546     unhideTab : function(id){
28547         var t = this.items[id];
28548         if(t.isHidden()){
28549            t.setHidden(false);
28550            this.hiddenCount--;
28551            this.autoSizeTabs();
28552         }
28553     },
28554
28555     /**
28556      * Adds an existing {@link Roo.TabPanelItem}.
28557      * @param {Roo.TabPanelItem} item The TabPanelItem to add
28558      */
28559     addTabItem : function(item){
28560         this.items[item.id] = item;
28561         this.items.push(item);
28562         if(this.resizeTabs){
28563            item.setWidth(this.currentTabWidth || this.preferredTabWidth);
28564            this.autoSizeTabs();
28565         }else{
28566             item.autoSize();
28567         }
28568     },
28569
28570     /**
28571      * Removes a {@link Roo.TabPanelItem}.
28572      * @param {String/Number} id The id or index of the TabPanelItem to remove.
28573      */
28574     removeTab : function(id){
28575         var items = this.items;
28576         var tab = items[id];
28577         if(!tab) { return; }
28578         var index = items.indexOf(tab);
28579         if(this.active == tab && items.length > 1){
28580             var newTab = this.getNextAvailable(index);
28581             if(newTab) {
28582                 newTab.activate();
28583             }
28584         }
28585         this.stripEl.dom.removeChild(tab.pnode.dom);
28586         if(tab.bodyEl.dom.parentNode == this.bodyEl.dom){ // if it was moved already prevent error
28587             this.bodyEl.dom.removeChild(tab.bodyEl.dom);
28588         }
28589         items.splice(index, 1);
28590         delete this.items[tab.id];
28591         tab.fireEvent("close", tab);
28592         tab.purgeListeners();
28593         this.autoSizeTabs();
28594     },
28595
28596     getNextAvailable : function(start){
28597         var items = this.items;
28598         var index = start;
28599         // look for a next tab that will slide over to
28600         // replace the one being removed
28601         while(index < items.length){
28602             var item = items[++index];
28603             if(item && !item.isHidden()){
28604                 return item;
28605             }
28606         }
28607         // if one isn't found select the previous tab (on the left)
28608         index = start;
28609         while(index >= 0){
28610             var item = items[--index];
28611             if(item && !item.isHidden()){
28612                 return item;
28613             }
28614         }
28615         return null;
28616     },
28617
28618     /**
28619      * Disables a {@link Roo.TabPanelItem}. It cannot be the active tab, if it is this call is ignored.
28620      * @param {String/Number} id The id or index of the TabPanelItem to disable.
28621      */
28622     disableTab : function(id){
28623         var tab = this.items[id];
28624         if(tab && this.active != tab){
28625             tab.disable();
28626         }
28627     },
28628
28629     /**
28630      * Enables a {@link Roo.TabPanelItem} that is disabled.
28631      * @param {String/Number} id The id or index of the TabPanelItem to enable.
28632      */
28633     enableTab : function(id){
28634         var tab = this.items[id];
28635         tab.enable();
28636     },
28637
28638     /**
28639      * Activates a {@link Roo.TabPanelItem}. The currently active one will be deactivated.
28640      * @param {String/Number} id The id or index of the TabPanelItem to activate.
28641      * @return {Roo.TabPanelItem} The TabPanelItem.
28642      */
28643     activate : function(id){
28644         var tab = this.items[id];
28645         if(!tab){
28646             return null;
28647         }
28648         if(tab == this.active || tab.disabled){
28649             return tab;
28650         }
28651         var e = {};
28652         this.fireEvent("beforetabchange", this, e, tab);
28653         if(e.cancel !== true && !tab.disabled){
28654             if(this.active){
28655                 this.active.hide();
28656             }
28657             this.active = this.items[id];
28658             this.active.show();
28659             this.fireEvent("tabchange", this, this.active);
28660         }
28661         return tab;
28662     },
28663
28664     /**
28665      * Gets the active {@link Roo.TabPanelItem}.
28666      * @return {Roo.TabPanelItem} The active TabPanelItem or null if none are active.
28667      */
28668     getActiveTab : function(){
28669         return this.active;
28670     },
28671
28672     /**
28673      * Updates the tab body element to fit the height of the container element
28674      * for overflow scrolling
28675      * @param {Number} targetHeight (optional) Override the starting height from the elements height
28676      */
28677     syncHeight : function(targetHeight){
28678         var height = (targetHeight || this.el.getHeight())-this.el.getBorderWidth("tb")-this.el.getPadding("tb");
28679         var bm = this.bodyEl.getMargins();
28680         var newHeight = height-(this.stripWrap.getHeight()||0)-(bm.top+bm.bottom);
28681         this.bodyEl.setHeight(newHeight);
28682         return newHeight;
28683     },
28684
28685     onResize : function(){
28686         if(this.monitorResize){
28687             this.autoSizeTabs();
28688         }
28689     },
28690
28691     /**
28692      * Disables tab resizing while tabs are being added (if {@link #resizeTabs} is false this does nothing)
28693      */
28694     beginUpdate : function(){
28695         this.updating = true;
28696     },
28697
28698     /**
28699      * Stops an update and resizes the tabs (if {@link #resizeTabs} is false this does nothing)
28700      */
28701     endUpdate : function(){
28702         this.updating = false;
28703         this.autoSizeTabs();
28704     },
28705
28706     /**
28707      * Manual call to resize the tabs (if {@link #resizeTabs} is false this does nothing)
28708      */
28709     autoSizeTabs : function(){
28710         var count = this.items.length;
28711         var vcount = count - this.hiddenCount;
28712         if(!this.resizeTabs || count < 1 || vcount < 1 || this.updating) {
28713             return;
28714         }
28715         var w = Math.max(this.el.getWidth() - this.cpad, 10);
28716         var availWidth = Math.floor(w / vcount);
28717         var b = this.stripBody;
28718         if(b.getWidth() > w){
28719             var tabs = this.items;
28720             this.setTabWidth(Math.max(availWidth, this.minTabWidth)-2);
28721             if(availWidth < this.minTabWidth){
28722                 /*if(!this.sleft){    // incomplete scrolling code
28723                     this.createScrollButtons();
28724                 }
28725                 this.showScroll();
28726                 this.stripClip.setWidth(w - (this.sleft.getWidth()+this.sright.getWidth()));*/
28727             }
28728         }else{
28729             if(this.currentTabWidth < this.preferredTabWidth){
28730                 this.setTabWidth(Math.min(availWidth, this.preferredTabWidth)-2);
28731             }
28732         }
28733     },
28734
28735     /**
28736      * Returns the number of tabs in this TabPanel.
28737      * @return {Number}
28738      */
28739      getCount : function(){
28740          return this.items.length;
28741      },
28742
28743     /**
28744      * Resizes all the tabs to the passed width
28745      * @param {Number} The new width
28746      */
28747     setTabWidth : function(width){
28748         this.currentTabWidth = width;
28749         for(var i = 0, len = this.items.length; i < len; i++) {
28750                 if(!this.items[i].isHidden()) {
28751                 this.items[i].setWidth(width);
28752             }
28753         }
28754     },
28755
28756     /**
28757      * Destroys this TabPanel
28758      * @param {Boolean} removeEl (optional) True to remove the element from the DOM as well (defaults to undefined)
28759      */
28760     destroy : function(removeEl){
28761         Roo.EventManager.removeResizeListener(this.onResize, this);
28762         for(var i = 0, len = this.items.length; i < len; i++){
28763             this.items[i].purgeListeners();
28764         }
28765         if(removeEl === true){
28766             this.el.update("");
28767             this.el.remove();
28768         }
28769     }
28770 });
28771
28772 /**
28773  * @class Roo.TabPanelItem
28774  * @extends Roo.util.Observable
28775  * Represents an individual item (tab plus body) in a TabPanel.
28776  * @param {Roo.TabPanel} tabPanel The {@link Roo.TabPanel} this TabPanelItem belongs to
28777  * @param {String} id The id of this TabPanelItem
28778  * @param {String} text The text for the tab of this TabPanelItem
28779  * @param {Boolean} closable True to allow this TabPanelItem to be closable (defaults to false)
28780  */
28781 Roo.TabPanelItem = function(tabPanel, id, text, closable){
28782     /**
28783      * The {@link Roo.TabPanel} this TabPanelItem belongs to
28784      * @type Roo.TabPanel
28785      */
28786     this.tabPanel = tabPanel;
28787     /**
28788      * The id for this TabPanelItem
28789      * @type String
28790      */
28791     this.id = id;
28792     /** @private */
28793     this.disabled = false;
28794     /** @private */
28795     this.text = text;
28796     /** @private */
28797     this.loaded = false;
28798     this.closable = closable;
28799
28800     /**
28801      * The body element for this TabPanelItem.
28802      * @type Roo.Element
28803      */
28804     this.bodyEl = Roo.get(tabPanel.createItemBody(tabPanel.bodyEl.dom, id));
28805     this.bodyEl.setVisibilityMode(Roo.Element.VISIBILITY);
28806     this.bodyEl.setStyle("display", "block");
28807     this.bodyEl.setStyle("zoom", "1");
28808     this.hideAction();
28809
28810     var els = tabPanel.createStripElements(tabPanel.stripEl.dom, text, closable);
28811     /** @private */
28812     this.el = Roo.get(els.el, true);
28813     this.inner = Roo.get(els.inner, true);
28814     this.textEl = Roo.get(this.el.dom.firstChild.firstChild.firstChild, true);
28815     this.pnode = Roo.get(els.el.parentNode, true);
28816     this.el.on("mousedown", this.onTabMouseDown, this);
28817     this.el.on("click", this.onTabClick, this);
28818     /** @private */
28819     if(closable){
28820         var c = Roo.get(els.close, true);
28821         c.dom.title = this.closeText;
28822         c.addClassOnOver("close-over");
28823         c.on("click", this.closeClick, this);
28824      }
28825
28826     this.addEvents({
28827          /**
28828          * @event activate
28829          * Fires when this tab becomes the active tab.
28830          * @param {Roo.TabPanel} tabPanel The parent TabPanel
28831          * @param {Roo.TabPanelItem} this
28832          */
28833         "activate": true,
28834         /**
28835          * @event beforeclose
28836          * Fires before this tab is closed. To cancel the close, set cancel to true on e (e.cancel = true).
28837          * @param {Roo.TabPanelItem} this
28838          * @param {Object} e Set cancel to true on this object to cancel the close.
28839          */
28840         "beforeclose": true,
28841         /**
28842          * @event close
28843          * Fires when this tab is closed.
28844          * @param {Roo.TabPanelItem} this
28845          */
28846          "close": true,
28847         /**
28848          * @event deactivate
28849          * Fires when this tab is no longer the active tab.
28850          * @param {Roo.TabPanel} tabPanel The parent TabPanel
28851          * @param {Roo.TabPanelItem} this
28852          */
28853          "deactivate" : true
28854     });
28855     this.hidden = false;
28856
28857     Roo.TabPanelItem.superclass.constructor.call(this);
28858 };
28859
28860 Roo.extend(Roo.TabPanelItem, Roo.util.Observable, {
28861     purgeListeners : function(){
28862        Roo.util.Observable.prototype.purgeListeners.call(this);
28863        this.el.removeAllListeners();
28864     },
28865     /**
28866      * Shows this TabPanelItem -- this <b>does not</b> deactivate the currently active TabPanelItem.
28867      */
28868     show : function(){
28869         this.pnode.addClass("on");
28870         this.showAction();
28871         if(Roo.isOpera){
28872             this.tabPanel.stripWrap.repaint();
28873         }
28874         this.fireEvent("activate", this.tabPanel, this);
28875     },
28876
28877     /**
28878      * Returns true if this tab is the active tab.
28879      * @return {Boolean}
28880      */
28881     isActive : function(){
28882         return this.tabPanel.getActiveTab() == this;
28883     },
28884
28885     /**
28886      * Hides this TabPanelItem -- if you don't activate another TabPanelItem this could look odd.
28887      */
28888     hide : function(){
28889         this.pnode.removeClass("on");
28890         this.hideAction();
28891         this.fireEvent("deactivate", this.tabPanel, this);
28892     },
28893
28894     hideAction : function(){
28895         this.bodyEl.hide();
28896         this.bodyEl.setStyle("position", "absolute");
28897         this.bodyEl.setLeft("-20000px");
28898         this.bodyEl.setTop("-20000px");
28899     },
28900
28901     showAction : function(){
28902         this.bodyEl.setStyle("position", "relative");
28903         this.bodyEl.setTop("");
28904         this.bodyEl.setLeft("");
28905         this.bodyEl.show();
28906     },
28907
28908     /**
28909      * Set the tooltip for the tab.
28910      * @param {String} tooltip The tab's tooltip
28911      */
28912     setTooltip : function(text){
28913         if(Roo.QuickTips && Roo.QuickTips.isEnabled()){
28914             this.textEl.dom.qtip = text;
28915             this.textEl.dom.removeAttribute('title');
28916         }else{
28917             this.textEl.dom.title = text;
28918         }
28919     },
28920
28921     onTabClick : function(e){
28922         e.preventDefault();
28923         this.tabPanel.activate(this.id);
28924     },
28925
28926     onTabMouseDown : function(e){
28927         e.preventDefault();
28928         this.tabPanel.activate(this.id);
28929     },
28930
28931     getWidth : function(){
28932         return this.inner.getWidth();
28933     },
28934
28935     setWidth : function(width){
28936         var iwidth = width - this.pnode.getPadding("lr");
28937         this.inner.setWidth(iwidth);
28938         this.textEl.setWidth(iwidth-this.inner.getPadding("lr"));
28939         this.pnode.setWidth(width);
28940     },
28941
28942     /**
28943      * Show or hide the tab
28944      * @param {Boolean} hidden True to hide or false to show.
28945      */
28946     setHidden : function(hidden){
28947         this.hidden = hidden;
28948         this.pnode.setStyle("display", hidden ? "none" : "");
28949     },
28950
28951     /**
28952      * Returns true if this tab is "hidden"
28953      * @return {Boolean}
28954      */
28955     isHidden : function(){
28956         return this.hidden;
28957     },
28958
28959     /**
28960      * Returns the text for this tab
28961      * @return {String}
28962      */
28963     getText : function(){
28964         return this.text;
28965     },
28966
28967     autoSize : function(){
28968         //this.el.beginMeasure();
28969         this.textEl.setWidth(1);
28970         /*
28971          *  #2804 [new] Tabs in Roojs
28972          *  increase the width by 2-4 pixels to prevent the ellipssis showing in chrome
28973          */
28974         this.setWidth(this.textEl.dom.scrollWidth+this.pnode.getPadding("lr")+this.inner.getPadding("lr") + 2);
28975         //this.el.endMeasure();
28976     },
28977
28978     /**
28979      * Sets the text for the tab (Note: this also sets the tooltip text)
28980      * @param {String} text The tab's text and tooltip
28981      */
28982     setText : function(text){
28983         this.text = text;
28984         this.textEl.update(text);
28985         this.setTooltip(text);
28986         if(!this.tabPanel.resizeTabs){
28987             this.autoSize();
28988         }
28989     },
28990     /**
28991      * Activates this TabPanelItem -- this <b>does</b> deactivate the currently active TabPanelItem.
28992      */
28993     activate : function(){
28994         this.tabPanel.activate(this.id);
28995     },
28996
28997     /**
28998      * Disables this TabPanelItem -- this does nothing if this is the active TabPanelItem.
28999      */
29000     disable : function(){
29001         if(this.tabPanel.active != this){
29002             this.disabled = true;
29003             this.pnode.addClass("disabled");
29004         }
29005     },
29006
29007     /**
29008      * Enables this TabPanelItem if it was previously disabled.
29009      */
29010     enable : function(){
29011         this.disabled = false;
29012         this.pnode.removeClass("disabled");
29013     },
29014
29015     /**
29016      * Sets the content for this TabPanelItem.
29017      * @param {String} content The content
29018      * @param {Boolean} loadScripts true to look for and load scripts
29019      */
29020     setContent : function(content, loadScripts){
29021         this.bodyEl.update(content, loadScripts);
29022     },
29023
29024     /**
29025      * Gets the {@link Roo.UpdateManager} for the body of this TabPanelItem. Enables you to perform Ajax updates.
29026      * @return {Roo.UpdateManager} The UpdateManager
29027      */
29028     getUpdateManager : function(){
29029         return this.bodyEl.getUpdateManager();
29030     },
29031
29032     /**
29033      * Set a URL to be used to load the content for this TabPanelItem.
29034      * @param {String/Function} url The URL to load the content from, or a function to call to get the URL
29035      * @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)
29036      * @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)
29037      * @return {Roo.UpdateManager} The UpdateManager
29038      */
29039     setUrl : function(url, params, loadOnce){
29040         if(this.refreshDelegate){
29041             this.un('activate', this.refreshDelegate);
29042         }
29043         this.refreshDelegate = this._handleRefresh.createDelegate(this, [url, params, loadOnce]);
29044         this.on("activate", this.refreshDelegate);
29045         return this.bodyEl.getUpdateManager();
29046     },
29047
29048     /** @private */
29049     _handleRefresh : function(url, params, loadOnce){
29050         if(!loadOnce || !this.loaded){
29051             var updater = this.bodyEl.getUpdateManager();
29052             updater.update(url, params, this._setLoaded.createDelegate(this));
29053         }
29054     },
29055
29056     /**
29057      *   Forces a content refresh from the URL specified in the {@link #setUrl} method.
29058      *   Will fail silently if the setUrl method has not been called.
29059      *   This does not activate the panel, just updates its content.
29060      */
29061     refresh : function(){
29062         if(this.refreshDelegate){
29063            this.loaded = false;
29064            this.refreshDelegate();
29065         }
29066     },
29067
29068     /** @private */
29069     _setLoaded : function(){
29070         this.loaded = true;
29071     },
29072
29073     /** @private */
29074     closeClick : function(e){
29075         var o = {};
29076         e.stopEvent();
29077         this.fireEvent("beforeclose", this, o);
29078         if(o.cancel !== true){
29079             this.tabPanel.removeTab(this.id);
29080         }
29081     },
29082     /**
29083      * The text displayed in the tooltip for the close icon.
29084      * @type String
29085      */
29086     closeText : "Close this tab"
29087 });
29088
29089 /** @private */
29090 Roo.TabPanel.prototype.createStrip = function(container){
29091     var strip = document.createElement("div");
29092     strip.className = "x-tabs-wrap";
29093     container.appendChild(strip);
29094     return strip;
29095 };
29096 /** @private */
29097 Roo.TabPanel.prototype.createStripList = function(strip){
29098     // div wrapper for retard IE
29099     // returns the "tr" element.
29100     strip.innerHTML = '<div class="x-tabs-strip-wrap">'+
29101         '<table class="x-tabs-strip" cellspacing="0" cellpadding="0" border="0"><tbody><tr>'+
29102         '<td class="x-tab-strip-toolbar"></td></tr></tbody></table></div>';
29103     return strip.firstChild.firstChild.firstChild.firstChild;
29104 };
29105 /** @private */
29106 Roo.TabPanel.prototype.createBody = function(container){
29107     var body = document.createElement("div");
29108     Roo.id(body, "tab-body");
29109     Roo.fly(body).addClass("x-tabs-body");
29110     container.appendChild(body);
29111     return body;
29112 };
29113 /** @private */
29114 Roo.TabPanel.prototype.createItemBody = function(bodyEl, id){
29115     var body = Roo.getDom(id);
29116     if(!body){
29117         body = document.createElement("div");
29118         body.id = id;
29119     }
29120     Roo.fly(body).addClass("x-tabs-item-body");
29121     bodyEl.insertBefore(body, bodyEl.firstChild);
29122     return body;
29123 };
29124 /** @private */
29125 Roo.TabPanel.prototype.createStripElements = function(stripEl, text, closable){
29126     var td = document.createElement("td");
29127     stripEl.insertBefore(td, stripEl.childNodes[stripEl.childNodes.length-1]);
29128     //stripEl.appendChild(td);
29129     if(closable){
29130         td.className = "x-tabs-closable";
29131         if(!this.closeTpl){
29132             this.closeTpl = new Roo.Template(
29133                '<a href="#" class="x-tabs-right"><span class="x-tabs-left"><em class="x-tabs-inner">' +
29134                '<span unselectable="on"' + (this.disableTooltips ? '' : ' title="{text}"') +' class="x-tabs-text">{text}</span>' +
29135                '<div unselectable="on" class="close-icon">&#160;</div></em></span></a>'
29136             );
29137         }
29138         var el = this.closeTpl.overwrite(td, {"text": text});
29139         var close = el.getElementsByTagName("div")[0];
29140         var inner = el.getElementsByTagName("em")[0];
29141         return {"el": el, "close": close, "inner": inner};
29142     } else {
29143         if(!this.tabTpl){
29144             this.tabTpl = new Roo.Template(
29145                '<a href="#" class="x-tabs-right"><span class="x-tabs-left"><em class="x-tabs-inner">' +
29146                '<span unselectable="on"' + (this.disableTooltips ? '' : ' title="{text}"') +' class="x-tabs-text">{text}</span></em></span></a>'
29147             );
29148         }
29149         var el = this.tabTpl.overwrite(td, {"text": text});
29150         var inner = el.getElementsByTagName("em")[0];
29151         return {"el": el, "inner": inner};
29152     }
29153 };/*
29154  * Based on:
29155  * Ext JS Library 1.1.1
29156  * Copyright(c) 2006-2007, Ext JS, LLC.
29157  *
29158  * Originally Released Under LGPL - original licence link has changed is not relivant.
29159  *
29160  * Fork - LGPL
29161  * <script type="text/javascript">
29162  */
29163
29164 /**
29165  * @class Roo.Button
29166  * @extends Roo.util.Observable
29167  * Simple Button class
29168  * @cfg {String} text The button text
29169  * @cfg {String} icon The path to an image to display in the button (the image will be set as the background-image
29170  * CSS property of the button by default, so if you want a mixed icon/text button, set cls:"x-btn-text-icon")
29171  * @cfg {Function} handler A function called when the button is clicked (can be used instead of click event)
29172  * @cfg {Object} scope The scope of the handler
29173  * @cfg {Number} minWidth The minimum width for this button (used to give a set of buttons a common width)
29174  * @cfg {String/Object} tooltip The tooltip for the button - can be a string or QuickTips config object
29175  * @cfg {Boolean} hidden True to start hidden (defaults to false)
29176  * @cfg {Boolean} disabled True to start disabled (defaults to false)
29177  * @cfg {Boolean} pressed True to start pressed (only if enableToggle = true)
29178  * @cfg {String} toggleGroup The group this toggle button is a member of (only 1 per group can be pressed, only
29179    applies if enableToggle = true)
29180  * @cfg {String/HTMLElement/Element} renderTo The element to append the button to
29181  * @cfg {Boolean/Object} repeat True to repeat fire the click event while the mouse is down. This can also be
29182   an {@link Roo.util.ClickRepeater} config object (defaults to false).
29183  * @constructor
29184  * Create a new button
29185  * @param {Object} config The config object
29186  */
29187 Roo.Button = function(renderTo, config)
29188 {
29189     if (!config) {
29190         config = renderTo;
29191         renderTo = config.renderTo || false;
29192     }
29193     
29194     Roo.apply(this, config);
29195     this.addEvents({
29196         /**
29197              * @event click
29198              * Fires when this button is clicked
29199              * @param {Button} this
29200              * @param {EventObject} e The click event
29201              */
29202             "click" : true,
29203         /**
29204              * @event toggle
29205              * Fires when the "pressed" state of this button changes (only if enableToggle = true)
29206              * @param {Button} this
29207              * @param {Boolean} pressed
29208              */
29209             "toggle" : true,
29210         /**
29211              * @event mouseover
29212              * Fires when the mouse hovers over the button
29213              * @param {Button} this
29214              * @param {Event} e The event object
29215              */
29216         'mouseover' : true,
29217         /**
29218              * @event mouseout
29219              * Fires when the mouse exits the button
29220              * @param {Button} this
29221              * @param {Event} e The event object
29222              */
29223         'mouseout': true,
29224          /**
29225              * @event render
29226              * Fires when the button is rendered
29227              * @param {Button} this
29228              */
29229         'render': true
29230     });
29231     if(this.menu){
29232         this.menu = Roo.menu.MenuMgr.get(this.menu);
29233     }
29234     // register listeners first!!  - so render can be captured..
29235     Roo.util.Observable.call(this);
29236     if(renderTo){
29237         this.render(renderTo);
29238     }
29239     
29240   
29241 };
29242
29243 Roo.extend(Roo.Button, Roo.util.Observable, {
29244     /**
29245      * 
29246      */
29247     
29248     /**
29249      * Read-only. True if this button is hidden
29250      * @type Boolean
29251      */
29252     hidden : false,
29253     /**
29254      * Read-only. True if this button is disabled
29255      * @type Boolean
29256      */
29257     disabled : false,
29258     /**
29259      * Read-only. True if this button is pressed (only if enableToggle = true)
29260      * @type Boolean
29261      */
29262     pressed : false,
29263
29264     /**
29265      * @cfg {Number} tabIndex 
29266      * The DOM tabIndex for this button (defaults to undefined)
29267      */
29268     tabIndex : undefined,
29269
29270     /**
29271      * @cfg {Boolean} enableToggle
29272      * True to enable pressed/not pressed toggling (defaults to false)
29273      */
29274     enableToggle: false,
29275     /**
29276      * @cfg {Mixed} menu
29277      * Standard menu attribute consisting of a reference to a menu object, a menu id or a menu config blob (defaults to undefined).
29278      */
29279     menu : undefined,
29280     /**
29281      * @cfg {String} menuAlign
29282      * The position to align the menu to (see {@link Roo.Element#alignTo} for more details, defaults to 'tl-bl?').
29283      */
29284     menuAlign : "tl-bl?",
29285
29286     /**
29287      * @cfg {String} iconCls
29288      * A css class which sets a background image to be used as the icon for this button (defaults to undefined).
29289      */
29290     iconCls : undefined,
29291     /**
29292      * @cfg {String} type
29293      * The button's type, corresponding to the DOM input element type attribute.  Either "submit," "reset" or "button" (default).
29294      */
29295     type : 'button',
29296
29297     // private
29298     menuClassTarget: 'tr',
29299
29300     /**
29301      * @cfg {String} clickEvent
29302      * The type of event to map to the button's event handler (defaults to 'click')
29303      */
29304     clickEvent : 'click',
29305
29306     /**
29307      * @cfg {Boolean} handleMouseEvents
29308      * False to disable visual cues on mouseover, mouseout and mousedown (defaults to true)
29309      */
29310     handleMouseEvents : true,
29311
29312     /**
29313      * @cfg {String} tooltipType
29314      * The type of tooltip to use. Either "qtip" (default) for QuickTips or "title" for title attribute.
29315      */
29316     tooltipType : 'qtip',
29317
29318     /**
29319      * @cfg {String} cls
29320      * A CSS class to apply to the button's main element.
29321      */
29322     
29323     /**
29324      * @cfg {Roo.Template} template (Optional)
29325      * An {@link Roo.Template} with which to create the Button's main element. This Template must
29326      * contain numeric substitution parameter 0 if it is to display the tRoo property. Changing the template could
29327      * require code modifications if required elements (e.g. a button) aren't present.
29328      */
29329
29330     // private
29331     render : function(renderTo){
29332         var btn;
29333         if(this.hideParent){
29334             this.parentEl = Roo.get(renderTo);
29335         }
29336         if(!this.dhconfig){
29337             if(!this.template){
29338                 if(!Roo.Button.buttonTemplate){
29339                     // hideous table template
29340                     Roo.Button.buttonTemplate = new Roo.Template(
29341                         '<table border="0" cellpadding="0" cellspacing="0" class="x-btn-wrap"><tbody><tr>',
29342                         '<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>',
29343                         "</tr></tbody></table>");
29344                 }
29345                 this.template = Roo.Button.buttonTemplate;
29346             }
29347             btn = this.template.append(renderTo, [this.text || '&#160;', this.type], true);
29348             var btnEl = btn.child("button:first");
29349             btnEl.on('focus', this.onFocus, this);
29350             btnEl.on('blur', this.onBlur, this);
29351             if(this.cls){
29352                 btn.addClass(this.cls);
29353             }
29354             if(this.icon){
29355                 btnEl.setStyle('background-image', 'url(' +this.icon +')');
29356             }
29357             if(this.iconCls){
29358                 btnEl.addClass(this.iconCls);
29359                 if(!this.cls){
29360                     btn.addClass(this.text ? 'x-btn-text-icon' : 'x-btn-icon');
29361                 }
29362             }
29363             if(this.tabIndex !== undefined){
29364                 btnEl.dom.tabIndex = this.tabIndex;
29365             }
29366             if(this.tooltip){
29367                 if(typeof this.tooltip == 'object'){
29368                     Roo.QuickTips.tips(Roo.apply({
29369                           target: btnEl.id
29370                     }, this.tooltip));
29371                 } else {
29372                     btnEl.dom[this.tooltipType] = this.tooltip;
29373                 }
29374             }
29375         }else{
29376             btn = Roo.DomHelper.append(Roo.get(renderTo).dom, this.dhconfig, true);
29377         }
29378         this.el = btn;
29379         if(this.id){
29380             this.el.dom.id = this.el.id = this.id;
29381         }
29382         if(this.menu){
29383             this.el.child(this.menuClassTarget).addClass("x-btn-with-menu");
29384             this.menu.on("show", this.onMenuShow, this);
29385             this.menu.on("hide", this.onMenuHide, this);
29386         }
29387         btn.addClass("x-btn");
29388         if(Roo.isIE && !Roo.isIE7){
29389             this.autoWidth.defer(1, this);
29390         }else{
29391             this.autoWidth();
29392         }
29393         if(this.handleMouseEvents){
29394             btn.on("mouseover", this.onMouseOver, this);
29395             btn.on("mouseout", this.onMouseOut, this);
29396             btn.on("mousedown", this.onMouseDown, this);
29397         }
29398         btn.on(this.clickEvent, this.onClick, this);
29399         //btn.on("mouseup", this.onMouseUp, this);
29400         if(this.hidden){
29401             this.hide();
29402         }
29403         if(this.disabled){
29404             this.disable();
29405         }
29406         Roo.ButtonToggleMgr.register(this);
29407         if(this.pressed){
29408             this.el.addClass("x-btn-pressed");
29409         }
29410         if(this.repeat){
29411             var repeater = new Roo.util.ClickRepeater(btn,
29412                 typeof this.repeat == "object" ? this.repeat : {}
29413             );
29414             repeater.on("click", this.onClick,  this);
29415         }
29416         
29417         this.fireEvent('render', this);
29418         
29419     },
29420     /**
29421      * Returns the button's underlying element
29422      * @return {Roo.Element} The element
29423      */
29424     getEl : function(){
29425         return this.el;  
29426     },
29427     
29428     /**
29429      * Destroys this Button and removes any listeners.
29430      */
29431     destroy : function(){
29432         Roo.ButtonToggleMgr.unregister(this);
29433         this.el.removeAllListeners();
29434         this.purgeListeners();
29435         this.el.remove();
29436     },
29437
29438     // private
29439     autoWidth : function(){
29440         if(this.el){
29441             this.el.setWidth("auto");
29442             if(Roo.isIE7 && Roo.isStrict){
29443                 var ib = this.el.child('button');
29444                 if(ib && ib.getWidth() > 20){
29445                     ib.clip();
29446                     ib.setWidth(Roo.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));
29447                 }
29448             }
29449             if(this.minWidth){
29450                 if(this.hidden){
29451                     this.el.beginMeasure();
29452                 }
29453                 if(this.el.getWidth() < this.minWidth){
29454                     this.el.setWidth(this.minWidth);
29455                 }
29456                 if(this.hidden){
29457                     this.el.endMeasure();
29458                 }
29459             }
29460         }
29461     },
29462
29463     /**
29464      * Assigns this button's click handler
29465      * @param {Function} handler The function to call when the button is clicked
29466      * @param {Object} scope (optional) Scope for the function passed in
29467      */
29468     setHandler : function(handler, scope){
29469         this.handler = handler;
29470         this.scope = scope;  
29471     },
29472     
29473     /**
29474      * Sets this button's text
29475      * @param {String} text The button text
29476      */
29477     setText : function(text){
29478         this.text = text;
29479         if(this.el){
29480             this.el.child("td.x-btn-center button.x-btn-text").update(text);
29481         }
29482         this.autoWidth();
29483     },
29484     
29485     /**
29486      * Gets the text for this button
29487      * @return {String} The button text
29488      */
29489     getText : function(){
29490         return this.text;  
29491     },
29492     
29493     /**
29494      * Show this button
29495      */
29496     show: function(){
29497         this.hidden = false;
29498         if(this.el){
29499             this[this.hideParent? 'parentEl' : 'el'].setStyle("display", "");
29500         }
29501     },
29502     
29503     /**
29504      * Hide this button
29505      */
29506     hide: function(){
29507         this.hidden = true;
29508         if(this.el){
29509             this[this.hideParent? 'parentEl' : 'el'].setStyle("display", "none");
29510         }
29511     },
29512     
29513     /**
29514      * Convenience function for boolean show/hide
29515      * @param {Boolean} visible True to show, false to hide
29516      */
29517     setVisible: function(visible){
29518         if(visible) {
29519             this.show();
29520         }else{
29521             this.hide();
29522         }
29523     },
29524     
29525     /**
29526      * If a state it passed, it becomes the pressed state otherwise the current state is toggled.
29527      * @param {Boolean} state (optional) Force a particular state
29528      */
29529     toggle : function(state){
29530         state = state === undefined ? !this.pressed : state;
29531         if(state != this.pressed){
29532             if(state){
29533                 this.el.addClass("x-btn-pressed");
29534                 this.pressed = true;
29535                 this.fireEvent("toggle", this, true);
29536             }else{
29537                 this.el.removeClass("x-btn-pressed");
29538                 this.pressed = false;
29539                 this.fireEvent("toggle", this, false);
29540             }
29541             if(this.toggleHandler){
29542                 this.toggleHandler.call(this.scope || this, this, state);
29543             }
29544         }
29545     },
29546     
29547     /**
29548      * Focus the button
29549      */
29550     focus : function(){
29551         this.el.child('button:first').focus();
29552     },
29553     
29554     /**
29555      * Disable this button
29556      */
29557     disable : function(){
29558         if(this.el){
29559             this.el.addClass("x-btn-disabled");
29560         }
29561         this.disabled = true;
29562     },
29563     
29564     /**
29565      * Enable this button
29566      */
29567     enable : function(){
29568         if(this.el){
29569             this.el.removeClass("x-btn-disabled");
29570         }
29571         this.disabled = false;
29572     },
29573
29574     /**
29575      * Convenience function for boolean enable/disable
29576      * @param {Boolean} enabled True to enable, false to disable
29577      */
29578     setDisabled : function(v){
29579         this[v !== true ? "enable" : "disable"]();
29580     },
29581
29582     // private
29583     onClick : function(e)
29584     {
29585         if(e){
29586             e.preventDefault();
29587         }
29588         if(e.button != 0){
29589             return;
29590         }
29591         if(!this.disabled){
29592             if(this.enableToggle){
29593                 this.toggle();
29594             }
29595             if(this.menu && !this.menu.isVisible()){
29596                 this.menu.show(this.el, this.menuAlign);
29597             }
29598             this.fireEvent("click", this, e);
29599             if(this.handler){
29600                 this.el.removeClass("x-btn-over");
29601                 this.handler.call(this.scope || this, this, e);
29602             }
29603         }
29604     },
29605     // private
29606     onMouseOver : function(e){
29607         if(!this.disabled){
29608             this.el.addClass("x-btn-over");
29609             this.fireEvent('mouseover', this, e);
29610         }
29611     },
29612     // private
29613     onMouseOut : function(e){
29614         if(!e.within(this.el,  true)){
29615             this.el.removeClass("x-btn-over");
29616             this.fireEvent('mouseout', this, e);
29617         }
29618     },
29619     // private
29620     onFocus : function(e){
29621         if(!this.disabled){
29622             this.el.addClass("x-btn-focus");
29623         }
29624     },
29625     // private
29626     onBlur : function(e){
29627         this.el.removeClass("x-btn-focus");
29628     },
29629     // private
29630     onMouseDown : function(e){
29631         if(!this.disabled && e.button == 0){
29632             this.el.addClass("x-btn-click");
29633             Roo.get(document).on('mouseup', this.onMouseUp, this);
29634         }
29635     },
29636     // private
29637     onMouseUp : function(e){
29638         if(e.button == 0){
29639             this.el.removeClass("x-btn-click");
29640             Roo.get(document).un('mouseup', this.onMouseUp, this);
29641         }
29642     },
29643     // private
29644     onMenuShow : function(e){
29645         this.el.addClass("x-btn-menu-active");
29646     },
29647     // private
29648     onMenuHide : function(e){
29649         this.el.removeClass("x-btn-menu-active");
29650     }   
29651 });
29652
29653 // Private utility class used by Button
29654 Roo.ButtonToggleMgr = function(){
29655    var groups = {};
29656    
29657    function toggleGroup(btn, state){
29658        if(state){
29659            var g = groups[btn.toggleGroup];
29660            for(var i = 0, l = g.length; i < l; i++){
29661                if(g[i] != btn){
29662                    g[i].toggle(false);
29663                }
29664            }
29665        }
29666    }
29667    
29668    return {
29669        register : function(btn){
29670            if(!btn.toggleGroup){
29671                return;
29672            }
29673            var g = groups[btn.toggleGroup];
29674            if(!g){
29675                g = groups[btn.toggleGroup] = [];
29676            }
29677            g.push(btn);
29678            btn.on("toggle", toggleGroup);
29679        },
29680        
29681        unregister : function(btn){
29682            if(!btn.toggleGroup){
29683                return;
29684            }
29685            var g = groups[btn.toggleGroup];
29686            if(g){
29687                g.remove(btn);
29688                btn.un("toggle", toggleGroup);
29689            }
29690        }
29691    };
29692 }();/*
29693  * Based on:
29694  * Ext JS Library 1.1.1
29695  * Copyright(c) 2006-2007, Ext JS, LLC.
29696  *
29697  * Originally Released Under LGPL - original licence link has changed is not relivant.
29698  *
29699  * Fork - LGPL
29700  * <script type="text/javascript">
29701  */
29702  
29703 /**
29704  * @class Roo.SplitButton
29705  * @extends Roo.Button
29706  * A split button that provides a built-in dropdown arrow that can fire an event separately from the default
29707  * click event of the button.  Typically this would be used to display a dropdown menu that provides additional
29708  * options to the primary button action, but any custom handler can provide the arrowclick implementation.
29709  * @cfg {Function} arrowHandler A function called when the arrow button is clicked (can be used instead of click event)
29710  * @cfg {String} arrowTooltip The title attribute of the arrow
29711  * @constructor
29712  * Create a new menu button
29713  * @param {String/HTMLElement/Element} renderTo The element to append the button to
29714  * @param {Object} config The config object
29715  */
29716 Roo.SplitButton = function(renderTo, config){
29717     Roo.SplitButton.superclass.constructor.call(this, renderTo, config);
29718     /**
29719      * @event arrowclick
29720      * Fires when this button's arrow is clicked
29721      * @param {SplitButton} this
29722      * @param {EventObject} e The click event
29723      */
29724     this.addEvents({"arrowclick":true});
29725 };
29726
29727 Roo.extend(Roo.SplitButton, Roo.Button, {
29728     render : function(renderTo){
29729         // this is one sweet looking template!
29730         var tpl = new Roo.Template(
29731             '<table cellspacing="0" class="x-btn-menu-wrap x-btn"><tr><td>',
29732             '<table cellspacing="0" class="x-btn-wrap x-btn-menu-text-wrap"><tbody>',
29733             '<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>',
29734             "</tbody></table></td><td>",
29735             '<table cellspacing="0" class="x-btn-wrap x-btn-menu-arrow-wrap"><tbody>',
29736             '<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>',
29737             "</tbody></table></td></tr></table>"
29738         );
29739         var btn = tpl.append(renderTo, [this.text, this.type], true);
29740         var btnEl = btn.child("button");
29741         if(this.cls){
29742             btn.addClass(this.cls);
29743         }
29744         if(this.icon){
29745             btnEl.setStyle('background-image', 'url(' +this.icon +')');
29746         }
29747         if(this.iconCls){
29748             btnEl.addClass(this.iconCls);
29749             if(!this.cls){
29750                 btn.addClass(this.text ? 'x-btn-text-icon' : 'x-btn-icon');
29751             }
29752         }
29753         this.el = btn;
29754         if(this.handleMouseEvents){
29755             btn.on("mouseover", this.onMouseOver, this);
29756             btn.on("mouseout", this.onMouseOut, this);
29757             btn.on("mousedown", this.onMouseDown, this);
29758             btn.on("mouseup", this.onMouseUp, this);
29759         }
29760         btn.on(this.clickEvent, this.onClick, this);
29761         if(this.tooltip){
29762             if(typeof this.tooltip == 'object'){
29763                 Roo.QuickTips.tips(Roo.apply({
29764                       target: btnEl.id
29765                 }, this.tooltip));
29766             } else {
29767                 btnEl.dom[this.tooltipType] = this.tooltip;
29768             }
29769         }
29770         if(this.arrowTooltip){
29771             btn.child("button:nth(2)").dom[this.tooltipType] = this.arrowTooltip;
29772         }
29773         if(this.hidden){
29774             this.hide();
29775         }
29776         if(this.disabled){
29777             this.disable();
29778         }
29779         if(this.pressed){
29780             this.el.addClass("x-btn-pressed");
29781         }
29782         if(Roo.isIE && !Roo.isIE7){
29783             this.autoWidth.defer(1, this);
29784         }else{
29785             this.autoWidth();
29786         }
29787         if(this.menu){
29788             this.menu.on("show", this.onMenuShow, this);
29789             this.menu.on("hide", this.onMenuHide, this);
29790         }
29791         this.fireEvent('render', this);
29792     },
29793
29794     // private
29795     autoWidth : function(){
29796         if(this.el){
29797             var tbl = this.el.child("table:first");
29798             var tbl2 = this.el.child("table:last");
29799             this.el.setWidth("auto");
29800             tbl.setWidth("auto");
29801             if(Roo.isIE7 && Roo.isStrict){
29802                 var ib = this.el.child('button:first');
29803                 if(ib && ib.getWidth() > 20){
29804                     ib.clip();
29805                     ib.setWidth(Roo.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));
29806                 }
29807             }
29808             if(this.minWidth){
29809                 if(this.hidden){
29810                     this.el.beginMeasure();
29811                 }
29812                 if((tbl.getWidth()+tbl2.getWidth()) < this.minWidth){
29813                     tbl.setWidth(this.minWidth-tbl2.getWidth());
29814                 }
29815                 if(this.hidden){
29816                     this.el.endMeasure();
29817                 }
29818             }
29819             this.el.setWidth(tbl.getWidth()+tbl2.getWidth());
29820         } 
29821     },
29822     /**
29823      * Sets this button's click handler
29824      * @param {Function} handler The function to call when the button is clicked
29825      * @param {Object} scope (optional) Scope for the function passed above
29826      */
29827     setHandler : function(handler, scope){
29828         this.handler = handler;
29829         this.scope = scope;  
29830     },
29831     
29832     /**
29833      * Sets this button's arrow click handler
29834      * @param {Function} handler The function to call when the arrow is clicked
29835      * @param {Object} scope (optional) Scope for the function passed above
29836      */
29837     setArrowHandler : function(handler, scope){
29838         this.arrowHandler = handler;
29839         this.scope = scope;  
29840     },
29841     
29842     /**
29843      * Focus the button
29844      */
29845     focus : function(){
29846         if(this.el){
29847             this.el.child("button:first").focus();
29848         }
29849     },
29850
29851     // private
29852     onClick : function(e){
29853         e.preventDefault();
29854         if(!this.disabled){
29855             if(e.getTarget(".x-btn-menu-arrow-wrap")){
29856                 if(this.menu && !this.menu.isVisible()){
29857                     this.menu.show(this.el, this.menuAlign);
29858                 }
29859                 this.fireEvent("arrowclick", this, e);
29860                 if(this.arrowHandler){
29861                     this.arrowHandler.call(this.scope || this, this, e);
29862                 }
29863             }else{
29864                 this.fireEvent("click", this, e);
29865                 if(this.handler){
29866                     this.handler.call(this.scope || this, this, e);
29867                 }
29868             }
29869         }
29870     },
29871     // private
29872     onMouseDown : function(e){
29873         if(!this.disabled){
29874             Roo.fly(e.getTarget("table")).addClass("x-btn-click");
29875         }
29876     },
29877     // private
29878     onMouseUp : function(e){
29879         Roo.fly(e.getTarget("table")).removeClass("x-btn-click");
29880     }   
29881 });
29882
29883
29884 // backwards compat
29885 Roo.MenuButton = Roo.SplitButton;/*
29886  * Based on:
29887  * Ext JS Library 1.1.1
29888  * Copyright(c) 2006-2007, Ext JS, LLC.
29889  *
29890  * Originally Released Under LGPL - original licence link has changed is not relivant.
29891  *
29892  * Fork - LGPL
29893  * <script type="text/javascript">
29894  */
29895
29896 /**
29897  * @class Roo.Toolbar
29898  * Basic Toolbar class.
29899  * @constructor
29900  * Creates a new Toolbar
29901  * @param {Object} container The config object
29902  */ 
29903 Roo.Toolbar = function(container, buttons, config)
29904 {
29905     /// old consturctor format still supported..
29906     if(container instanceof Array){ // omit the container for later rendering
29907         buttons = container;
29908         config = buttons;
29909         container = null;
29910     }
29911     if (typeof(container) == 'object' && container.xtype) {
29912         config = container;
29913         container = config.container;
29914         buttons = config.buttons || []; // not really - use items!!
29915     }
29916     var xitems = [];
29917     if (config && config.items) {
29918         xitems = config.items;
29919         delete config.items;
29920     }
29921     Roo.apply(this, config);
29922     this.buttons = buttons;
29923     
29924     if(container){
29925         this.render(container);
29926     }
29927     this.xitems = xitems;
29928     Roo.each(xitems, function(b) {
29929         this.add(b);
29930     }, this);
29931     
29932 };
29933
29934 Roo.Toolbar.prototype = {
29935     /**
29936      * @cfg {Array} items
29937      * array of button configs or elements to add (will be converted to a MixedCollection)
29938      */
29939     
29940     /**
29941      * @cfg {String/HTMLElement/Element} container
29942      * The id or element that will contain the toolbar
29943      */
29944     // private
29945     render : function(ct){
29946         this.el = Roo.get(ct);
29947         if(this.cls){
29948             this.el.addClass(this.cls);
29949         }
29950         // using a table allows for vertical alignment
29951         // 100% width is needed by Safari...
29952         this.el.update('<div class="x-toolbar x-small-editor"><table cellspacing="0"><tr></tr></table></div>');
29953         this.tr = this.el.child("tr", true);
29954         var autoId = 0;
29955         this.items = new Roo.util.MixedCollection(false, function(o){
29956             return o.id || ("item" + (++autoId));
29957         });
29958         if(this.buttons){
29959             this.add.apply(this, this.buttons);
29960             delete this.buttons;
29961         }
29962     },
29963
29964     /**
29965      * Adds element(s) to the toolbar -- this function takes a variable number of 
29966      * arguments of mixed type and adds them to the toolbar.
29967      * @param {Mixed} arg1 The following types of arguments are all valid:<br />
29968      * <ul>
29969      * <li>{@link Roo.Toolbar.Button} config: A valid button config object (equivalent to {@link #addButton})</li>
29970      * <li>HtmlElement: Any standard HTML element (equivalent to {@link #addElement})</li>
29971      * <li>Field: Any form field (equivalent to {@link #addField})</li>
29972      * <li>Item: Any subclass of {@link Roo.Toolbar.Item} (equivalent to {@link #addItem})</li>
29973      * <li>String: Any generic string (gets wrapped in a {@link Roo.Toolbar.TextItem}, equivalent to {@link #addText}).
29974      * Note that there are a few special strings that are treated differently as explained nRoo.</li>
29975      * <li>'separator' or '-': Creates a separator element (equivalent to {@link #addSeparator})</li>
29976      * <li>' ': Creates a spacer element (equivalent to {@link #addSpacer})</li>
29977      * <li>'->': Creates a fill element (equivalent to {@link #addFill})</li>
29978      * </ul>
29979      * @param {Mixed} arg2
29980      * @param {Mixed} etc.
29981      */
29982     add : function(){
29983         var a = arguments, l = a.length;
29984         for(var i = 0; i < l; i++){
29985             this._add(a[i]);
29986         }
29987     },
29988     // private..
29989     _add : function(el) {
29990         
29991         if (el.xtype) {
29992             el = Roo.factory(el, typeof(Roo.Toolbar[el.xtype]) == 'undefined' ? Roo.form : Roo.Toolbar);
29993         }
29994         
29995         if (el.applyTo){ // some kind of form field
29996             return this.addField(el);
29997         } 
29998         if (el.render){ // some kind of Toolbar.Item
29999             return this.addItem(el);
30000         }
30001         if (typeof el == "string"){ // string
30002             if(el == "separator" || el == "-"){
30003                 return this.addSeparator();
30004             }
30005             if (el == " "){
30006                 return this.addSpacer();
30007             }
30008             if(el == "->"){
30009                 return this.addFill();
30010             }
30011             return this.addText(el);
30012             
30013         }
30014         if(el.tagName){ // element
30015             return this.addElement(el);
30016         }
30017         if(typeof el == "object"){ // must be button config?
30018             return this.addButton(el);
30019         }
30020         // and now what?!?!
30021         return false;
30022         
30023     },
30024     
30025     /**
30026      * Add an Xtype element
30027      * @param {Object} xtype Xtype Object
30028      * @return {Object} created Object
30029      */
30030     addxtype : function(e){
30031         return this.add(e);  
30032     },
30033     
30034     /**
30035      * Returns the Element for this toolbar.
30036      * @return {Roo.Element}
30037      */
30038     getEl : function(){
30039         return this.el;  
30040     },
30041     
30042     /**
30043      * Adds a separator
30044      * @return {Roo.Toolbar.Item} The separator item
30045      */
30046     addSeparator : function(){
30047         return this.addItem(new Roo.Toolbar.Separator());
30048     },
30049
30050     /**
30051      * Adds a spacer element
30052      * @return {Roo.Toolbar.Spacer} The spacer item
30053      */
30054     addSpacer : function(){
30055         return this.addItem(new Roo.Toolbar.Spacer());
30056     },
30057
30058     /**
30059      * Adds a fill element that forces subsequent additions to the right side of the toolbar
30060      * @return {Roo.Toolbar.Fill} The fill item
30061      */
30062     addFill : function(){
30063         return this.addItem(new Roo.Toolbar.Fill());
30064     },
30065
30066     /**
30067      * Adds any standard HTML element to the toolbar
30068      * @param {String/HTMLElement/Element} el The element or id of the element to add
30069      * @return {Roo.Toolbar.Item} The element's item
30070      */
30071     addElement : function(el){
30072         return this.addItem(new Roo.Toolbar.Item(el));
30073     },
30074     /**
30075      * Collection of items on the toolbar.. (only Toolbar Items, so use fields to retrieve fields)
30076      * @type Roo.util.MixedCollection  
30077      */
30078     items : false,
30079      
30080     /**
30081      * Adds any Toolbar.Item or subclass
30082      * @param {Roo.Toolbar.Item} item
30083      * @return {Roo.Toolbar.Item} The item
30084      */
30085     addItem : function(item){
30086         var td = this.nextBlock();
30087         item.render(td);
30088         this.items.add(item);
30089         return item;
30090     },
30091     
30092     /**
30093      * Adds a button (or buttons). See {@link Roo.Toolbar.Button} for more info on the config.
30094      * @param {Object/Array} config A button config or array of configs
30095      * @return {Roo.Toolbar.Button/Array}
30096      */
30097     addButton : function(config){
30098         if(config instanceof Array){
30099             var buttons = [];
30100             for(var i = 0, len = config.length; i < len; i++) {
30101                 buttons.push(this.addButton(config[i]));
30102             }
30103             return buttons;
30104         }
30105         var b = config;
30106         if(!(config instanceof Roo.Toolbar.Button)){
30107             b = config.split ?
30108                 new Roo.Toolbar.SplitButton(config) :
30109                 new Roo.Toolbar.Button(config);
30110         }
30111         var td = this.nextBlock();
30112         b.render(td);
30113         this.items.add(b);
30114         return b;
30115     },
30116     
30117     /**
30118      * Adds text to the toolbar
30119      * @param {String} text The text to add
30120      * @return {Roo.Toolbar.Item} The element's item
30121      */
30122     addText : function(text){
30123         return this.addItem(new Roo.Toolbar.TextItem(text));
30124     },
30125     
30126     /**
30127      * Inserts any {@link Roo.Toolbar.Item}/{@link Roo.Toolbar.Button} at the specified index.
30128      * @param {Number} index The index where the item is to be inserted
30129      * @param {Object/Roo.Toolbar.Item/Roo.Toolbar.Button (may be Array)} item The button, or button config object to be inserted.
30130      * @return {Roo.Toolbar.Button/Item}
30131      */
30132     insertButton : function(index, item){
30133         if(item instanceof Array){
30134             var buttons = [];
30135             for(var i = 0, len = item.length; i < len; i++) {
30136                buttons.push(this.insertButton(index + i, item[i]));
30137             }
30138             return buttons;
30139         }
30140         if (!(item instanceof Roo.Toolbar.Button)){
30141            item = new Roo.Toolbar.Button(item);
30142         }
30143         var td = document.createElement("td");
30144         this.tr.insertBefore(td, this.tr.childNodes[index]);
30145         item.render(td);
30146         this.items.insert(index, item);
30147         return item;
30148     },
30149     
30150     /**
30151      * Adds a new element to the toolbar from the passed {@link Roo.DomHelper} config.
30152      * @param {Object} config
30153      * @return {Roo.Toolbar.Item} The element's item
30154      */
30155     addDom : function(config, returnEl){
30156         var td = this.nextBlock();
30157         Roo.DomHelper.overwrite(td, config);
30158         var ti = new Roo.Toolbar.Item(td.firstChild);
30159         ti.render(td);
30160         this.items.add(ti);
30161         return ti;
30162     },
30163
30164     /**
30165      * Collection of fields on the toolbar.. usefull for quering (value is false if there are no fields)
30166      * @type Roo.util.MixedCollection  
30167      */
30168     fields : false,
30169     
30170     /**
30171      * Adds a dynamically rendered Roo.form field (TextField, ComboBox, etc).
30172      * Note: the field should not have been rendered yet. For a field that has already been
30173      * rendered, use {@link #addElement}.
30174      * @param {Roo.form.Field} field
30175      * @return {Roo.ToolbarItem}
30176      */
30177      
30178       
30179     addField : function(field) {
30180         if (!this.fields) {
30181             var autoId = 0;
30182             this.fields = new Roo.util.MixedCollection(false, function(o){
30183                 return o.id || ("item" + (++autoId));
30184             });
30185
30186         }
30187         
30188         var td = this.nextBlock();
30189         field.render(td);
30190         var ti = new Roo.Toolbar.Item(td.firstChild);
30191         ti.render(td);
30192         this.items.add(ti);
30193         this.fields.add(field);
30194         return ti;
30195     },
30196     /**
30197      * Hide the toolbar
30198      * @method hide
30199      */
30200      
30201       
30202     hide : function()
30203     {
30204         this.el.child('div').setVisibilityMode(Roo.Element.DISPLAY);
30205         this.el.child('div').hide();
30206     },
30207     /**
30208      * Show the toolbar
30209      * @method show
30210      */
30211     show : function()
30212     {
30213         this.el.child('div').show();
30214     },
30215       
30216     // private
30217     nextBlock : function(){
30218         var td = document.createElement("td");
30219         this.tr.appendChild(td);
30220         return td;
30221     },
30222
30223     // private
30224     destroy : function(){
30225         if(this.items){ // rendered?
30226             Roo.destroy.apply(Roo, this.items.items);
30227         }
30228         if(this.fields){ // rendered?
30229             Roo.destroy.apply(Roo, this.fields.items);
30230         }
30231         Roo.Element.uncache(this.el, this.tr);
30232     }
30233 };
30234
30235 /**
30236  * @class Roo.Toolbar.Item
30237  * The base class that other classes should extend in order to get some basic common toolbar item functionality.
30238  * @constructor
30239  * Creates a new Item
30240  * @param {HTMLElement} el 
30241  */
30242 Roo.Toolbar.Item = function(el){
30243     var cfg = {};
30244     if (typeof (el.xtype) != 'undefined') {
30245         cfg = el;
30246         el = cfg.el;
30247     }
30248     
30249     this.el = Roo.getDom(el);
30250     this.id = Roo.id(this.el);
30251     this.hidden = false;
30252     
30253     this.addEvents({
30254          /**
30255              * @event render
30256              * Fires when the button is rendered
30257              * @param {Button} this
30258              */
30259         'render': true
30260     });
30261     Roo.Toolbar.Item.superclass.constructor.call(this,cfg);
30262 };
30263 Roo.extend(Roo.Toolbar.Item, Roo.util.Observable, {
30264 //Roo.Toolbar.Item.prototype = {
30265     
30266     /**
30267      * Get this item's HTML Element
30268      * @return {HTMLElement}
30269      */
30270     getEl : function(){
30271        return this.el;  
30272     },
30273
30274     // private
30275     render : function(td){
30276         
30277          this.td = td;
30278         td.appendChild(this.el);
30279         
30280         this.fireEvent('render', this);
30281     },
30282     
30283     /**
30284      * Removes and destroys this item.
30285      */
30286     destroy : function(){
30287         this.td.parentNode.removeChild(this.td);
30288     },
30289     
30290     /**
30291      * Shows this item.
30292      */
30293     show: function(){
30294         this.hidden = false;
30295         this.td.style.display = "";
30296     },
30297     
30298     /**
30299      * Hides this item.
30300      */
30301     hide: function(){
30302         this.hidden = true;
30303         this.td.style.display = "none";
30304     },
30305     
30306     /**
30307      * Convenience function for boolean show/hide.
30308      * @param {Boolean} visible true to show/false to hide
30309      */
30310     setVisible: function(visible){
30311         if(visible) {
30312             this.show();
30313         }else{
30314             this.hide();
30315         }
30316     },
30317     
30318     /**
30319      * Try to focus this item.
30320      */
30321     focus : function(){
30322         Roo.fly(this.el).focus();
30323     },
30324     
30325     /**
30326      * Disables this item.
30327      */
30328     disable : function(){
30329         Roo.fly(this.td).addClass("x-item-disabled");
30330         this.disabled = true;
30331         this.el.disabled = true;
30332     },
30333     
30334     /**
30335      * Enables this item.
30336      */
30337     enable : function(){
30338         Roo.fly(this.td).removeClass("x-item-disabled");
30339         this.disabled = false;
30340         this.el.disabled = false;
30341     }
30342 });
30343
30344
30345 /**
30346  * @class Roo.Toolbar.Separator
30347  * @extends Roo.Toolbar.Item
30348  * A simple toolbar separator class
30349  * @constructor
30350  * Creates a new Separator
30351  */
30352 Roo.Toolbar.Separator = function(cfg){
30353     
30354     var s = document.createElement("span");
30355     s.className = "ytb-sep";
30356     if (cfg) {
30357         cfg.el = s;
30358     }
30359     
30360     Roo.Toolbar.Separator.superclass.constructor.call(this, cfg || s);
30361 };
30362 Roo.extend(Roo.Toolbar.Separator, Roo.Toolbar.Item, {
30363     enable:Roo.emptyFn,
30364     disable:Roo.emptyFn,
30365     focus:Roo.emptyFn
30366 });
30367
30368 /**
30369  * @class Roo.Toolbar.Spacer
30370  * @extends Roo.Toolbar.Item
30371  * A simple element that adds extra horizontal space to a toolbar.
30372  * @constructor
30373  * Creates a new Spacer
30374  */
30375 Roo.Toolbar.Spacer = function(cfg){
30376     var s = document.createElement("div");
30377     s.className = "ytb-spacer";
30378     if (cfg) {
30379         cfg.el = s;
30380     }
30381     Roo.Toolbar.Spacer.superclass.constructor.call(this, cfg || s);
30382 };
30383 Roo.extend(Roo.Toolbar.Spacer, Roo.Toolbar.Item, {
30384     enable:Roo.emptyFn,
30385     disable:Roo.emptyFn,
30386     focus:Roo.emptyFn
30387 });
30388
30389 /**
30390  * @class Roo.Toolbar.Fill
30391  * @extends Roo.Toolbar.Spacer
30392  * A simple element that adds a greedy (100% width) horizontal space to a toolbar.
30393  * @constructor
30394  * Creates a new Spacer
30395  */
30396 Roo.Toolbar.Fill = Roo.extend(Roo.Toolbar.Spacer, {
30397     // private
30398     render : function(td){
30399         td.style.width = '100%';
30400         Roo.Toolbar.Fill.superclass.render.call(this, td);
30401     }
30402 });
30403
30404 /**
30405  * @class Roo.Toolbar.TextItem
30406  * @extends Roo.Toolbar.Item
30407  * A simple class that renders text directly into a toolbar.
30408  * @constructor
30409  * Creates a new TextItem
30410  * @param {String} text
30411  */
30412 Roo.Toolbar.TextItem = function(cfg){
30413     var  text = cfg || "";
30414     if (typeof(cfg) == 'object') {
30415         text = cfg.text || "";
30416     }  else {
30417         cfg = null;
30418     }
30419     var s = document.createElement("span");
30420     s.className = "ytb-text";
30421     s.innerHTML = text;
30422     if (cfg) {
30423         cfg.el  = s;
30424     }
30425     
30426     Roo.Toolbar.TextItem.superclass.constructor.call(this, cfg ||  s);
30427 };
30428 Roo.extend(Roo.Toolbar.TextItem, Roo.Toolbar.Item, {
30429     
30430      
30431     enable:Roo.emptyFn,
30432     disable:Roo.emptyFn,
30433     focus:Roo.emptyFn
30434 });
30435
30436 /**
30437  * @class Roo.Toolbar.Button
30438  * @extends Roo.Button
30439  * A button that renders into a toolbar.
30440  * @constructor
30441  * Creates a new Button
30442  * @param {Object} config A standard {@link Roo.Button} config object
30443  */
30444 Roo.Toolbar.Button = function(config){
30445     Roo.Toolbar.Button.superclass.constructor.call(this, null, config);
30446 };
30447 Roo.extend(Roo.Toolbar.Button, Roo.Button, {
30448     render : function(td){
30449         this.td = td;
30450         Roo.Toolbar.Button.superclass.render.call(this, td);
30451     },
30452     
30453     /**
30454      * Removes and destroys this button
30455      */
30456     destroy : function(){
30457         Roo.Toolbar.Button.superclass.destroy.call(this);
30458         this.td.parentNode.removeChild(this.td);
30459     },
30460     
30461     /**
30462      * Shows this button
30463      */
30464     show: function(){
30465         this.hidden = false;
30466         this.td.style.display = "";
30467     },
30468     
30469     /**
30470      * Hides this button
30471      */
30472     hide: function(){
30473         this.hidden = true;
30474         this.td.style.display = "none";
30475     },
30476
30477     /**
30478      * Disables this item
30479      */
30480     disable : function(){
30481         Roo.fly(this.td).addClass("x-item-disabled");
30482         this.disabled = true;
30483     },
30484
30485     /**
30486      * Enables this item
30487      */
30488     enable : function(){
30489         Roo.fly(this.td).removeClass("x-item-disabled");
30490         this.disabled = false;
30491     }
30492 });
30493 // backwards compat
30494 Roo.ToolbarButton = Roo.Toolbar.Button;
30495
30496 /**
30497  * @class Roo.Toolbar.SplitButton
30498  * @extends Roo.SplitButton
30499  * A menu button that renders into a toolbar.
30500  * @constructor
30501  * Creates a new SplitButton
30502  * @param {Object} config A standard {@link Roo.SplitButton} config object
30503  */
30504 Roo.Toolbar.SplitButton = function(config){
30505     Roo.Toolbar.SplitButton.superclass.constructor.call(this, null, config);
30506 };
30507 Roo.extend(Roo.Toolbar.SplitButton, Roo.SplitButton, {
30508     render : function(td){
30509         this.td = td;
30510         Roo.Toolbar.SplitButton.superclass.render.call(this, td);
30511     },
30512     
30513     /**
30514      * Removes and destroys this button
30515      */
30516     destroy : function(){
30517         Roo.Toolbar.SplitButton.superclass.destroy.call(this);
30518         this.td.parentNode.removeChild(this.td);
30519     },
30520     
30521     /**
30522      * Shows this button
30523      */
30524     show: function(){
30525         this.hidden = false;
30526         this.td.style.display = "";
30527     },
30528     
30529     /**
30530      * Hides this button
30531      */
30532     hide: function(){
30533         this.hidden = true;
30534         this.td.style.display = "none";
30535     }
30536 });
30537
30538 // backwards compat
30539 Roo.Toolbar.MenuButton = Roo.Toolbar.SplitButton;/*
30540  * Based on:
30541  * Ext JS Library 1.1.1
30542  * Copyright(c) 2006-2007, Ext JS, LLC.
30543  *
30544  * Originally Released Under LGPL - original licence link has changed is not relivant.
30545  *
30546  * Fork - LGPL
30547  * <script type="text/javascript">
30548  */
30549  
30550 /**
30551  * @class Roo.PagingToolbar
30552  * @extends Roo.Toolbar
30553  * A specialized toolbar that is bound to a {@link Roo.data.Store} and provides automatic paging controls.
30554  * @constructor
30555  * Create a new PagingToolbar
30556  * @param {Object} config The config object
30557  */
30558 Roo.PagingToolbar = function(el, ds, config)
30559 {
30560     // old args format still supported... - xtype is prefered..
30561     if (typeof(el) == 'object' && el.xtype) {
30562         // created from xtype...
30563         config = el;
30564         ds = el.dataSource;
30565         el = config.container;
30566     }
30567     var items = [];
30568     if (config.items) {
30569         items = config.items;
30570         config.items = [];
30571     }
30572     
30573     Roo.PagingToolbar.superclass.constructor.call(this, el, null, config);
30574     this.ds = ds;
30575     this.cursor = 0;
30576     this.renderButtons(this.el);
30577     this.bind(ds);
30578     
30579     // supprot items array.
30580    
30581     Roo.each(items, function(e) {
30582         this.add(Roo.factory(e));
30583     },this);
30584     
30585 };
30586
30587 Roo.extend(Roo.PagingToolbar, Roo.Toolbar, {
30588     /**
30589      * @cfg {Roo.data.Store} dataSource
30590      * The underlying data store providing the paged data
30591      */
30592     /**
30593      * @cfg {String/HTMLElement/Element} container
30594      * container The id or element that will contain the toolbar
30595      */
30596     /**
30597      * @cfg {Boolean} displayInfo
30598      * True to display the displayMsg (defaults to false)
30599      */
30600     /**
30601      * @cfg {Number} pageSize
30602      * The number of records to display per page (defaults to 20)
30603      */
30604     pageSize: 20,
30605     /**
30606      * @cfg {String} displayMsg
30607      * The paging status message to display (defaults to "Displaying {start} - {end} of {total}")
30608      */
30609     displayMsg : 'Displaying {0} - {1} of {2}',
30610     /**
30611      * @cfg {String} emptyMsg
30612      * The message to display when no records are found (defaults to "No data to display")
30613      */
30614     emptyMsg : 'No data to display',
30615     /**
30616      * Customizable piece of the default paging text (defaults to "Page")
30617      * @type String
30618      */
30619     beforePageText : "Page",
30620     /**
30621      * Customizable piece of the default paging text (defaults to "of %0")
30622      * @type String
30623      */
30624     afterPageText : "of {0}",
30625     /**
30626      * Customizable piece of the default paging text (defaults to "First Page")
30627      * @type String
30628      */
30629     firstText : "First Page",
30630     /**
30631      * Customizable piece of the default paging text (defaults to "Previous Page")
30632      * @type String
30633      */
30634     prevText : "Previous Page",
30635     /**
30636      * Customizable piece of the default paging text (defaults to "Next Page")
30637      * @type String
30638      */
30639     nextText : "Next Page",
30640     /**
30641      * Customizable piece of the default paging text (defaults to "Last Page")
30642      * @type String
30643      */
30644     lastText : "Last Page",
30645     /**
30646      * Customizable piece of the default paging text (defaults to "Refresh")
30647      * @type String
30648      */
30649     refreshText : "Refresh",
30650
30651     // private
30652     renderButtons : function(el){
30653         Roo.PagingToolbar.superclass.render.call(this, el);
30654         this.first = this.addButton({
30655             tooltip: this.firstText,
30656             cls: "x-btn-icon x-grid-page-first",
30657             disabled: true,
30658             handler: this.onClick.createDelegate(this, ["first"])
30659         });
30660         this.prev = this.addButton({
30661             tooltip: this.prevText,
30662             cls: "x-btn-icon x-grid-page-prev",
30663             disabled: true,
30664             handler: this.onClick.createDelegate(this, ["prev"])
30665         });
30666         //this.addSeparator();
30667         this.add(this.beforePageText);
30668         this.field = Roo.get(this.addDom({
30669            tag: "input",
30670            type: "text",
30671            size: "3",
30672            value: "1",
30673            cls: "x-grid-page-number"
30674         }).el);
30675         this.field.on("keydown", this.onPagingKeydown, this);
30676         this.field.on("focus", function(){this.dom.select();});
30677         this.afterTextEl = this.addText(String.format(this.afterPageText, 1));
30678         this.field.setHeight(18);
30679         //this.addSeparator();
30680         this.next = this.addButton({
30681             tooltip: this.nextText,
30682             cls: "x-btn-icon x-grid-page-next",
30683             disabled: true,
30684             handler: this.onClick.createDelegate(this, ["next"])
30685         });
30686         this.last = this.addButton({
30687             tooltip: this.lastText,
30688             cls: "x-btn-icon x-grid-page-last",
30689             disabled: true,
30690             handler: this.onClick.createDelegate(this, ["last"])
30691         });
30692         //this.addSeparator();
30693         this.loading = this.addButton({
30694             tooltip: this.refreshText,
30695             cls: "x-btn-icon x-grid-loading",
30696             handler: this.onClick.createDelegate(this, ["refresh"])
30697         });
30698
30699         if(this.displayInfo){
30700             this.displayEl = Roo.fly(this.el.dom.firstChild).createChild({cls:'x-paging-info'});
30701         }
30702     },
30703
30704     // private
30705     updateInfo : function(){
30706         if(this.displayEl){
30707             var count = this.ds.getCount();
30708             var msg = count == 0 ?
30709                 this.emptyMsg :
30710                 String.format(
30711                     this.displayMsg,
30712                     this.cursor+1, this.cursor+count, this.ds.getTotalCount()    
30713                 );
30714             this.displayEl.update(msg);
30715         }
30716     },
30717
30718     // private
30719     onLoad : function(ds, r, o){
30720        this.cursor = o.params ? o.params.start : 0;
30721        var d = this.getPageData(), ap = d.activePage, ps = d.pages;
30722
30723        this.afterTextEl.el.innerHTML = String.format(this.afterPageText, d.pages);
30724        this.field.dom.value = ap;
30725        this.first.setDisabled(ap == 1);
30726        this.prev.setDisabled(ap == 1);
30727        this.next.setDisabled(ap == ps);
30728        this.last.setDisabled(ap == ps);
30729        this.loading.enable();
30730        this.updateInfo();
30731     },
30732
30733     // private
30734     getPageData : function(){
30735         var total = this.ds.getTotalCount();
30736         return {
30737             total : total,
30738             activePage : Math.ceil((this.cursor+this.pageSize)/this.pageSize),
30739             pages :  total < this.pageSize ? 1 : Math.ceil(total/this.pageSize)
30740         };
30741     },
30742
30743     // private
30744     onLoadError : function(){
30745         this.loading.enable();
30746     },
30747
30748     // private
30749     onPagingKeydown : function(e){
30750         var k = e.getKey();
30751         var d = this.getPageData();
30752         if(k == e.RETURN){
30753             var v = this.field.dom.value, pageNum;
30754             if(!v || isNaN(pageNum = parseInt(v, 10))){
30755                 this.field.dom.value = d.activePage;
30756                 return;
30757             }
30758             pageNum = Math.min(Math.max(1, pageNum), d.pages) - 1;
30759             this.ds.load({params:{start: pageNum * this.pageSize, limit: this.pageSize}});
30760             e.stopEvent();
30761         }
30762         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))
30763         {
30764           var pageNum = (k == e.HOME || (k == e.DOWN && e.ctrlKey) || (k == e.LEFT && e.ctrlKey) || (k == e.PAGEDOWN && e.ctrlKey)) ? 1 : d.pages;
30765           this.field.dom.value = pageNum;
30766           this.ds.load({params:{start: (pageNum - 1) * this.pageSize, limit: this.pageSize}});
30767           e.stopEvent();
30768         }
30769         else if(k == e.UP || k == e.RIGHT || k == e.PAGEUP || k == e.DOWN || k == e.LEFT || k == e.PAGEDOWN)
30770         {
30771           var v = this.field.dom.value, pageNum; 
30772           var increment = (e.shiftKey) ? 10 : 1;
30773           if(k == e.DOWN || k == e.LEFT || k == e.PAGEDOWN) {
30774             increment *= -1;
30775           }
30776           if(!v || isNaN(pageNum = parseInt(v, 10))) {
30777             this.field.dom.value = d.activePage;
30778             return;
30779           }
30780           else if(parseInt(v, 10) + increment >= 1 & parseInt(v, 10) + increment <= d.pages)
30781           {
30782             this.field.dom.value = parseInt(v, 10) + increment;
30783             pageNum = Math.min(Math.max(1, pageNum + increment), d.pages) - 1;
30784             this.ds.load({params:{start: pageNum * this.pageSize, limit: this.pageSize}});
30785           }
30786           e.stopEvent();
30787         }
30788     },
30789
30790     // private
30791     beforeLoad : function(){
30792         if(this.loading){
30793             this.loading.disable();
30794         }
30795     },
30796
30797     // private
30798     onClick : function(which){
30799         var ds = this.ds;
30800         switch(which){
30801             case "first":
30802                 ds.load({params:{start: 0, limit: this.pageSize}});
30803             break;
30804             case "prev":
30805                 ds.load({params:{start: Math.max(0, this.cursor-this.pageSize), limit: this.pageSize}});
30806             break;
30807             case "next":
30808                 ds.load({params:{start: this.cursor+this.pageSize, limit: this.pageSize}});
30809             break;
30810             case "last":
30811                 var total = ds.getTotalCount();
30812                 var extra = total % this.pageSize;
30813                 var lastStart = extra ? (total - extra) : total-this.pageSize;
30814                 ds.load({params:{start: lastStart, limit: this.pageSize}});
30815             break;
30816             case "refresh":
30817                 ds.load({params:{start: this.cursor, limit: this.pageSize}});
30818             break;
30819         }
30820     },
30821
30822     /**
30823      * Unbinds the paging toolbar from the specified {@link Roo.data.Store}
30824      * @param {Roo.data.Store} store The data store to unbind
30825      */
30826     unbind : function(ds){
30827         ds.un("beforeload", this.beforeLoad, this);
30828         ds.un("load", this.onLoad, this);
30829         ds.un("loadexception", this.onLoadError, this);
30830         ds.un("remove", this.updateInfo, this);
30831         ds.un("add", this.updateInfo, this);
30832         this.ds = undefined;
30833     },
30834
30835     /**
30836      * Binds the paging toolbar to the specified {@link Roo.data.Store}
30837      * @param {Roo.data.Store} store The data store to bind
30838      */
30839     bind : function(ds){
30840         ds.on("beforeload", this.beforeLoad, this);
30841         ds.on("load", this.onLoad, this);
30842         ds.on("loadexception", this.onLoadError, this);
30843         ds.on("remove", this.updateInfo, this);
30844         ds.on("add", this.updateInfo, this);
30845         this.ds = ds;
30846     }
30847 });/*
30848  * Based on:
30849  * Ext JS Library 1.1.1
30850  * Copyright(c) 2006-2007, Ext JS, LLC.
30851  *
30852  * Originally Released Under LGPL - original licence link has changed is not relivant.
30853  *
30854  * Fork - LGPL
30855  * <script type="text/javascript">
30856  */
30857
30858 /**
30859  * @class Roo.Resizable
30860  * @extends Roo.util.Observable
30861  * <p>Applies drag handles to an element to make it resizable. The drag handles are inserted into the element
30862  * and positioned absolute. Some elements, such as a textarea or image, don't support this. To overcome that, you can wrap
30863  * 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
30864  * the element will be wrapped for you automatically.</p>
30865  * <p>Here is the list of valid resize handles:</p>
30866  * <pre>
30867 Value   Description
30868 ------  -------------------
30869  'n'     north
30870  's'     south
30871  'e'     east
30872  'w'     west
30873  'nw'    northwest
30874  'sw'    southwest
30875  'se'    southeast
30876  'ne'    northeast
30877  'hd'    horizontal drag
30878  'all'   all
30879 </pre>
30880  * <p>Here's an example showing the creation of a typical Resizable:</p>
30881  * <pre><code>
30882 var resizer = new Roo.Resizable("element-id", {
30883     handles: 'all',
30884     minWidth: 200,
30885     minHeight: 100,
30886     maxWidth: 500,
30887     maxHeight: 400,
30888     pinned: true
30889 });
30890 resizer.on("resize", myHandler);
30891 </code></pre>
30892  * <p>To hide a particular handle, set its display to none in CSS, or through script:<br>
30893  * resizer.east.setDisplayed(false);</p>
30894  * @cfg {Boolean/String/Element} resizeChild True to resize the first child, or id/element to resize (defaults to false)
30895  * @cfg {Array/String} adjustments String "auto" or an array [width, height] with values to be <b>added</b> to the
30896  * resize operation's new size (defaults to [0, 0])
30897  * @cfg {Number} minWidth The minimum width for the element (defaults to 5)
30898  * @cfg {Number} minHeight The minimum height for the element (defaults to 5)
30899  * @cfg {Number} maxWidth The maximum width for the element (defaults to 10000)
30900  * @cfg {Number} maxHeight The maximum height for the element (defaults to 10000)
30901  * @cfg {Boolean} enabled False to disable resizing (defaults to true)
30902  * @cfg {Boolean} wrap True to wrap an element with a div if needed (required for textareas and images, defaults to false)
30903  * @cfg {Number} width The width of the element in pixels (defaults to null)
30904  * @cfg {Number} height The height of the element in pixels (defaults to null)
30905  * @cfg {Boolean} animate True to animate the resize (not compatible with dynamic sizing, defaults to false)
30906  * @cfg {Number} duration Animation duration if animate = true (defaults to .35)
30907  * @cfg {Boolean} dynamic True to resize the element while dragging instead of using a proxy (defaults to false)
30908  * @cfg {String} handles String consisting of the resize handles to display (defaults to undefined)
30909  * @cfg {Boolean} multiDirectional <b>Deprecated</b>.  The old style of adding multi-direction resize handles, deprecated
30910  * in favor of the handles config option (defaults to false)
30911  * @cfg {Boolean} disableTrackOver True to disable mouse tracking. This is only applied at config time. (defaults to false)
30912  * @cfg {String} easing Animation easing if animate = true (defaults to 'easingOutStrong')
30913  * @cfg {Number} widthIncrement The increment to snap the width resize in pixels (dynamic must be true, defaults to 0)
30914  * @cfg {Number} heightIncrement The increment to snap the height resize in pixels (dynamic must be true, defaults to 0)
30915  * @cfg {Boolean} pinned True to ensure that the resize handles are always visible, false to display them only when the
30916  * user mouses over the resizable borders. This is only applied at config time. (defaults to false)
30917  * @cfg {Boolean} preserveRatio True to preserve the original ratio between height and width during resize (defaults to false)
30918  * @cfg {Boolean} transparent True for transparent handles. This is only applied at config time. (defaults to false)
30919  * @cfg {Number} minX The minimum allowed page X for the element (only used for west resizing, defaults to 0)
30920  * @cfg {Number} minY The minimum allowed page Y for the element (only used for north resizing, defaults to 0)
30921  * @cfg {Boolean} draggable Convenience to initialize drag drop (defaults to false)
30922  * @constructor
30923  * Create a new resizable component
30924  * @param {String/HTMLElement/Roo.Element} el The id or element to resize
30925  * @param {Object} config configuration options
30926   */
30927 Roo.Resizable = function(el, config)
30928 {
30929     this.el = Roo.get(el);
30930
30931     if(config && config.wrap){
30932         config.resizeChild = this.el;
30933         this.el = this.el.wrap(typeof config.wrap == "object" ? config.wrap : {cls:"xresizable-wrap"});
30934         this.el.id = this.el.dom.id = config.resizeChild.id + "-rzwrap";
30935         this.el.setStyle("overflow", "hidden");
30936         this.el.setPositioning(config.resizeChild.getPositioning());
30937         config.resizeChild.clearPositioning();
30938         if(!config.width || !config.height){
30939             var csize = config.resizeChild.getSize();
30940             this.el.setSize(csize.width, csize.height);
30941         }
30942         if(config.pinned && !config.adjustments){
30943             config.adjustments = "auto";
30944         }
30945     }
30946
30947     this.proxy = this.el.createProxy({tag: "div", cls: "x-resizable-proxy", id: this.el.id + "-rzproxy"});
30948     this.proxy.unselectable();
30949     this.proxy.enableDisplayMode('block');
30950
30951     Roo.apply(this, config);
30952
30953     if(this.pinned){
30954         this.disableTrackOver = true;
30955         this.el.addClass("x-resizable-pinned");
30956     }
30957     // if the element isn't positioned, make it relative
30958     var position = this.el.getStyle("position");
30959     if(position != "absolute" && position != "fixed"){
30960         this.el.setStyle("position", "relative");
30961     }
30962     if(!this.handles){ // no handles passed, must be legacy style
30963         this.handles = 's,e,se';
30964         if(this.multiDirectional){
30965             this.handles += ',n,w';
30966         }
30967     }
30968     if(this.handles == "all"){
30969         this.handles = "n s e w ne nw se sw";
30970     }
30971     var hs = this.handles.split(/\s*?[,;]\s*?| /);
30972     var ps = Roo.Resizable.positions;
30973     for(var i = 0, len = hs.length; i < len; i++){
30974         if(hs[i] && ps[hs[i]]){
30975             var pos = ps[hs[i]];
30976             this[pos] = new Roo.Resizable.Handle(this, pos, this.disableTrackOver, this.transparent);
30977         }
30978     }
30979     // legacy
30980     this.corner = this.southeast;
30981     
30982     // updateBox = the box can move..
30983     if(this.handles.indexOf("n") != -1 || this.handles.indexOf("w") != -1 || this.handles.indexOf("hd") != -1) {
30984         this.updateBox = true;
30985     }
30986
30987     this.activeHandle = null;
30988
30989     if(this.resizeChild){
30990         if(typeof this.resizeChild == "boolean"){
30991             this.resizeChild = Roo.get(this.el.dom.firstChild, true);
30992         }else{
30993             this.resizeChild = Roo.get(this.resizeChild, true);
30994         }
30995     }
30996     
30997     if(this.adjustments == "auto"){
30998         var rc = this.resizeChild;
30999         var hw = this.west, he = this.east, hn = this.north, hs = this.south;
31000         if(rc && (hw || hn)){
31001             rc.position("relative");
31002             rc.setLeft(hw ? hw.el.getWidth() : 0);
31003             rc.setTop(hn ? hn.el.getHeight() : 0);
31004         }
31005         this.adjustments = [
31006             (he ? -he.el.getWidth() : 0) + (hw ? -hw.el.getWidth() : 0),
31007             (hn ? -hn.el.getHeight() : 0) + (hs ? -hs.el.getHeight() : 0) -1
31008         ];
31009     }
31010
31011     if(this.draggable){
31012         this.dd = this.dynamic ?
31013             this.el.initDD(null) : this.el.initDDProxy(null, {dragElId: this.proxy.id});
31014         this.dd.setHandleElId(this.resizeChild ? this.resizeChild.id : this.el.id);
31015     }
31016
31017     // public events
31018     this.addEvents({
31019         /**
31020          * @event beforeresize
31021          * Fired before resize is allowed. Set enabled to false to cancel resize.
31022          * @param {Roo.Resizable} this
31023          * @param {Roo.EventObject} e The mousedown event
31024          */
31025         "beforeresize" : true,
31026         /**
31027          * @event resizing
31028          * Fired a resizing.
31029          * @param {Roo.Resizable} this
31030          * @param {Number} x The new x position
31031          * @param {Number} y The new y position
31032          * @param {Number} w The new w width
31033          * @param {Number} h The new h hight
31034          * @param {Roo.EventObject} e The mouseup event
31035          */
31036         "resizing" : true,
31037         /**
31038          * @event resize
31039          * Fired after a resize.
31040          * @param {Roo.Resizable} this
31041          * @param {Number} width The new width
31042          * @param {Number} height The new height
31043          * @param {Roo.EventObject} e The mouseup event
31044          */
31045         "resize" : true
31046     });
31047
31048     if(this.width !== null && this.height !== null){
31049         this.resizeTo(this.width, this.height);
31050     }else{
31051         this.updateChildSize();
31052     }
31053     if(Roo.isIE){
31054         this.el.dom.style.zoom = 1;
31055     }
31056     Roo.Resizable.superclass.constructor.call(this);
31057 };
31058
31059 Roo.extend(Roo.Resizable, Roo.util.Observable, {
31060         resizeChild : false,
31061         adjustments : [0, 0],
31062         minWidth : 5,
31063         minHeight : 5,
31064         maxWidth : 10000,
31065         maxHeight : 10000,
31066         enabled : true,
31067         animate : false,
31068         duration : .35,
31069         dynamic : false,
31070         handles : false,
31071         multiDirectional : false,
31072         disableTrackOver : false,
31073         easing : 'easeOutStrong',
31074         widthIncrement : 0,
31075         heightIncrement : 0,
31076         pinned : false,
31077         width : null,
31078         height : null,
31079         preserveRatio : false,
31080         transparent: false,
31081         minX: 0,
31082         minY: 0,
31083         draggable: false,
31084
31085         /**
31086          * @cfg {String/HTMLElement/Element} constrainTo Constrain the resize to a particular element
31087          */
31088         constrainTo: undefined,
31089         /**
31090          * @cfg {Roo.lib.Region} resizeRegion Constrain the resize to a particular region
31091          */
31092         resizeRegion: undefined,
31093
31094
31095     /**
31096      * Perform a manual resize
31097      * @param {Number} width
31098      * @param {Number} height
31099      */
31100     resizeTo : function(width, height){
31101         this.el.setSize(width, height);
31102         this.updateChildSize();
31103         this.fireEvent("resize", this, width, height, null);
31104     },
31105
31106     // private
31107     startSizing : function(e, handle){
31108         this.fireEvent("beforeresize", this, e);
31109         if(this.enabled){ // 2nd enabled check in case disabled before beforeresize handler
31110
31111             if(!this.overlay){
31112                 this.overlay = this.el.createProxy({tag: "div", cls: "x-resizable-overlay", html: "&#160;"});
31113                 this.overlay.unselectable();
31114                 this.overlay.enableDisplayMode("block");
31115                 this.overlay.on("mousemove", this.onMouseMove, this);
31116                 this.overlay.on("mouseup", this.onMouseUp, this);
31117             }
31118             this.overlay.setStyle("cursor", handle.el.getStyle("cursor"));
31119
31120             this.resizing = true;
31121             this.startBox = this.el.getBox();
31122             this.startPoint = e.getXY();
31123             this.offsets = [(this.startBox.x + this.startBox.width) - this.startPoint[0],
31124                             (this.startBox.y + this.startBox.height) - this.startPoint[1]];
31125
31126             this.overlay.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
31127             this.overlay.show();
31128
31129             if(this.constrainTo) {
31130                 var ct = Roo.get(this.constrainTo);
31131                 this.resizeRegion = ct.getRegion().adjust(
31132                     ct.getFrameWidth('t'),
31133                     ct.getFrameWidth('l'),
31134                     -ct.getFrameWidth('b'),
31135                     -ct.getFrameWidth('r')
31136                 );
31137             }
31138
31139             this.proxy.setStyle('visibility', 'hidden'); // workaround display none
31140             this.proxy.show();
31141             this.proxy.setBox(this.startBox);
31142             if(!this.dynamic){
31143                 this.proxy.setStyle('visibility', 'visible');
31144             }
31145         }
31146     },
31147
31148     // private
31149     onMouseDown : function(handle, e){
31150         if(this.enabled){
31151             e.stopEvent();
31152             this.activeHandle = handle;
31153             this.startSizing(e, handle);
31154         }
31155     },
31156
31157     // private
31158     onMouseUp : function(e){
31159         var size = this.resizeElement();
31160         this.resizing = false;
31161         this.handleOut();
31162         this.overlay.hide();
31163         this.proxy.hide();
31164         this.fireEvent("resize", this, size.width, size.height, e);
31165     },
31166
31167     // private
31168     updateChildSize : function(){
31169         
31170         if(this.resizeChild){
31171             var el = this.el;
31172             var child = this.resizeChild;
31173             var adj = this.adjustments;
31174             if(el.dom.offsetWidth){
31175                 var b = el.getSize(true);
31176                 child.setSize(b.width+adj[0], b.height+adj[1]);
31177             }
31178             // Second call here for IE
31179             // The first call enables instant resizing and
31180             // the second call corrects scroll bars if they
31181             // exist
31182             if(Roo.isIE){
31183                 setTimeout(function(){
31184                     if(el.dom.offsetWidth){
31185                         var b = el.getSize(true);
31186                         child.setSize(b.width+adj[0], b.height+adj[1]);
31187                     }
31188                 }, 10);
31189             }
31190         }
31191     },
31192
31193     // private
31194     snap : function(value, inc, min){
31195         if(!inc || !value) {
31196             return value;
31197         }
31198         var newValue = value;
31199         var m = value % inc;
31200         if(m > 0){
31201             if(m > (inc/2)){
31202                 newValue = value + (inc-m);
31203             }else{
31204                 newValue = value - m;
31205             }
31206         }
31207         return Math.max(min, newValue);
31208     },
31209
31210     // private
31211     resizeElement : function(){
31212         var box = this.proxy.getBox();
31213         if(this.updateBox){
31214             this.el.setBox(box, false, this.animate, this.duration, null, this.easing);
31215         }else{
31216             this.el.setSize(box.width, box.height, this.animate, this.duration, null, this.easing);
31217         }
31218         this.updateChildSize();
31219         if(!this.dynamic){
31220             this.proxy.hide();
31221         }
31222         return box;
31223     },
31224
31225     // private
31226     constrain : function(v, diff, m, mx){
31227         if(v - diff < m){
31228             diff = v - m;
31229         }else if(v - diff > mx){
31230             diff = mx - v;
31231         }
31232         return diff;
31233     },
31234
31235     // private
31236     onMouseMove : function(e){
31237         
31238         if(this.enabled){
31239             try{// try catch so if something goes wrong the user doesn't get hung
31240
31241             if(this.resizeRegion && !this.resizeRegion.contains(e.getPoint())) {
31242                 return;
31243             }
31244
31245             //var curXY = this.startPoint;
31246             var curSize = this.curSize || this.startBox;
31247             var x = this.startBox.x, y = this.startBox.y;
31248             var ox = x, oy = y;
31249             var w = curSize.width, h = curSize.height;
31250             var ow = w, oh = h;
31251             var mw = this.minWidth, mh = this.minHeight;
31252             var mxw = this.maxWidth, mxh = this.maxHeight;
31253             var wi = this.widthIncrement;
31254             var hi = this.heightIncrement;
31255
31256             var eventXY = e.getXY();
31257             var diffX = -(this.startPoint[0] - Math.max(this.minX, eventXY[0]));
31258             var diffY = -(this.startPoint[1] - Math.max(this.minY, eventXY[1]));
31259
31260             var pos = this.activeHandle.position;
31261
31262             switch(pos){
31263                 case "east":
31264                     w += diffX;
31265                     w = Math.min(Math.max(mw, w), mxw);
31266                     break;
31267              
31268                 case "south":
31269                     h += diffY;
31270                     h = Math.min(Math.max(mh, h), mxh);
31271                     break;
31272                 case "southeast":
31273                     w += diffX;
31274                     h += diffY;
31275                     w = Math.min(Math.max(mw, w), mxw);
31276                     h = Math.min(Math.max(mh, h), mxh);
31277                     break;
31278                 case "north":
31279                     diffY = this.constrain(h, diffY, mh, mxh);
31280                     y += diffY;
31281                     h -= diffY;
31282                     break;
31283                 case "hdrag":
31284                     
31285                     if (wi) {
31286                         var adiffX = Math.abs(diffX);
31287                         var sub = (adiffX % wi); // how much 
31288                         if (sub > (wi/2)) { // far enough to snap
31289                             diffX = (diffX > 0) ? diffX-sub + wi : diffX+sub - wi;
31290                         } else {
31291                             // remove difference.. 
31292                             diffX = (diffX > 0) ? diffX-sub : diffX+sub;
31293                         }
31294                     }
31295                     x += diffX;
31296                     x = Math.max(this.minX, x);
31297                     break;
31298                 case "west":
31299                     diffX = this.constrain(w, diffX, mw, mxw);
31300                     x += diffX;
31301                     w -= diffX;
31302                     break;
31303                 case "northeast":
31304                     w += diffX;
31305                     w = Math.min(Math.max(mw, w), mxw);
31306                     diffY = this.constrain(h, diffY, mh, mxh);
31307                     y += diffY;
31308                     h -= diffY;
31309                     break;
31310                 case "northwest":
31311                     diffX = this.constrain(w, diffX, mw, mxw);
31312                     diffY = this.constrain(h, diffY, mh, mxh);
31313                     y += diffY;
31314                     h -= diffY;
31315                     x += diffX;
31316                     w -= diffX;
31317                     break;
31318                case "southwest":
31319                     diffX = this.constrain(w, diffX, mw, mxw);
31320                     h += diffY;
31321                     h = Math.min(Math.max(mh, h), mxh);
31322                     x += diffX;
31323                     w -= diffX;
31324                     break;
31325             }
31326
31327             var sw = this.snap(w, wi, mw);
31328             var sh = this.snap(h, hi, mh);
31329             if(sw != w || sh != h){
31330                 switch(pos){
31331                     case "northeast":
31332                         y -= sh - h;
31333                     break;
31334                     case "north":
31335                         y -= sh - h;
31336                         break;
31337                     case "southwest":
31338                         x -= sw - w;
31339                     break;
31340                     case "west":
31341                         x -= sw - w;
31342                         break;
31343                     case "northwest":
31344                         x -= sw - w;
31345                         y -= sh - h;
31346                     break;
31347                 }
31348                 w = sw;
31349                 h = sh;
31350             }
31351
31352             if(this.preserveRatio){
31353                 switch(pos){
31354                     case "southeast":
31355                     case "east":
31356                         h = oh * (w/ow);
31357                         h = Math.min(Math.max(mh, h), mxh);
31358                         w = ow * (h/oh);
31359                        break;
31360                     case "south":
31361                         w = ow * (h/oh);
31362                         w = Math.min(Math.max(mw, w), mxw);
31363                         h = oh * (w/ow);
31364                         break;
31365                     case "northeast":
31366                         w = ow * (h/oh);
31367                         w = Math.min(Math.max(mw, w), mxw);
31368                         h = oh * (w/ow);
31369                     break;
31370                     case "north":
31371                         var tw = w;
31372                         w = ow * (h/oh);
31373                         w = Math.min(Math.max(mw, w), mxw);
31374                         h = oh * (w/ow);
31375                         x += (tw - w) / 2;
31376                         break;
31377                     case "southwest":
31378                         h = oh * (w/ow);
31379                         h = Math.min(Math.max(mh, h), mxh);
31380                         var tw = w;
31381                         w = ow * (h/oh);
31382                         x += tw - w;
31383                         break;
31384                     case "west":
31385                         var th = h;
31386                         h = oh * (w/ow);
31387                         h = Math.min(Math.max(mh, h), mxh);
31388                         y += (th - h) / 2;
31389                         var tw = w;
31390                         w = ow * (h/oh);
31391                         x += tw - w;
31392                        break;
31393                     case "northwest":
31394                         var tw = w;
31395                         var th = h;
31396                         h = oh * (w/ow);
31397                         h = Math.min(Math.max(mh, h), mxh);
31398                         w = ow * (h/oh);
31399                         y += th - h;
31400                         x += tw - w;
31401                        break;
31402
31403                 }
31404             }
31405             if (pos == 'hdrag') {
31406                 w = ow;
31407             }
31408             this.proxy.setBounds(x, y, w, h);
31409             if(this.dynamic){
31410                 this.resizeElement();
31411             }
31412             }catch(e){}
31413         }
31414         this.fireEvent("resizing", this, x, y, w, h, e);
31415     },
31416
31417     // private
31418     handleOver : function(){
31419         if(this.enabled){
31420             this.el.addClass("x-resizable-over");
31421         }
31422     },
31423
31424     // private
31425     handleOut : function(){
31426         if(!this.resizing){
31427             this.el.removeClass("x-resizable-over");
31428         }
31429     },
31430
31431     /**
31432      * Returns the element this component is bound to.
31433      * @return {Roo.Element}
31434      */
31435     getEl : function(){
31436         return this.el;
31437     },
31438
31439     /**
31440      * Returns the resizeChild element (or null).
31441      * @return {Roo.Element}
31442      */
31443     getResizeChild : function(){
31444         return this.resizeChild;
31445     },
31446     groupHandler : function()
31447     {
31448         
31449     },
31450     /**
31451      * Destroys this resizable. If the element was wrapped and
31452      * removeEl is not true then the element remains.
31453      * @param {Boolean} removeEl (optional) true to remove the element from the DOM
31454      */
31455     destroy : function(removeEl){
31456         this.proxy.remove();
31457         if(this.overlay){
31458             this.overlay.removeAllListeners();
31459             this.overlay.remove();
31460         }
31461         var ps = Roo.Resizable.positions;
31462         for(var k in ps){
31463             if(typeof ps[k] != "function" && this[ps[k]]){
31464                 var h = this[ps[k]];
31465                 h.el.removeAllListeners();
31466                 h.el.remove();
31467             }
31468         }
31469         if(removeEl){
31470             this.el.update("");
31471             this.el.remove();
31472         }
31473     }
31474 });
31475
31476 // private
31477 // hash to map config positions to true positions
31478 Roo.Resizable.positions = {
31479     n: "north", s: "south", e: "east", w: "west", se: "southeast", sw: "southwest", nw: "northwest", ne: "northeast", 
31480     hd: "hdrag"
31481 };
31482
31483 // private
31484 Roo.Resizable.Handle = function(rz, pos, disableTrackOver, transparent){
31485     if(!this.tpl){
31486         // only initialize the template if resizable is used
31487         var tpl = Roo.DomHelper.createTemplate(
31488             {tag: "div", cls: "x-resizable-handle x-resizable-handle-{0}"}
31489         );
31490         tpl.compile();
31491         Roo.Resizable.Handle.prototype.tpl = tpl;
31492     }
31493     this.position = pos;
31494     this.rz = rz;
31495     // show north drag fro topdra
31496     var handlepos = pos == 'hdrag' ? 'north' : pos;
31497     
31498     this.el = this.tpl.append(rz.el.dom, [handlepos], true);
31499     if (pos == 'hdrag') {
31500         this.el.setStyle('cursor', 'pointer');
31501     }
31502     this.el.unselectable();
31503     if(transparent){
31504         this.el.setOpacity(0);
31505     }
31506     this.el.on("mousedown", this.onMouseDown, this);
31507     if(!disableTrackOver){
31508         this.el.on("mouseover", this.onMouseOver, this);
31509         this.el.on("mouseout", this.onMouseOut, this);
31510     }
31511 };
31512
31513 // private
31514 Roo.Resizable.Handle.prototype = {
31515     afterResize : function(rz){
31516         Roo.log('after?');
31517         // do nothing
31518     },
31519     // private
31520     onMouseDown : function(e){
31521         this.rz.onMouseDown(this, e);
31522     },
31523     // private
31524     onMouseOver : function(e){
31525         this.rz.handleOver(this, e);
31526     },
31527     // private
31528     onMouseOut : function(e){
31529         this.rz.handleOut(this, e);
31530     }
31531 };/*
31532  * Based on:
31533  * Ext JS Library 1.1.1
31534  * Copyright(c) 2006-2007, Ext JS, LLC.
31535  *
31536  * Originally Released Under LGPL - original licence link has changed is not relivant.
31537  *
31538  * Fork - LGPL
31539  * <script type="text/javascript">
31540  */
31541
31542 /**
31543  * @class Roo.Editor
31544  * @extends Roo.Component
31545  * A base editor field that handles displaying/hiding on demand and has some built-in sizing and event handling logic.
31546  * @constructor
31547  * Create a new Editor
31548  * @param {Roo.form.Field} field The Field object (or descendant)
31549  * @param {Object} config The config object
31550  */
31551 Roo.Editor = function(field, config){
31552     Roo.Editor.superclass.constructor.call(this, config);
31553     this.field = field;
31554     this.addEvents({
31555         /**
31556              * @event beforestartedit
31557              * Fires when editing is initiated, but before the value changes.  Editing can be canceled by returning
31558              * false from the handler of this event.
31559              * @param {Editor} this
31560              * @param {Roo.Element} boundEl The underlying element bound to this editor
31561              * @param {Mixed} value The field value being set
31562              */
31563         "beforestartedit" : true,
31564         /**
31565              * @event startedit
31566              * Fires when this editor is displayed
31567              * @param {Roo.Element} boundEl The underlying element bound to this editor
31568              * @param {Mixed} value The starting field value
31569              */
31570         "startedit" : true,
31571         /**
31572              * @event beforecomplete
31573              * Fires after a change has been made to the field, but before the change is reflected in the underlying
31574              * field.  Saving the change to the field can be canceled by returning false from the handler of this event.
31575              * Note that if the value has not changed and ignoreNoChange = true, the editing will still end but this
31576              * event will not fire since no edit actually occurred.
31577              * @param {Editor} this
31578              * @param {Mixed} value The current field value
31579              * @param {Mixed} startValue The original field value
31580              */
31581         "beforecomplete" : true,
31582         /**
31583              * @event complete
31584              * Fires after editing is complete and any changed value has been written to the underlying field.
31585              * @param {Editor} this
31586              * @param {Mixed} value The current field value
31587              * @param {Mixed} startValue The original field value
31588              */
31589         "complete" : true,
31590         /**
31591          * @event specialkey
31592          * Fires when any key related to navigation (arrows, tab, enter, esc, etc.) is pressed.  You can check
31593          * {@link Roo.EventObject#getKey} to determine which key was pressed.
31594          * @param {Roo.form.Field} this
31595          * @param {Roo.EventObject} e The event object
31596          */
31597         "specialkey" : true
31598     });
31599 };
31600
31601 Roo.extend(Roo.Editor, Roo.Component, {
31602     /**
31603      * @cfg {Boolean/String} autosize
31604      * True for the editor to automatically adopt the size of the underlying field, "width" to adopt the width only,
31605      * or "height" to adopt the height only (defaults to false)
31606      */
31607     /**
31608      * @cfg {Boolean} revertInvalid
31609      * True to automatically revert the field value and cancel the edit when the user completes an edit and the field
31610      * validation fails (defaults to true)
31611      */
31612     /**
31613      * @cfg {Boolean} ignoreNoChange
31614      * True to skip the the edit completion process (no save, no events fired) if the user completes an edit and
31615      * the value has not changed (defaults to false).  Applies only to string values - edits for other data types
31616      * will never be ignored.
31617      */
31618     /**
31619      * @cfg {Boolean} hideEl
31620      * False to keep the bound element visible while the editor is displayed (defaults to true)
31621      */
31622     /**
31623      * @cfg {Mixed} value
31624      * The data value of the underlying field (defaults to "")
31625      */
31626     value : "",
31627     /**
31628      * @cfg {String} alignment
31629      * The position to align to (see {@link Roo.Element#alignTo} for more details, defaults to "c-c?").
31630      */
31631     alignment: "c-c?",
31632     /**
31633      * @cfg {Boolean/String} shadow "sides" for sides/bottom only, "frame" for 4-way shadow, and "drop"
31634      * for bottom-right shadow (defaults to "frame")
31635      */
31636     shadow : "frame",
31637     /**
31638      * @cfg {Boolean} constrain True to constrain the editor to the viewport
31639      */
31640     constrain : false,
31641     /**
31642      * @cfg {Boolean} completeOnEnter True to complete the edit when the enter key is pressed (defaults to false)
31643      */
31644     completeOnEnter : false,
31645     /**
31646      * @cfg {Boolean} cancelOnEsc True to cancel the edit when the escape key is pressed (defaults to false)
31647      */
31648     cancelOnEsc : false,
31649     /**
31650      * @cfg {Boolean} updateEl True to update the innerHTML of the bound element when the update completes (defaults to false)
31651      */
31652     updateEl : false,
31653
31654     // private
31655     onRender : function(ct, position){
31656         this.el = new Roo.Layer({
31657             shadow: this.shadow,
31658             cls: "x-editor",
31659             parentEl : ct,
31660             shim : this.shim,
31661             shadowOffset:4,
31662             id: this.id,
31663             constrain: this.constrain
31664         });
31665         this.el.setStyle("overflow", Roo.isGecko ? "auto" : "hidden");
31666         if(this.field.msgTarget != 'title'){
31667             this.field.msgTarget = 'qtip';
31668         }
31669         this.field.render(this.el);
31670         if(Roo.isGecko){
31671             this.field.el.dom.setAttribute('autocomplete', 'off');
31672         }
31673         this.field.on("specialkey", this.onSpecialKey, this);
31674         if(this.swallowKeys){
31675             this.field.el.swallowEvent(['keydown','keypress']);
31676         }
31677         this.field.show();
31678         this.field.on("blur", this.onBlur, this);
31679         if(this.field.grow){
31680             this.field.on("autosize", this.el.sync,  this.el, {delay:1});
31681         }
31682     },
31683
31684     onSpecialKey : function(field, e)
31685     {
31686         //Roo.log('editor onSpecialKey');
31687         if(this.completeOnEnter && e.getKey() == e.ENTER){
31688             e.stopEvent();
31689             this.completeEdit();
31690             return;
31691         }
31692         // do not fire special key otherwise it might hide close the editor...
31693         if(e.getKey() == e.ENTER){    
31694             return;
31695         }
31696         if(this.cancelOnEsc && e.getKey() == e.ESC){
31697             this.cancelEdit();
31698             return;
31699         } 
31700         this.fireEvent('specialkey', field, e);
31701     
31702     },
31703
31704     /**
31705      * Starts the editing process and shows the editor.
31706      * @param {String/HTMLElement/Element} el The element to edit
31707      * @param {String} value (optional) A value to initialize the editor with. If a value is not provided, it defaults
31708       * to the innerHTML of el.
31709      */
31710     startEdit : function(el, value){
31711         if(this.editing){
31712             this.completeEdit();
31713         }
31714         this.boundEl = Roo.get(el);
31715         var v = value !== undefined ? value : this.boundEl.dom.innerHTML;
31716         if(!this.rendered){
31717             this.render(this.parentEl || document.body);
31718         }
31719         if(this.fireEvent("beforestartedit", this, this.boundEl, v) === false){
31720             return;
31721         }
31722         this.startValue = v;
31723         this.field.setValue(v);
31724         if(this.autoSize){
31725             var sz = this.boundEl.getSize();
31726             switch(this.autoSize){
31727                 case "width":
31728                 this.setSize(sz.width,  "");
31729                 break;
31730                 case "height":
31731                 this.setSize("",  sz.height);
31732                 break;
31733                 default:
31734                 this.setSize(sz.width,  sz.height);
31735             }
31736         }
31737         this.el.alignTo(this.boundEl, this.alignment);
31738         this.editing = true;
31739         if(Roo.QuickTips){
31740             Roo.QuickTips.disable();
31741         }
31742         this.show();
31743     },
31744
31745     /**
31746      * Sets the height and width of this editor.
31747      * @param {Number} width The new width
31748      * @param {Number} height The new height
31749      */
31750     setSize : function(w, h){
31751         this.field.setSize(w, h);
31752         if(this.el){
31753             this.el.sync();
31754         }
31755     },
31756
31757     /**
31758      * Realigns the editor to the bound field based on the current alignment config value.
31759      */
31760     realign : function(){
31761         this.el.alignTo(this.boundEl, this.alignment);
31762     },
31763
31764     /**
31765      * Ends the editing process, persists the changed value to the underlying field, and hides the editor.
31766      * @param {Boolean} remainVisible Override the default behavior and keep the editor visible after edit (defaults to false)
31767      */
31768     completeEdit : function(remainVisible){
31769         if(!this.editing){
31770             return;
31771         }
31772         var v = this.getValue();
31773         if(this.revertInvalid !== false && !this.field.isValid()){
31774             v = this.startValue;
31775             this.cancelEdit(true);
31776         }
31777         if(String(v) === String(this.startValue) && this.ignoreNoChange){
31778             this.editing = false;
31779             this.hide();
31780             return;
31781         }
31782         if(this.fireEvent("beforecomplete", this, v, this.startValue) !== false){
31783             this.editing = false;
31784             if(this.updateEl && this.boundEl){
31785                 this.boundEl.update(v);
31786             }
31787             if(remainVisible !== true){
31788                 this.hide();
31789             }
31790             this.fireEvent("complete", this, v, this.startValue);
31791         }
31792     },
31793
31794     // private
31795     onShow : function(){
31796         this.el.show();
31797         if(this.hideEl !== false){
31798             this.boundEl.hide();
31799         }
31800         this.field.show();
31801         if(Roo.isIE && !this.fixIEFocus){ // IE has problems with focusing the first time
31802             this.fixIEFocus = true;
31803             this.deferredFocus.defer(50, this);
31804         }else{
31805             this.field.focus();
31806         }
31807         this.fireEvent("startedit", this.boundEl, this.startValue);
31808     },
31809
31810     deferredFocus : function(){
31811         if(this.editing){
31812             this.field.focus();
31813         }
31814     },
31815
31816     /**
31817      * Cancels the editing process and hides the editor without persisting any changes.  The field value will be
31818      * reverted to the original starting value.
31819      * @param {Boolean} remainVisible Override the default behavior and keep the editor visible after
31820      * cancel (defaults to false)
31821      */
31822     cancelEdit : function(remainVisible){
31823         if(this.editing){
31824             this.setValue(this.startValue);
31825             if(remainVisible !== true){
31826                 this.hide();
31827             }
31828         }
31829     },
31830
31831     // private
31832     onBlur : function(){
31833         if(this.allowBlur !== true && this.editing){
31834             this.completeEdit();
31835         }
31836     },
31837
31838     // private
31839     onHide : function(){
31840         if(this.editing){
31841             this.completeEdit();
31842             return;
31843         }
31844         this.field.blur();
31845         if(this.field.collapse){
31846             this.field.collapse();
31847         }
31848         this.el.hide();
31849         if(this.hideEl !== false){
31850             this.boundEl.show();
31851         }
31852         if(Roo.QuickTips){
31853             Roo.QuickTips.enable();
31854         }
31855     },
31856
31857     /**
31858      * Sets the data value of the editor
31859      * @param {Mixed} value Any valid value supported by the underlying field
31860      */
31861     setValue : function(v){
31862         this.field.setValue(v);
31863     },
31864
31865     /**
31866      * Gets the data value of the editor
31867      * @return {Mixed} The data value
31868      */
31869     getValue : function(){
31870         return this.field.getValue();
31871     }
31872 });/*
31873  * Based on:
31874  * Ext JS Library 1.1.1
31875  * Copyright(c) 2006-2007, Ext JS, LLC.
31876  *
31877  * Originally Released Under LGPL - original licence link has changed is not relivant.
31878  *
31879  * Fork - LGPL
31880  * <script type="text/javascript">
31881  */
31882  
31883 /**
31884  * @class Roo.BasicDialog
31885  * @extends Roo.util.Observable
31886  * Lightweight Dialog Class.  The code below shows the creation of a typical dialog using existing HTML markup:
31887  * <pre><code>
31888 var dlg = new Roo.BasicDialog("my-dlg", {
31889     height: 200,
31890     width: 300,
31891     minHeight: 100,
31892     minWidth: 150,
31893     modal: true,
31894     proxyDrag: true,
31895     shadow: true
31896 });
31897 dlg.addKeyListener(27, dlg.hide, dlg); // ESC can also close the dialog
31898 dlg.addButton('OK', dlg.hide, dlg);    // Could call a save function instead of hiding
31899 dlg.addButton('Cancel', dlg.hide, dlg);
31900 dlg.show();
31901 </code></pre>
31902   <b>A Dialog should always be a direct child of the body element.</b>
31903  * @cfg {Boolean/DomHelper} autoCreate True to auto create from scratch, or using a DomHelper Object (defaults to false)
31904  * @cfg {String} title Default text to display in the title bar (defaults to null)
31905  * @cfg {Number} width Width of the dialog in pixels (can also be set via CSS).  Determined by browser if unspecified.
31906  * @cfg {Number} height Height of the dialog in pixels (can also be set via CSS).  Determined by browser if unspecified.
31907  * @cfg {Number} x The default left page coordinate of the dialog (defaults to center screen)
31908  * @cfg {Number} y The default top page coordinate of the dialog (defaults to center screen)
31909  * @cfg {String/Element} animateTarget Id or element from which the dialog should animate while opening
31910  * (defaults to null with no animation)
31911  * @cfg {Boolean} resizable False to disable manual dialog resizing (defaults to true)
31912  * @cfg {String} resizeHandles Which resize handles to display - see the {@link Roo.Resizable} handles config
31913  * property for valid values (defaults to 'all')
31914  * @cfg {Number} minHeight The minimum allowable height for a resizable dialog (defaults to 80)
31915  * @cfg {Number} minWidth The minimum allowable width for a resizable dialog (defaults to 200)
31916  * @cfg {Boolean} modal True to show the dialog modally, preventing user interaction with the rest of the page (defaults to false)
31917  * @cfg {Boolean} autoScroll True to allow the dialog body contents to overflow and display scrollbars (defaults to false)
31918  * @cfg {Boolean} closable False to remove the built-in top-right corner close button (defaults to true)
31919  * @cfg {Boolean} collapsible False to remove the built-in top-right corner collapse button (defaults to true)
31920  * @cfg {Boolean} constraintoviewport True to keep the dialog constrained within the visible viewport boundaries (defaults to true)
31921  * @cfg {Boolean} syncHeightBeforeShow True to cause the dimensions to be recalculated before the dialog is shown (defaults to false)
31922  * @cfg {Boolean} draggable False to disable dragging of the dialog within the viewport (defaults to true)
31923  * @cfg {Boolean} autoTabs If true, all elements with class 'x-dlg-tab' will get automatically converted to tabs (defaults to false)
31924  * @cfg {String} tabTag The tag name of tab elements, used when autoTabs = true (defaults to 'div')
31925  * @cfg {Boolean} proxyDrag True to drag a lightweight proxy element rather than the dialog itself, used when
31926  * draggable = true (defaults to false)
31927  * @cfg {Boolean} fixedcenter True to ensure that anytime the dialog is shown or resized it gets centered (defaults to false)
31928  * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
31929  * shadow (defaults to false)
31930  * @cfg {Number} shadowOffset The number of pixels to offset the shadow if displayed (defaults to 5)
31931  * @cfg {String} buttonAlign Valid values are "left," "center" and "right" (defaults to "right")
31932  * @cfg {Number} minButtonWidth Minimum width of all dialog buttons (defaults to 75)
31933  * @cfg {Array} buttons Array of buttons
31934  * @cfg {Boolean} shim True to create an iframe shim that prevents selects from showing through (defaults to false)
31935  * @constructor
31936  * Create a new BasicDialog.
31937  * @param {String/HTMLElement/Roo.Element} el The container element or DOM node, or its id
31938  * @param {Object} config Configuration options
31939  */
31940 Roo.BasicDialog = function(el, config){
31941     this.el = Roo.get(el);
31942     var dh = Roo.DomHelper;
31943     if(!this.el && config && config.autoCreate){
31944         if(typeof config.autoCreate == "object"){
31945             if(!config.autoCreate.id){
31946                 config.autoCreate.id = el;
31947             }
31948             this.el = dh.append(document.body,
31949                         config.autoCreate, true);
31950         }else{
31951             this.el = dh.append(document.body,
31952                         {tag: "div", id: el, style:'visibility:hidden;'}, true);
31953         }
31954     }
31955     el = this.el;
31956     el.setDisplayed(true);
31957     el.hide = this.hideAction;
31958     this.id = el.id;
31959     el.addClass("x-dlg");
31960
31961     Roo.apply(this, config);
31962
31963     this.proxy = el.createProxy("x-dlg-proxy");
31964     this.proxy.hide = this.hideAction;
31965     this.proxy.setOpacity(.5);
31966     this.proxy.hide();
31967
31968     if(config.width){
31969         el.setWidth(config.width);
31970     }
31971     if(config.height){
31972         el.setHeight(config.height);
31973     }
31974     this.size = el.getSize();
31975     if(typeof config.x != "undefined" && typeof config.y != "undefined"){
31976         this.xy = [config.x,config.y];
31977     }else{
31978         this.xy = el.getCenterXY(true);
31979     }
31980     /** The header element @type Roo.Element */
31981     this.header = el.child("> .x-dlg-hd");
31982     /** The body element @type Roo.Element */
31983     this.body = el.child("> .x-dlg-bd");
31984     /** The footer element @type Roo.Element */
31985     this.footer = el.child("> .x-dlg-ft");
31986
31987     if(!this.header){
31988         this.header = el.createChild({tag: "div", cls:"x-dlg-hd", html: "&#160;"}, this.body ? this.body.dom : null);
31989     }
31990     if(!this.body){
31991         this.body = el.createChild({tag: "div", cls:"x-dlg-bd"});
31992     }
31993
31994     this.header.unselectable();
31995     if(this.title){
31996         this.header.update(this.title);
31997     }
31998     // this element allows the dialog to be focused for keyboard event
31999     this.focusEl = el.createChild({tag: "a", href:"#", cls:"x-dlg-focus", tabIndex:"-1"});
32000     this.focusEl.swallowEvent("click", true);
32001
32002     this.header.wrap({cls:"x-dlg-hd-right"}).wrap({cls:"x-dlg-hd-left"}, true);
32003
32004     // wrap the body and footer for special rendering
32005     this.bwrap = this.body.wrap({tag: "div", cls:"x-dlg-dlg-body"});
32006     if(this.footer){
32007         this.bwrap.dom.appendChild(this.footer.dom);
32008     }
32009
32010     this.bg = this.el.createChild({
32011         tag: "div", cls:"x-dlg-bg",
32012         html: '<div class="x-dlg-bg-left"><div class="x-dlg-bg-right"><div class="x-dlg-bg-center">&#160;</div></div></div>'
32013     });
32014     this.centerBg = this.bg.child("div.x-dlg-bg-center");
32015
32016
32017     if(this.autoScroll !== false && !this.autoTabs){
32018         this.body.setStyle("overflow", "auto");
32019     }
32020
32021     this.toolbox = this.el.createChild({cls: "x-dlg-toolbox"});
32022
32023     if(this.closable !== false){
32024         this.el.addClass("x-dlg-closable");
32025         this.close = this.toolbox.createChild({cls:"x-dlg-close"});
32026         this.close.on("click", this.closeClick, this);
32027         this.close.addClassOnOver("x-dlg-close-over");
32028     }
32029     if(this.collapsible !== false){
32030         this.collapseBtn = this.toolbox.createChild({cls:"x-dlg-collapse"});
32031         this.collapseBtn.on("click", this.collapseClick, this);
32032         this.collapseBtn.addClassOnOver("x-dlg-collapse-over");
32033         this.header.on("dblclick", this.collapseClick, this);
32034     }
32035     if(this.resizable !== false){
32036         this.el.addClass("x-dlg-resizable");
32037         this.resizer = new Roo.Resizable(el, {
32038             minWidth: this.minWidth || 80,
32039             minHeight:this.minHeight || 80,
32040             handles: this.resizeHandles || "all",
32041             pinned: true
32042         });
32043         this.resizer.on("beforeresize", this.beforeResize, this);
32044         this.resizer.on("resize", this.onResize, this);
32045     }
32046     if(this.draggable !== false){
32047         el.addClass("x-dlg-draggable");
32048         if (!this.proxyDrag) {
32049             var dd = new Roo.dd.DD(el.dom.id, "WindowDrag");
32050         }
32051         else {
32052             var dd = new Roo.dd.DDProxy(el.dom.id, "WindowDrag", {dragElId: this.proxy.id});
32053         }
32054         dd.setHandleElId(this.header.id);
32055         dd.endDrag = this.endMove.createDelegate(this);
32056         dd.startDrag = this.startMove.createDelegate(this);
32057         dd.onDrag = this.onDrag.createDelegate(this);
32058         dd.scroll = false;
32059         this.dd = dd;
32060     }
32061     if(this.modal){
32062         this.mask = dh.append(document.body, {tag: "div", cls:"x-dlg-mask"}, true);
32063         this.mask.enableDisplayMode("block");
32064         this.mask.hide();
32065         this.el.addClass("x-dlg-modal");
32066     }
32067     if(this.shadow){
32068         this.shadow = new Roo.Shadow({
32069             mode : typeof this.shadow == "string" ? this.shadow : "sides",
32070             offset : this.shadowOffset
32071         });
32072     }else{
32073         this.shadowOffset = 0;
32074     }
32075     if(Roo.useShims && this.shim !== false){
32076         this.shim = this.el.createShim();
32077         this.shim.hide = this.hideAction;
32078         this.shim.hide();
32079     }else{
32080         this.shim = false;
32081     }
32082     if(this.autoTabs){
32083         this.initTabs();
32084     }
32085     if (this.buttons) { 
32086         var bts= this.buttons;
32087         this.buttons = [];
32088         Roo.each(bts, function(b) {
32089             this.addButton(b);
32090         }, this);
32091     }
32092     
32093     
32094     this.addEvents({
32095         /**
32096          * @event keydown
32097          * Fires when a key is pressed
32098          * @param {Roo.BasicDialog} this
32099          * @param {Roo.EventObject} e
32100          */
32101         "keydown" : true,
32102         /**
32103          * @event move
32104          * Fires when this dialog is moved by the user.
32105          * @param {Roo.BasicDialog} this
32106          * @param {Number} x The new page X
32107          * @param {Number} y The new page Y
32108          */
32109         "move" : true,
32110         /**
32111          * @event resize
32112          * Fires when this dialog is resized by the user.
32113          * @param {Roo.BasicDialog} this
32114          * @param {Number} width The new width
32115          * @param {Number} height The new height
32116          */
32117         "resize" : true,
32118         /**
32119          * @event beforehide
32120          * Fires before this dialog is hidden.
32121          * @param {Roo.BasicDialog} this
32122          */
32123         "beforehide" : true,
32124         /**
32125          * @event hide
32126          * Fires when this dialog is hidden.
32127          * @param {Roo.BasicDialog} this
32128          */
32129         "hide" : true,
32130         /**
32131          * @event beforeshow
32132          * Fires before this dialog is shown.
32133          * @param {Roo.BasicDialog} this
32134          */
32135         "beforeshow" : true,
32136         /**
32137          * @event show
32138          * Fires when this dialog is shown.
32139          * @param {Roo.BasicDialog} this
32140          */
32141         "show" : true
32142     });
32143     el.on("keydown", this.onKeyDown, this);
32144     el.on("mousedown", this.toFront, this);
32145     Roo.EventManager.onWindowResize(this.adjustViewport, this, true);
32146     this.el.hide();
32147     Roo.DialogManager.register(this);
32148     Roo.BasicDialog.superclass.constructor.call(this);
32149 };
32150
32151 Roo.extend(Roo.BasicDialog, Roo.util.Observable, {
32152     shadowOffset: Roo.isIE ? 6 : 5,
32153     minHeight: 80,
32154     minWidth: 200,
32155     minButtonWidth: 75,
32156     defaultButton: null,
32157     buttonAlign: "right",
32158     tabTag: 'div',
32159     firstShow: true,
32160
32161     /**
32162      * Sets the dialog title text
32163      * @param {String} text The title text to display
32164      * @return {Roo.BasicDialog} this
32165      */
32166     setTitle : function(text){
32167         this.header.update(text);
32168         return this;
32169     },
32170
32171     // private
32172     closeClick : function(){
32173         this.hide();
32174     },
32175
32176     // private
32177     collapseClick : function(){
32178         this[this.collapsed ? "expand" : "collapse"]();
32179     },
32180
32181     /**
32182      * Collapses the dialog to its minimized state (only the title bar is visible).
32183      * Equivalent to the user clicking the collapse dialog button.
32184      */
32185     collapse : function(){
32186         if(!this.collapsed){
32187             this.collapsed = true;
32188             this.el.addClass("x-dlg-collapsed");
32189             this.restoreHeight = this.el.getHeight();
32190             this.resizeTo(this.el.getWidth(), this.header.getHeight());
32191         }
32192     },
32193
32194     /**
32195      * Expands a collapsed dialog back to its normal state.  Equivalent to the user
32196      * clicking the expand dialog button.
32197      */
32198     expand : function(){
32199         if(this.collapsed){
32200             this.collapsed = false;
32201             this.el.removeClass("x-dlg-collapsed");
32202             this.resizeTo(this.el.getWidth(), this.restoreHeight);
32203         }
32204     },
32205
32206     /**
32207      * Reinitializes the tabs component, clearing out old tabs and finding new ones.
32208      * @return {Roo.TabPanel} The tabs component
32209      */
32210     initTabs : function(){
32211         var tabs = this.getTabs();
32212         while(tabs.getTab(0)){
32213             tabs.removeTab(0);
32214         }
32215         this.el.select(this.tabTag+'.x-dlg-tab').each(function(el){
32216             var dom = el.dom;
32217             tabs.addTab(Roo.id(dom), dom.title);
32218             dom.title = "";
32219         });
32220         tabs.activate(0);
32221         return tabs;
32222     },
32223
32224     // private
32225     beforeResize : function(){
32226         this.resizer.minHeight = Math.max(this.minHeight, this.getHeaderFooterHeight(true)+40);
32227     },
32228
32229     // private
32230     onResize : function(){
32231         this.refreshSize();
32232         this.syncBodyHeight();
32233         this.adjustAssets();
32234         this.focus();
32235         this.fireEvent("resize", this, this.size.width, this.size.height);
32236     },
32237
32238     // private
32239     onKeyDown : function(e){
32240         if(this.isVisible()){
32241             this.fireEvent("keydown", this, e);
32242         }
32243     },
32244
32245     /**
32246      * Resizes the dialog.
32247      * @param {Number} width
32248      * @param {Number} height
32249      * @return {Roo.BasicDialog} this
32250      */
32251     resizeTo : function(width, height){
32252         this.el.setSize(width, height);
32253         this.size = {width: width, height: height};
32254         this.syncBodyHeight();
32255         if(this.fixedcenter){
32256             this.center();
32257         }
32258         if(this.isVisible()){
32259             this.constrainXY();
32260             this.adjustAssets();
32261         }
32262         this.fireEvent("resize", this, width, height);
32263         return this;
32264     },
32265
32266
32267     /**
32268      * Resizes the dialog to fit the specified content size.
32269      * @param {Number} width
32270      * @param {Number} height
32271      * @return {Roo.BasicDialog} this
32272      */
32273     setContentSize : function(w, h){
32274         h += this.getHeaderFooterHeight() + this.body.getMargins("tb");
32275         w += this.body.getMargins("lr") + this.bwrap.getMargins("lr") + this.centerBg.getPadding("lr");
32276         //if(!this.el.isBorderBox()){
32277             h +=  this.body.getPadding("tb") + this.bwrap.getBorderWidth("tb") + this.body.getBorderWidth("tb") + this.el.getBorderWidth("tb");
32278             w += this.body.getPadding("lr") + this.bwrap.getBorderWidth("lr") + this.body.getBorderWidth("lr") + this.bwrap.getPadding("lr") + this.el.getBorderWidth("lr");
32279         //}
32280         if(this.tabs){
32281             h += this.tabs.stripWrap.getHeight() + this.tabs.bodyEl.getMargins("tb") + this.tabs.bodyEl.getPadding("tb");
32282             w += this.tabs.bodyEl.getMargins("lr") + this.tabs.bodyEl.getPadding("lr");
32283         }
32284         this.resizeTo(w, h);
32285         return this;
32286     },
32287
32288     /**
32289      * Adds a key listener for when this dialog is displayed.  This allows you to hook in a function that will be
32290      * executed in response to a particular key being pressed while the dialog is active.
32291      * @param {Number/Array/Object} key Either the numeric key code, array of key codes or an object with the following options:
32292      *                                  {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
32293      * @param {Function} fn The function to call
32294      * @param {Object} scope (optional) The scope of the function
32295      * @return {Roo.BasicDialog} this
32296      */
32297     addKeyListener : function(key, fn, scope){
32298         var keyCode, shift, ctrl, alt;
32299         if(typeof key == "object" && !(key instanceof Array)){
32300             keyCode = key["key"];
32301             shift = key["shift"];
32302             ctrl = key["ctrl"];
32303             alt = key["alt"];
32304         }else{
32305             keyCode = key;
32306         }
32307         var handler = function(dlg, e){
32308             if((!shift || e.shiftKey) && (!ctrl || e.ctrlKey) &&  (!alt || e.altKey)){
32309                 var k = e.getKey();
32310                 if(keyCode instanceof Array){
32311                     for(var i = 0, len = keyCode.length; i < len; i++){
32312                         if(keyCode[i] == k){
32313                           fn.call(scope || window, dlg, k, e);
32314                           return;
32315                         }
32316                     }
32317                 }else{
32318                     if(k == keyCode){
32319                         fn.call(scope || window, dlg, k, e);
32320                     }
32321                 }
32322             }
32323         };
32324         this.on("keydown", handler);
32325         return this;
32326     },
32327
32328     /**
32329      * Returns the TabPanel component (creates it if it doesn't exist).
32330      * Note: If you wish to simply check for the existence of tabs without creating them,
32331      * check for a null 'tabs' property.
32332      * @return {Roo.TabPanel} The tabs component
32333      */
32334     getTabs : function(){
32335         if(!this.tabs){
32336             this.el.addClass("x-dlg-auto-tabs");
32337             this.body.addClass(this.tabPosition == "bottom" ? "x-tabs-bottom" : "x-tabs-top");
32338             this.tabs = new Roo.TabPanel(this.body.dom, this.tabPosition == "bottom");
32339         }
32340         return this.tabs;
32341     },
32342
32343     /**
32344      * Adds a button to the footer section of the dialog.
32345      * @param {String/Object} config A string becomes the button text, an object can either be a Button config
32346      * object or a valid Roo.DomHelper element config
32347      * @param {Function} handler The function called when the button is clicked
32348      * @param {Object} scope (optional) The scope of the handler function (accepts position as a property)
32349      * @return {Roo.Button} The new button
32350      */
32351     addButton : function(config, handler, scope){
32352         var dh = Roo.DomHelper;
32353         if(!this.footer){
32354             this.footer = dh.append(this.bwrap, {tag: "div", cls:"x-dlg-ft"}, true);
32355         }
32356         if(!this.btnContainer){
32357             var tb = this.footer.createChild({
32358
32359                 cls:"x-dlg-btns x-dlg-btns-"+this.buttonAlign,
32360                 html:'<table cellspacing="0"><tbody><tr></tr></tbody></table><div class="x-clear"></div>'
32361             }, null, true);
32362             this.btnContainer = tb.firstChild.firstChild.firstChild;
32363         }
32364         var bconfig = {
32365             handler: handler,
32366             scope: scope,
32367             minWidth: this.minButtonWidth,
32368             hideParent:true
32369         };
32370         if(typeof config == "string"){
32371             bconfig.text = config;
32372         }else{
32373             if(config.tag){
32374                 bconfig.dhconfig = config;
32375             }else{
32376                 Roo.apply(bconfig, config);
32377             }
32378         }
32379         var fc = false;
32380         if ((typeof(bconfig.position) != 'undefined') && bconfig.position < this.btnContainer.childNodes.length-1) {
32381             bconfig.position = Math.max(0, bconfig.position);
32382             fc = this.btnContainer.childNodes[bconfig.position];
32383         }
32384          
32385         var btn = new Roo.Button(
32386             fc ? 
32387                 this.btnContainer.insertBefore(document.createElement("td"),fc)
32388                 : this.btnContainer.appendChild(document.createElement("td")),
32389             //Roo.get(this.btnContainer).createChild( { tag: 'td'},  fc ),
32390             bconfig
32391         );
32392         this.syncBodyHeight();
32393         if(!this.buttons){
32394             /**
32395              * Array of all the buttons that have been added to this dialog via addButton
32396              * @type Array
32397              */
32398             this.buttons = [];
32399         }
32400         this.buttons.push(btn);
32401         return btn;
32402     },
32403
32404     /**
32405      * Sets the default button to be focused when the dialog is displayed.
32406      * @param {Roo.BasicDialog.Button} btn The button object returned by {@link #addButton}
32407      * @return {Roo.BasicDialog} this
32408      */
32409     setDefaultButton : function(btn){
32410         this.defaultButton = btn;
32411         return this;
32412     },
32413
32414     // private
32415     getHeaderFooterHeight : function(safe){
32416         var height = 0;
32417         if(this.header){
32418            height += this.header.getHeight();
32419         }
32420         if(this.footer){
32421            var fm = this.footer.getMargins();
32422             height += (this.footer.getHeight()+fm.top+fm.bottom);
32423         }
32424         height += this.bwrap.getPadding("tb")+this.bwrap.getBorderWidth("tb");
32425         height += this.centerBg.getPadding("tb");
32426         return height;
32427     },
32428
32429     // private
32430     syncBodyHeight : function()
32431     {
32432         var bd = this.body, // the text
32433             cb = this.centerBg, // wrapper around bottom.. but does not seem to be used..
32434             bw = this.bwrap;
32435         var height = this.size.height - this.getHeaderFooterHeight(false);
32436         bd.setHeight(height-bd.getMargins("tb"));
32437         var hh = this.header.getHeight();
32438         var h = this.size.height-hh;
32439         cb.setHeight(h);
32440         
32441         bw.setLeftTop(cb.getPadding("l"), hh+cb.getPadding("t"));
32442         bw.setHeight(h-cb.getPadding("tb"));
32443         
32444         bw.setWidth(this.el.getWidth(true)-cb.getPadding("lr"));
32445         bd.setWidth(bw.getWidth(true));
32446         if(this.tabs){
32447             this.tabs.syncHeight();
32448             if(Roo.isIE){
32449                 this.tabs.el.repaint();
32450             }
32451         }
32452     },
32453
32454     /**
32455      * Restores the previous state of the dialog if Roo.state is configured.
32456      * @return {Roo.BasicDialog} this
32457      */
32458     restoreState : function(){
32459         var box = Roo.state.Manager.get(this.stateId || (this.el.id + "-state"));
32460         if(box && box.width){
32461             this.xy = [box.x, box.y];
32462             this.resizeTo(box.width, box.height);
32463         }
32464         return this;
32465     },
32466
32467     // private
32468     beforeShow : function(){
32469         this.expand();
32470         if(this.fixedcenter){
32471             this.xy = this.el.getCenterXY(true);
32472         }
32473         if(this.modal){
32474             Roo.get(document.body).addClass("x-body-masked");
32475             this.mask.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
32476             this.mask.show();
32477         }
32478         this.constrainXY();
32479     },
32480
32481     // private
32482     animShow : function(){
32483         var b = Roo.get(this.animateTarget).getBox();
32484         this.proxy.setSize(b.width, b.height);
32485         this.proxy.setLocation(b.x, b.y);
32486         this.proxy.show();
32487         this.proxy.setBounds(this.xy[0], this.xy[1], this.size.width, this.size.height,
32488                     true, .35, this.showEl.createDelegate(this));
32489     },
32490
32491     /**
32492      * Shows the dialog.
32493      * @param {String/HTMLElement/Roo.Element} animateTarget (optional) Reset the animation target
32494      * @return {Roo.BasicDialog} this
32495      */
32496     show : function(animateTarget){
32497         if (this.fireEvent("beforeshow", this) === false){
32498             return;
32499         }
32500         if(this.syncHeightBeforeShow){
32501             this.syncBodyHeight();
32502         }else if(this.firstShow){
32503             this.firstShow = false;
32504             this.syncBodyHeight(); // sync the height on the first show instead of in the constructor
32505         }
32506         this.animateTarget = animateTarget || this.animateTarget;
32507         if(!this.el.isVisible()){
32508             this.beforeShow();
32509             if(this.animateTarget && Roo.get(this.animateTarget)){
32510                 this.animShow();
32511             }else{
32512                 this.showEl();
32513             }
32514         }
32515         return this;
32516     },
32517
32518     // private
32519     showEl : function(){
32520         this.proxy.hide();
32521         this.el.setXY(this.xy);
32522         this.el.show();
32523         this.adjustAssets(true);
32524         this.toFront();
32525         this.focus();
32526         // IE peekaboo bug - fix found by Dave Fenwick
32527         if(Roo.isIE){
32528             this.el.repaint();
32529         }
32530         this.fireEvent("show", this);
32531     },
32532
32533     /**
32534      * Focuses the dialog.  If a defaultButton is set, it will receive focus, otherwise the
32535      * dialog itself will receive focus.
32536      */
32537     focus : function(){
32538         if(this.defaultButton){
32539             this.defaultButton.focus();
32540         }else{
32541             this.focusEl.focus();
32542         }
32543     },
32544
32545     // private
32546     constrainXY : function(){
32547         if(this.constraintoviewport !== false){
32548             if(!this.viewSize){
32549                 if(this.container){
32550                     var s = this.container.getSize();
32551                     this.viewSize = [s.width, s.height];
32552                 }else{
32553                     this.viewSize = [Roo.lib.Dom.getViewWidth(),Roo.lib.Dom.getViewHeight()];
32554                 }
32555             }
32556             var s = Roo.get(this.container||document).getScroll();
32557
32558             var x = this.xy[0], y = this.xy[1];
32559             var w = this.size.width, h = this.size.height;
32560             var vw = this.viewSize[0], vh = this.viewSize[1];
32561             // only move it if it needs it
32562             var moved = false;
32563             // first validate right/bottom
32564             if(x + w > vw+s.left){
32565                 x = vw - w;
32566                 moved = true;
32567             }
32568             if(y + h > vh+s.top){
32569                 y = vh - h;
32570                 moved = true;
32571             }
32572             // then make sure top/left isn't negative
32573             if(x < s.left){
32574                 x = s.left;
32575                 moved = true;
32576             }
32577             if(y < s.top){
32578                 y = s.top;
32579                 moved = true;
32580             }
32581             if(moved){
32582                 // cache xy
32583                 this.xy = [x, y];
32584                 if(this.isVisible()){
32585                     this.el.setLocation(x, y);
32586                     this.adjustAssets();
32587                 }
32588             }
32589         }
32590     },
32591
32592     // private
32593     onDrag : function(){
32594         if(!this.proxyDrag){
32595             this.xy = this.el.getXY();
32596             this.adjustAssets();
32597         }
32598     },
32599
32600     // private
32601     adjustAssets : function(doShow){
32602         var x = this.xy[0], y = this.xy[1];
32603         var w = this.size.width, h = this.size.height;
32604         if(doShow === true){
32605             if(this.shadow){
32606                 this.shadow.show(this.el);
32607             }
32608             if(this.shim){
32609                 this.shim.show();
32610             }
32611         }
32612         if(this.shadow && this.shadow.isVisible()){
32613             this.shadow.show(this.el);
32614         }
32615         if(this.shim && this.shim.isVisible()){
32616             this.shim.setBounds(x, y, w, h);
32617         }
32618     },
32619
32620     // private
32621     adjustViewport : function(w, h){
32622         if(!w || !h){
32623             w = Roo.lib.Dom.getViewWidth();
32624             h = Roo.lib.Dom.getViewHeight();
32625         }
32626         // cache the size
32627         this.viewSize = [w, h];
32628         if(this.modal && this.mask.isVisible()){
32629             this.mask.setSize(w, h); // first make sure the mask isn't causing overflow
32630             this.mask.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
32631         }
32632         if(this.isVisible()){
32633             this.constrainXY();
32634         }
32635     },
32636
32637     /**
32638      * Destroys this dialog and all its supporting elements (including any tabs, shim,
32639      * shadow, proxy, mask, etc.)  Also removes all event listeners.
32640      * @param {Boolean} removeEl (optional) true to remove the element from the DOM
32641      */
32642     destroy : function(removeEl){
32643         if(this.isVisible()){
32644             this.animateTarget = null;
32645             this.hide();
32646         }
32647         Roo.EventManager.removeResizeListener(this.adjustViewport, this);
32648         if(this.tabs){
32649             this.tabs.destroy(removeEl);
32650         }
32651         Roo.destroy(
32652              this.shim,
32653              this.proxy,
32654              this.resizer,
32655              this.close,
32656              this.mask
32657         );
32658         if(this.dd){
32659             this.dd.unreg();
32660         }
32661         if(this.buttons){
32662            for(var i = 0, len = this.buttons.length; i < len; i++){
32663                this.buttons[i].destroy();
32664            }
32665         }
32666         this.el.removeAllListeners();
32667         if(removeEl === true){
32668             this.el.update("");
32669             this.el.remove();
32670         }
32671         Roo.DialogManager.unregister(this);
32672     },
32673
32674     // private
32675     startMove : function(){
32676         if(this.proxyDrag){
32677             this.proxy.show();
32678         }
32679         if(this.constraintoviewport !== false){
32680             this.dd.constrainTo(document.body, {right: this.shadowOffset, bottom: this.shadowOffset});
32681         }
32682     },
32683
32684     // private
32685     endMove : function(){
32686         if(!this.proxyDrag){
32687             Roo.dd.DD.prototype.endDrag.apply(this.dd, arguments);
32688         }else{
32689             Roo.dd.DDProxy.prototype.endDrag.apply(this.dd, arguments);
32690             this.proxy.hide();
32691         }
32692         this.refreshSize();
32693         this.adjustAssets();
32694         this.focus();
32695         this.fireEvent("move", this, this.xy[0], this.xy[1]);
32696     },
32697
32698     /**
32699      * Brings this dialog to the front of any other visible dialogs
32700      * @return {Roo.BasicDialog} this
32701      */
32702     toFront : function(){
32703         Roo.DialogManager.bringToFront(this);
32704         return this;
32705     },
32706
32707     /**
32708      * Sends this dialog to the back (under) of any other visible dialogs
32709      * @return {Roo.BasicDialog} this
32710      */
32711     toBack : function(){
32712         Roo.DialogManager.sendToBack(this);
32713         return this;
32714     },
32715
32716     /**
32717      * Centers this dialog in the viewport
32718      * @return {Roo.BasicDialog} this
32719      */
32720     center : function(){
32721         var xy = this.el.getCenterXY(true);
32722         this.moveTo(xy[0], xy[1]);
32723         return this;
32724     },
32725
32726     /**
32727      * Moves the dialog's top-left corner to the specified point
32728      * @param {Number} x
32729      * @param {Number} y
32730      * @return {Roo.BasicDialog} this
32731      */
32732     moveTo : function(x, y){
32733         this.xy = [x,y];
32734         if(this.isVisible()){
32735             this.el.setXY(this.xy);
32736             this.adjustAssets();
32737         }
32738         return this;
32739     },
32740
32741     /**
32742      * Aligns the dialog to the specified element
32743      * @param {String/HTMLElement/Roo.Element} element The element to align to.
32744      * @param {String} position The position to align to (see {@link Roo.Element#alignTo} for more details).
32745      * @param {Array} offsets (optional) Offset the positioning by [x, y]
32746      * @return {Roo.BasicDialog} this
32747      */
32748     alignTo : function(element, position, offsets){
32749         this.xy = this.el.getAlignToXY(element, position, offsets);
32750         if(this.isVisible()){
32751             this.el.setXY(this.xy);
32752             this.adjustAssets();
32753         }
32754         return this;
32755     },
32756
32757     /**
32758      * Anchors an element to another element and realigns it when the window is resized.
32759      * @param {String/HTMLElement/Roo.Element} element The element to align to.
32760      * @param {String} position The position to align to (see {@link Roo.Element#alignTo} for more details)
32761      * @param {Array} offsets (optional) Offset the positioning by [x, y]
32762      * @param {Boolean/Number} monitorScroll (optional) true to monitor body scroll and reposition. If this parameter
32763      * is a number, it is used as the buffer delay (defaults to 50ms).
32764      * @return {Roo.BasicDialog} this
32765      */
32766     anchorTo : function(el, alignment, offsets, monitorScroll){
32767         var action = function(){
32768             this.alignTo(el, alignment, offsets);
32769         };
32770         Roo.EventManager.onWindowResize(action, this);
32771         var tm = typeof monitorScroll;
32772         if(tm != 'undefined'){
32773             Roo.EventManager.on(window, 'scroll', action, this,
32774                 {buffer: tm == 'number' ? monitorScroll : 50});
32775         }
32776         action.call(this);
32777         return this;
32778     },
32779
32780     /**
32781      * Returns true if the dialog is visible
32782      * @return {Boolean}
32783      */
32784     isVisible : function(){
32785         return this.el.isVisible();
32786     },
32787
32788     // private
32789     animHide : function(callback){
32790         var b = Roo.get(this.animateTarget).getBox();
32791         this.proxy.show();
32792         this.proxy.setBounds(this.xy[0], this.xy[1], this.size.width, this.size.height);
32793         this.el.hide();
32794         this.proxy.setBounds(b.x, b.y, b.width, b.height, true, .35,
32795                     this.hideEl.createDelegate(this, [callback]));
32796     },
32797
32798     /**
32799      * Hides the dialog.
32800      * @param {Function} callback (optional) Function to call when the dialog is hidden
32801      * @return {Roo.BasicDialog} this
32802      */
32803     hide : function(callback){
32804         if (this.fireEvent("beforehide", this) === false){
32805             return;
32806         }
32807         if(this.shadow){
32808             this.shadow.hide();
32809         }
32810         if(this.shim) {
32811           this.shim.hide();
32812         }
32813         // sometimes animateTarget seems to get set.. causing problems...
32814         // this just double checks..
32815         if(this.animateTarget && Roo.get(this.animateTarget)) {
32816            this.animHide(callback);
32817         }else{
32818             this.el.hide();
32819             this.hideEl(callback);
32820         }
32821         return this;
32822     },
32823
32824     // private
32825     hideEl : function(callback){
32826         this.proxy.hide();
32827         if(this.modal){
32828             this.mask.hide();
32829             Roo.get(document.body).removeClass("x-body-masked");
32830         }
32831         this.fireEvent("hide", this);
32832         if(typeof callback == "function"){
32833             callback();
32834         }
32835     },
32836
32837     // private
32838     hideAction : function(){
32839         this.setLeft("-10000px");
32840         this.setTop("-10000px");
32841         this.setStyle("visibility", "hidden");
32842     },
32843
32844     // private
32845     refreshSize : function(){
32846         this.size = this.el.getSize();
32847         this.xy = this.el.getXY();
32848         Roo.state.Manager.set(this.stateId || this.el.id + "-state", this.el.getBox());
32849     },
32850
32851     // private
32852     // z-index is managed by the DialogManager and may be overwritten at any time
32853     setZIndex : function(index){
32854         if(this.modal){
32855             this.mask.setStyle("z-index", index);
32856         }
32857         if(this.shim){
32858             this.shim.setStyle("z-index", ++index);
32859         }
32860         if(this.shadow){
32861             this.shadow.setZIndex(++index);
32862         }
32863         this.el.setStyle("z-index", ++index);
32864         if(this.proxy){
32865             this.proxy.setStyle("z-index", ++index);
32866         }
32867         if(this.resizer){
32868             this.resizer.proxy.setStyle("z-index", ++index);
32869         }
32870
32871         this.lastZIndex = index;
32872     },
32873
32874     /**
32875      * Returns the element for this dialog
32876      * @return {Roo.Element} The underlying dialog Element
32877      */
32878     getEl : function(){
32879         return this.el;
32880     }
32881 });
32882
32883 /**
32884  * @class Roo.DialogManager
32885  * Provides global access to BasicDialogs that have been created and
32886  * support for z-indexing (layering) multiple open dialogs.
32887  */
32888 Roo.DialogManager = function(){
32889     var list = {};
32890     var accessList = [];
32891     var front = null;
32892
32893     // private
32894     var sortDialogs = function(d1, d2){
32895         return (!d1._lastAccess || d1._lastAccess < d2._lastAccess) ? -1 : 1;
32896     };
32897
32898     // private
32899     var orderDialogs = function(){
32900         accessList.sort(sortDialogs);
32901         var seed = Roo.DialogManager.zseed;
32902         for(var i = 0, len = accessList.length; i < len; i++){
32903             var dlg = accessList[i];
32904             if(dlg){
32905                 dlg.setZIndex(seed + (i*10));
32906             }
32907         }
32908     };
32909
32910     return {
32911         /**
32912          * The starting z-index for BasicDialogs (defaults to 9000)
32913          * @type Number The z-index value
32914          */
32915         zseed : 9000,
32916
32917         // private
32918         register : function(dlg){
32919             list[dlg.id] = dlg;
32920             accessList.push(dlg);
32921         },
32922
32923         // private
32924         unregister : function(dlg){
32925             delete list[dlg.id];
32926             var i=0;
32927             var len=0;
32928             if(!accessList.indexOf){
32929                 for(  i = 0, len = accessList.length; i < len; i++){
32930                     if(accessList[i] == dlg){
32931                         accessList.splice(i, 1);
32932                         return;
32933                     }
32934                 }
32935             }else{
32936                  i = accessList.indexOf(dlg);
32937                 if(i != -1){
32938                     accessList.splice(i, 1);
32939                 }
32940             }
32941         },
32942
32943         /**
32944          * Gets a registered dialog by id
32945          * @param {String/Object} id The id of the dialog or a dialog
32946          * @return {Roo.BasicDialog} this
32947          */
32948         get : function(id){
32949             return typeof id == "object" ? id : list[id];
32950         },
32951
32952         /**
32953          * Brings the specified dialog to the front
32954          * @param {String/Object} dlg The id of the dialog or a dialog
32955          * @return {Roo.BasicDialog} this
32956          */
32957         bringToFront : function(dlg){
32958             dlg = this.get(dlg);
32959             if(dlg != front){
32960                 front = dlg;
32961                 dlg._lastAccess = new Date().getTime();
32962                 orderDialogs();
32963             }
32964             return dlg;
32965         },
32966
32967         /**
32968          * Sends the specified dialog to the back
32969          * @param {String/Object} dlg The id of the dialog or a dialog
32970          * @return {Roo.BasicDialog} this
32971          */
32972         sendToBack : function(dlg){
32973             dlg = this.get(dlg);
32974             dlg._lastAccess = -(new Date().getTime());
32975             orderDialogs();
32976             return dlg;
32977         },
32978
32979         /**
32980          * Hides all dialogs
32981          */
32982         hideAll : function(){
32983             for(var id in list){
32984                 if(list[id] && typeof list[id] != "function" && list[id].isVisible()){
32985                     list[id].hide();
32986                 }
32987             }
32988         }
32989     };
32990 }();
32991
32992 /**
32993  * @class Roo.LayoutDialog
32994  * @extends Roo.BasicDialog
32995  * Dialog which provides adjustments for working with a layout in a Dialog.
32996  * Add your necessary layout config options to the dialog's config.<br>
32997  * Example usage (including a nested layout):
32998  * <pre><code>
32999 if(!dialog){
33000     dialog = new Roo.LayoutDialog("download-dlg", {
33001         modal: true,
33002         width:600,
33003         height:450,
33004         shadow:true,
33005         minWidth:500,
33006         minHeight:350,
33007         autoTabs:true,
33008         proxyDrag:true,
33009         // layout config merges with the dialog config
33010         center:{
33011             tabPosition: "top",
33012             alwaysShowTabs: true
33013         }
33014     });
33015     dialog.addKeyListener(27, dialog.hide, dialog);
33016     dialog.setDefaultButton(dialog.addButton("Close", dialog.hide, dialog));
33017     dialog.addButton("Build It!", this.getDownload, this);
33018
33019     // we can even add nested layouts
33020     var innerLayout = new Roo.BorderLayout("dl-inner", {
33021         east: {
33022             initialSize: 200,
33023             autoScroll:true,
33024             split:true
33025         },
33026         center: {
33027             autoScroll:true
33028         }
33029     });
33030     innerLayout.beginUpdate();
33031     innerLayout.add("east", new Roo.ContentPanel("dl-details"));
33032     innerLayout.add("center", new Roo.ContentPanel("selection-panel"));
33033     innerLayout.endUpdate(true);
33034
33035     var layout = dialog.getLayout();
33036     layout.beginUpdate();
33037     layout.add("center", new Roo.ContentPanel("standard-panel",
33038                         {title: "Download the Source", fitToFrame:true}));
33039     layout.add("center", new Roo.NestedLayoutPanel(innerLayout,
33040                {title: "Build your own roo.js"}));
33041     layout.getRegion("center").showPanel(sp);
33042     layout.endUpdate();
33043 }
33044 </code></pre>
33045     * @constructor
33046     * @param {String/HTMLElement/Roo.Element} el The id of or container element, or config
33047     * @param {Object} config configuration options
33048   */
33049 Roo.LayoutDialog = function(el, cfg){
33050     
33051     var config=  cfg;
33052     if (typeof(cfg) == 'undefined') {
33053         config = Roo.apply({}, el);
33054         // not sure why we use documentElement here.. - it should always be body.
33055         // IE7 borks horribly if we use documentElement.
33056         // webkit also does not like documentElement - it creates a body element...
33057         el = Roo.get( document.body || document.documentElement ).createChild();
33058         //config.autoCreate = true;
33059     }
33060     
33061     
33062     config.autoTabs = false;
33063     Roo.LayoutDialog.superclass.constructor.call(this, el, config);
33064     this.body.setStyle({overflow:"hidden", position:"relative"});
33065     this.layout = new Roo.BorderLayout(this.body.dom, config);
33066     this.layout.monitorWindowResize = false;
33067     this.el.addClass("x-dlg-auto-layout");
33068     // fix case when center region overwrites center function
33069     this.center = Roo.BasicDialog.prototype.center;
33070     this.on("show", this.layout.layout, this.layout, true);
33071     if (config.items) {
33072         var xitems = config.items;
33073         delete config.items;
33074         Roo.each(xitems, this.addxtype, this);
33075     }
33076     
33077     
33078 };
33079 Roo.extend(Roo.LayoutDialog, Roo.BasicDialog, {
33080     /**
33081      * Ends update of the layout <strike>and resets display to none</strike>. Use standard beginUpdate/endUpdate on the layout.
33082      * @deprecated
33083      */
33084     endUpdate : function(){
33085         this.layout.endUpdate();
33086     },
33087
33088     /**
33089      * Begins an update of the layout <strike>and sets display to block and visibility to hidden</strike>. Use standard beginUpdate/endUpdate on the layout.
33090      *  @deprecated
33091      */
33092     beginUpdate : function(){
33093         this.layout.beginUpdate();
33094     },
33095
33096     /**
33097      * Get the BorderLayout for this dialog
33098      * @return {Roo.BorderLayout}
33099      */
33100     getLayout : function(){
33101         return this.layout;
33102     },
33103
33104     showEl : function(){
33105         Roo.LayoutDialog.superclass.showEl.apply(this, arguments);
33106         if(Roo.isIE7){
33107             this.layout.layout();
33108         }
33109     },
33110
33111     // private
33112     // Use the syncHeightBeforeShow config option to control this automatically
33113     syncBodyHeight : function(){
33114         Roo.LayoutDialog.superclass.syncBodyHeight.call(this);
33115         if(this.layout){this.layout.layout();}
33116     },
33117     
33118       /**
33119      * Add an xtype element (actually adds to the layout.)
33120      * @return {Object} xdata xtype object data.
33121      */
33122     
33123     addxtype : function(c) {
33124         return this.layout.addxtype(c);
33125     }
33126 });/*
33127  * Based on:
33128  * Ext JS Library 1.1.1
33129  * Copyright(c) 2006-2007, Ext JS, LLC.
33130  *
33131  * Originally Released Under LGPL - original licence link has changed is not relivant.
33132  *
33133  * Fork - LGPL
33134  * <script type="text/javascript">
33135  */
33136  
33137 /**
33138  * @class Roo.MessageBox
33139  * Utility class for generating different styles of message boxes.  The alias Roo.Msg can also be used.
33140  * Example usage:
33141  *<pre><code>
33142 // Basic alert:
33143 Roo.Msg.alert('Status', 'Changes saved successfully.');
33144
33145 // Prompt for user data:
33146 Roo.Msg.prompt('Name', 'Please enter your name:', function(btn, text){
33147     if (btn == 'ok'){
33148         // process text value...
33149     }
33150 });
33151
33152 // Show a dialog using config options:
33153 Roo.Msg.show({
33154    title:'Save Changes?',
33155    msg: 'Your are closing a tab that has unsaved changes. Would you like to save your changes?',
33156    buttons: Roo.Msg.YESNOCANCEL,
33157    fn: processResult,
33158    animEl: 'elId'
33159 });
33160 </code></pre>
33161  * @singleton
33162  */
33163 Roo.MessageBox = function(){
33164     var dlg, opt, mask, waitTimer;
33165     var bodyEl, msgEl, textboxEl, textareaEl, progressEl, pp;
33166     var buttons, activeTextEl, bwidth;
33167
33168     // private
33169     var handleButton = function(button){
33170         dlg.hide();
33171         Roo.callback(opt.fn, opt.scope||window, [button, activeTextEl.dom.value], 1);
33172     };
33173
33174     // private
33175     var handleHide = function(){
33176         if(opt && opt.cls){
33177             dlg.el.removeClass(opt.cls);
33178         }
33179         if(waitTimer){
33180             Roo.TaskMgr.stop(waitTimer);
33181             waitTimer = null;
33182         }
33183     };
33184
33185     // private
33186     var updateButtons = function(b){
33187         var width = 0;
33188         if(!b){
33189             buttons["ok"].hide();
33190             buttons["cancel"].hide();
33191             buttons["yes"].hide();
33192             buttons["no"].hide();
33193             dlg.footer.dom.style.display = 'none';
33194             return width;
33195         }
33196         dlg.footer.dom.style.display = '';
33197         for(var k in buttons){
33198             if(typeof buttons[k] != "function"){
33199                 if(b[k]){
33200                     buttons[k].show();
33201                     buttons[k].setText(typeof b[k] == "string" ? b[k] : Roo.MessageBox.buttonText[k]);
33202                     width += buttons[k].el.getWidth()+15;
33203                 }else{
33204                     buttons[k].hide();
33205                 }
33206             }
33207         }
33208         return width;
33209     };
33210
33211     // private
33212     var handleEsc = function(d, k, e){
33213         if(opt && opt.closable !== false){
33214             dlg.hide();
33215         }
33216         if(e){
33217             e.stopEvent();
33218         }
33219     };
33220
33221     return {
33222         /**
33223          * Returns a reference to the underlying {@link Roo.BasicDialog} element
33224          * @return {Roo.BasicDialog} The BasicDialog element
33225          */
33226         getDialog : function(){
33227            if(!dlg){
33228                 dlg = new Roo.BasicDialog("x-msg-box", {
33229                     autoCreate : true,
33230                     shadow: true,
33231                     draggable: true,
33232                     resizable:false,
33233                     constraintoviewport:false,
33234                     fixedcenter:true,
33235                     collapsible : false,
33236                     shim:true,
33237                     modal: true,
33238                     width:400, height:100,
33239                     buttonAlign:"center",
33240                     closeClick : function(){
33241                         if(opt && opt.buttons && opt.buttons.no && !opt.buttons.cancel){
33242                             handleButton("no");
33243                         }else{
33244                             handleButton("cancel");
33245                         }
33246                     }
33247                 });
33248                 dlg.on("hide", handleHide);
33249                 mask = dlg.mask;
33250                 dlg.addKeyListener(27, handleEsc);
33251                 buttons = {};
33252                 var bt = this.buttonText;
33253                 buttons["ok"] = dlg.addButton(bt["ok"], handleButton.createCallback("ok"));
33254                 buttons["yes"] = dlg.addButton(bt["yes"], handleButton.createCallback("yes"));
33255                 buttons["no"] = dlg.addButton(bt["no"], handleButton.createCallback("no"));
33256                 buttons["cancel"] = dlg.addButton(bt["cancel"], handleButton.createCallback("cancel"));
33257                 bodyEl = dlg.body.createChild({
33258
33259                     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>'
33260                 });
33261                 msgEl = bodyEl.dom.firstChild;
33262                 textboxEl = Roo.get(bodyEl.dom.childNodes[2]);
33263                 textboxEl.enableDisplayMode();
33264                 textboxEl.addKeyListener([10,13], function(){
33265                     if(dlg.isVisible() && opt && opt.buttons){
33266                         if(opt.buttons.ok){
33267                             handleButton("ok");
33268                         }else if(opt.buttons.yes){
33269                             handleButton("yes");
33270                         }
33271                     }
33272                 });
33273                 textareaEl = Roo.get(bodyEl.dom.childNodes[3]);
33274                 textareaEl.enableDisplayMode();
33275                 progressEl = Roo.get(bodyEl.dom.childNodes[4]);
33276                 progressEl.enableDisplayMode();
33277                 var pf = progressEl.dom.firstChild;
33278                 if (pf) {
33279                     pp = Roo.get(pf.firstChild);
33280                     pp.setHeight(pf.offsetHeight);
33281                 }
33282                 
33283             }
33284             return dlg;
33285         },
33286
33287         /**
33288          * Updates the message box body text
33289          * @param {String} text (optional) Replaces the message box element's innerHTML with the specified string (defaults to
33290          * the XHTML-compliant non-breaking space character '&amp;#160;')
33291          * @return {Roo.MessageBox} This message box
33292          */
33293         updateText : function(text){
33294             if(!dlg.isVisible() && !opt.width){
33295                 dlg.resizeTo(this.maxWidth, 100); // resize first so content is never clipped from previous shows
33296             }
33297             msgEl.innerHTML = text || '&#160;';
33298       
33299             var cw =  Math.max(msgEl.offsetWidth, msgEl.parentNode.scrollWidth);
33300             //Roo.log("guesed size: " + JSON.stringify([cw,msgEl.offsetWidth, msgEl.parentNode.scrollWidth]));
33301             var w = Math.max(
33302                     Math.min(opt.width || cw , this.maxWidth), 
33303                     Math.max(opt.minWidth || this.minWidth, bwidth)
33304             );
33305             if(opt.prompt){
33306                 activeTextEl.setWidth(w);
33307             }
33308             if(dlg.isVisible()){
33309                 dlg.fixedcenter = false;
33310             }
33311             // to big, make it scroll. = But as usual stupid IE does not support
33312             // !important..
33313             
33314             if ( bodyEl.getHeight() > (Roo.lib.Dom.getViewHeight() - 100)) {
33315                 bodyEl.setHeight ( Roo.lib.Dom.getViewHeight() - 100 );
33316                 bodyEl.dom.style.overflowY = 'auto' + ( Roo.isIE ? '' : ' !important');
33317             } else {
33318                 bodyEl.dom.style.height = '';
33319                 bodyEl.dom.style.overflowY = '';
33320             }
33321             if (cw > w) {
33322                 bodyEl.dom.style.get = 'auto' + ( Roo.isIE ? '' : ' !important');
33323             } else {
33324                 bodyEl.dom.style.overflowX = '';
33325             }
33326             
33327             dlg.setContentSize(w, bodyEl.getHeight());
33328             if(dlg.isVisible()){
33329                 dlg.fixedcenter = true;
33330             }
33331             return this;
33332         },
33333
33334         /**
33335          * Updates a progress-style message box's text and progress bar.  Only relevant on message boxes
33336          * initiated via {@link Roo.MessageBox#progress} or by calling {@link Roo.MessageBox#show} with progress: true.
33337          * @param {Number} value Any number between 0 and 1 (e.g., .5)
33338          * @param {String} text (optional) If defined, the message box's body text is replaced with the specified string (defaults to undefined)
33339          * @return {Roo.MessageBox} This message box
33340          */
33341         updateProgress : function(value, text){
33342             if(text){
33343                 this.updateText(text);
33344             }
33345             if (pp) { // weird bug on my firefox - for some reason this is not defined
33346                 pp.setWidth(Math.floor(value*progressEl.dom.firstChild.offsetWidth));
33347             }
33348             return this;
33349         },        
33350
33351         /**
33352          * Returns true if the message box is currently displayed
33353          * @return {Boolean} True if the message box is visible, else false
33354          */
33355         isVisible : function(){
33356             return dlg && dlg.isVisible();  
33357         },
33358
33359         /**
33360          * Hides the message box if it is displayed
33361          */
33362         hide : function(){
33363             if(this.isVisible()){
33364                 dlg.hide();
33365             }  
33366         },
33367
33368         /**
33369          * Displays a new message box, or reinitializes an existing message box, based on the config options
33370          * passed in. All functions (e.g. prompt, alert, etc) on MessageBox call this function internally.
33371          * The following config object properties are supported:
33372          * <pre>
33373 Property    Type             Description
33374 ----------  ---------------  ------------------------------------------------------------------------------------
33375 animEl            String/Element   An id or Element from which the message box should animate as it opens and
33376                                    closes (defaults to undefined)
33377 buttons           Object/Boolean   A button config object (e.g., Roo.MessageBox.OKCANCEL or {ok:'Foo',
33378                                    cancel:'Bar'}), or false to not show any buttons (defaults to false)
33379 closable          Boolean          False to hide the top-right close button (defaults to true).  Note that
33380                                    progress and wait dialogs will ignore this property and always hide the
33381                                    close button as they can only be closed programmatically.
33382 cls               String           A custom CSS class to apply to the message box element
33383 defaultTextHeight Number           The default height in pixels of the message box's multiline textarea if
33384                                    displayed (defaults to 75)
33385 fn                Function         A callback function to execute after closing the dialog.  The arguments to the
33386                                    function will be btn (the name of the button that was clicked, if applicable,
33387                                    e.g. "ok"), and text (the value of the active text field, if applicable).
33388                                    Progress and wait dialogs will ignore this option since they do not respond to
33389                                    user actions and can only be closed programmatically, so any required function
33390                                    should be called by the same code after it closes the dialog.
33391 icon              String           A CSS class that provides a background image to be used as an icon for
33392                                    the dialog (e.g., Roo.MessageBox.WARNING or 'custom-class', defaults to '')
33393 maxWidth          Number           The maximum width in pixels of the message box (defaults to 600)
33394 minWidth          Number           The minimum width in pixels of the message box (defaults to 100)
33395 modal             Boolean          False to allow user interaction with the page while the message box is
33396                                    displayed (defaults to true)
33397 msg               String           A string that will replace the existing message box body text (defaults
33398                                    to the XHTML-compliant non-breaking space character '&#160;')
33399 multiline         Boolean          True to prompt the user to enter multi-line text (defaults to false)
33400 progress          Boolean          True to display a progress bar (defaults to false)
33401 progressText      String           The text to display inside the progress bar if progress = true (defaults to '')
33402 prompt            Boolean          True to prompt the user to enter single-line text (defaults to false)
33403 proxyDrag         Boolean          True to display a lightweight proxy while dragging (defaults to false)
33404 title             String           The title text
33405 value             String           The string value to set into the active textbox element if displayed
33406 wait              Boolean          True to display a progress bar (defaults to false)
33407 width             Number           The width of the dialog in pixels
33408 </pre>
33409          *
33410          * Example usage:
33411          * <pre><code>
33412 Roo.Msg.show({
33413    title: 'Address',
33414    msg: 'Please enter your address:',
33415    width: 300,
33416    buttons: Roo.MessageBox.OKCANCEL,
33417    multiline: true,
33418    fn: saveAddress,
33419    animEl: 'addAddressBtn'
33420 });
33421 </code></pre>
33422          * @param {Object} config Configuration options
33423          * @return {Roo.MessageBox} This message box
33424          */
33425         show : function(options)
33426         {
33427             
33428             // this causes nightmares if you show one dialog after another
33429             // especially on callbacks..
33430              
33431             if(this.isVisible()){
33432                 
33433                 this.hide();
33434                 Roo.log("[Roo.Messagebox] Show called while message displayed:" );
33435                 Roo.log("Old Dialog Message:" +  msgEl.innerHTML );
33436                 Roo.log("New Dialog Message:" +  options.msg )
33437                 //this.alert("ERROR", "Multiple dialogs where displayed at the same time");
33438                 //throw "Roo.MessageBox ERROR : Multiple dialogs where displayed at the same time";
33439                 
33440             }
33441             var d = this.getDialog();
33442             opt = options;
33443             d.setTitle(opt.title || "&#160;");
33444             d.close.setDisplayed(opt.closable !== false);
33445             activeTextEl = textboxEl;
33446             opt.prompt = opt.prompt || (opt.multiline ? true : false);
33447             if(opt.prompt){
33448                 if(opt.multiline){
33449                     textboxEl.hide();
33450                     textareaEl.show();
33451                     textareaEl.setHeight(typeof opt.multiline == "number" ?
33452                         opt.multiline : this.defaultTextHeight);
33453                     activeTextEl = textareaEl;
33454                 }else{
33455                     textboxEl.show();
33456                     textareaEl.hide();
33457                 }
33458             }else{
33459                 textboxEl.hide();
33460                 textareaEl.hide();
33461             }
33462             progressEl.setDisplayed(opt.progress === true);
33463             this.updateProgress(0);
33464             activeTextEl.dom.value = opt.value || "";
33465             if(opt.prompt){
33466                 dlg.setDefaultButton(activeTextEl);
33467             }else{
33468                 var bs = opt.buttons;
33469                 var db = null;
33470                 if(bs && bs.ok){
33471                     db = buttons["ok"];
33472                 }else if(bs && bs.yes){
33473                     db = buttons["yes"];
33474                 }
33475                 dlg.setDefaultButton(db);
33476             }
33477             bwidth = updateButtons(opt.buttons);
33478             this.updateText(opt.msg);
33479             if(opt.cls){
33480                 d.el.addClass(opt.cls);
33481             }
33482             d.proxyDrag = opt.proxyDrag === true;
33483             d.modal = opt.modal !== false;
33484             d.mask = opt.modal !== false ? mask : false;
33485             if(!d.isVisible()){
33486                 // force it to the end of the z-index stack so it gets a cursor in FF
33487                 document.body.appendChild(dlg.el.dom);
33488                 d.animateTarget = null;
33489                 d.show(options.animEl);
33490             }
33491             return this;
33492         },
33493
33494         /**
33495          * Displays a message box with a progress bar.  This message box has no buttons and is not closeable by
33496          * the user.  You are responsible for updating the progress bar as needed via {@link Roo.MessageBox#updateProgress}
33497          * and closing the message box when the process is complete.
33498          * @param {String} title The title bar text
33499          * @param {String} msg The message box body text
33500          * @return {Roo.MessageBox} This message box
33501          */
33502         progress : function(title, msg){
33503             this.show({
33504                 title : title,
33505                 msg : msg,
33506                 buttons: false,
33507                 progress:true,
33508                 closable:false,
33509                 minWidth: this.minProgressWidth,
33510                 modal : true
33511             });
33512             return this;
33513         },
33514
33515         /**
33516          * Displays a standard read-only message box with an OK button (comparable to the basic JavaScript Window.alert).
33517          * If a callback function is passed it will be called after the user clicks the button, and the
33518          * id of the button that was clicked will be passed as the only parameter to the callback
33519          * (could also be the top-right close button).
33520          * @param {String} title The title bar text
33521          * @param {String} msg The message box body text
33522          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33523          * @param {Object} scope (optional) The scope of the callback function
33524          * @return {Roo.MessageBox} This message box
33525          */
33526         alert : function(title, msg, fn, scope){
33527             this.show({
33528                 title : title,
33529                 msg : msg,
33530                 buttons: this.OK,
33531                 fn: fn,
33532                 scope : scope,
33533                 modal : true
33534             });
33535             return this;
33536         },
33537
33538         /**
33539          * Displays a message box with an infinitely auto-updating progress bar.  This can be used to block user
33540          * interaction while waiting for a long-running process to complete that does not have defined intervals.
33541          * You are responsible for closing the message box when the process is complete.
33542          * @param {String} msg The message box body text
33543          * @param {String} title (optional) The title bar text
33544          * @return {Roo.MessageBox} This message box
33545          */
33546         wait : function(msg, title){
33547             this.show({
33548                 title : title,
33549                 msg : msg,
33550                 buttons: false,
33551                 closable:false,
33552                 progress:true,
33553                 modal:true,
33554                 width:300,
33555                 wait:true
33556             });
33557             waitTimer = Roo.TaskMgr.start({
33558                 run: function(i){
33559                     Roo.MessageBox.updateProgress(((((i+20)%20)+1)*5)*.01);
33560                 },
33561                 interval: 1000
33562             });
33563             return this;
33564         },
33565
33566         /**
33567          * Displays a confirmation message box with Yes and No buttons (comparable to JavaScript's Window.confirm).
33568          * If a callback function is passed it will be called after the user clicks either button, and the id of the
33569          * button that was clicked will be passed as the only parameter to the callback (could also be the top-right close button).
33570          * @param {String} title The title bar text
33571          * @param {String} msg The message box body text
33572          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33573          * @param {Object} scope (optional) The scope of the callback function
33574          * @return {Roo.MessageBox} This message box
33575          */
33576         confirm : function(title, msg, fn, scope){
33577             this.show({
33578                 title : title,
33579                 msg : msg,
33580                 buttons: this.YESNO,
33581                 fn: fn,
33582                 scope : scope,
33583                 modal : true
33584             });
33585             return this;
33586         },
33587
33588         /**
33589          * Displays a message box with OK and Cancel buttons prompting the user to enter some text (comparable to
33590          * JavaScript's Window.prompt).  The prompt can be a single-line or multi-line textbox.  If a callback function
33591          * is passed it will be called after the user clicks either button, and the id of the button that was clicked
33592          * (could also be the top-right close button) and the text that was entered will be passed as the two
33593          * parameters to the callback.
33594          * @param {String} title The title bar text
33595          * @param {String} msg The message box body text
33596          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33597          * @param {Object} scope (optional) The scope of the callback function
33598          * @param {Boolean/Number} multiline (optional) True to create a multiline textbox using the defaultTextHeight
33599          * property, or the height in pixels to create the textbox (defaults to false / single-line)
33600          * @return {Roo.MessageBox} This message box
33601          */
33602         prompt : function(title, msg, fn, scope, multiline){
33603             this.show({
33604                 title : title,
33605                 msg : msg,
33606                 buttons: this.OKCANCEL,
33607                 fn: fn,
33608                 minWidth:250,
33609                 scope : scope,
33610                 prompt:true,
33611                 multiline: multiline,
33612                 modal : true
33613             });
33614             return this;
33615         },
33616
33617         /**
33618          * Button config that displays a single OK button
33619          * @type Object
33620          */
33621         OK : {ok:true},
33622         /**
33623          * Button config that displays Yes and No buttons
33624          * @type Object
33625          */
33626         YESNO : {yes:true, no:true},
33627         /**
33628          * Button config that displays OK and Cancel buttons
33629          * @type Object
33630          */
33631         OKCANCEL : {ok:true, cancel:true},
33632         /**
33633          * Button config that displays Yes, No and Cancel buttons
33634          * @type Object
33635          */
33636         YESNOCANCEL : {yes:true, no:true, cancel:true},
33637
33638         /**
33639          * The default height in pixels of the message box's multiline textarea if displayed (defaults to 75)
33640          * @type Number
33641          */
33642         defaultTextHeight : 75,
33643         /**
33644          * The maximum width in pixels of the message box (defaults to 600)
33645          * @type Number
33646          */
33647         maxWidth : 600,
33648         /**
33649          * The minimum width in pixels of the message box (defaults to 100)
33650          * @type Number
33651          */
33652         minWidth : 100,
33653         /**
33654          * The minimum width in pixels of the message box if it is a progress-style dialog.  This is useful
33655          * for setting a different minimum width than text-only dialogs may need (defaults to 250)
33656          * @type Number
33657          */
33658         minProgressWidth : 250,
33659         /**
33660          * An object containing the default button text strings that can be overriden for localized language support.
33661          * Supported properties are: ok, cancel, yes and no.
33662          * Customize the default text like so: Roo.MessageBox.buttonText.yes = "S?";
33663          * @type Object
33664          */
33665         buttonText : {
33666             ok : "OK",
33667             cancel : "Cancel",
33668             yes : "Yes",
33669             no : "No"
33670         }
33671     };
33672 }();
33673
33674 /**
33675  * Shorthand for {@link Roo.MessageBox}
33676  */
33677 Roo.Msg = Roo.MessageBox;/*
33678  * Based on:
33679  * Ext JS Library 1.1.1
33680  * Copyright(c) 2006-2007, Ext JS, LLC.
33681  *
33682  * Originally Released Under LGPL - original licence link has changed is not relivant.
33683  *
33684  * Fork - LGPL
33685  * <script type="text/javascript">
33686  */
33687 /**
33688  * @class Roo.QuickTips
33689  * Provides attractive and customizable tooltips for any element.
33690  * @singleton
33691  */
33692 Roo.QuickTips = function(){
33693     var el, tipBody, tipBodyText, tipTitle, tm, cfg, close, tagEls = {}, esc, removeCls = null, bdLeft, bdRight;
33694     var ce, bd, xy, dd;
33695     var visible = false, disabled = true, inited = false;
33696     var showProc = 1, hideProc = 1, dismissProc = 1, locks = [];
33697     
33698     var onOver = function(e){
33699         if(disabled){
33700             return;
33701         }
33702         var t = e.getTarget();
33703         if(!t || t.nodeType !== 1 || t == document || t == document.body){
33704             return;
33705         }
33706         if(ce && t == ce.el){
33707             clearTimeout(hideProc);
33708             return;
33709         }
33710         if(t && tagEls[t.id]){
33711             tagEls[t.id].el = t;
33712             showProc = show.defer(tm.showDelay, tm, [tagEls[t.id]]);
33713             return;
33714         }
33715         var ttp, et = Roo.fly(t);
33716         var ns = cfg.namespace;
33717         if(tm.interceptTitles && t.title){
33718             ttp = t.title;
33719             t.qtip = ttp;
33720             t.removeAttribute("title");
33721             e.preventDefault();
33722         }else{
33723             ttp = t.qtip || et.getAttributeNS(ns, cfg.attribute) || et.getAttributeNS(cfg.alt_namespace, cfg.attribute) ;
33724         }
33725         if(ttp){
33726             showProc = show.defer(tm.showDelay, tm, [{
33727                 el: t, 
33728                 text: ttp.replace(/\\n/g,'<br/>'),
33729                 width: et.getAttributeNS(ns, cfg.width),
33730                 autoHide: et.getAttributeNS(ns, cfg.hide) != "user",
33731                 title: et.getAttributeNS(ns, cfg.title),
33732                     cls: et.getAttributeNS(ns, cfg.cls)
33733             }]);
33734         }
33735     };
33736     
33737     var onOut = function(e){
33738         clearTimeout(showProc);
33739         var t = e.getTarget();
33740         if(t && ce && ce.el == t && (tm.autoHide && ce.autoHide !== false)){
33741             hideProc = setTimeout(hide, tm.hideDelay);
33742         }
33743     };
33744     
33745     var onMove = function(e){
33746         if(disabled){
33747             return;
33748         }
33749         xy = e.getXY();
33750         xy[1] += 18;
33751         if(tm.trackMouse && ce){
33752             el.setXY(xy);
33753         }
33754     };
33755     
33756     var onDown = function(e){
33757         clearTimeout(showProc);
33758         clearTimeout(hideProc);
33759         if(!e.within(el)){
33760             if(tm.hideOnClick){
33761                 hide();
33762                 tm.disable();
33763                 tm.enable.defer(100, tm);
33764             }
33765         }
33766     };
33767     
33768     var getPad = function(){
33769         return 2;//bdLeft.getPadding('l')+bdRight.getPadding('r');
33770     };
33771
33772     var show = function(o){
33773         if(disabled){
33774             return;
33775         }
33776         clearTimeout(dismissProc);
33777         ce = o;
33778         if(removeCls){ // in case manually hidden
33779             el.removeClass(removeCls);
33780             removeCls = null;
33781         }
33782         if(ce.cls){
33783             el.addClass(ce.cls);
33784             removeCls = ce.cls;
33785         }
33786         if(ce.title){
33787             tipTitle.update(ce.title);
33788             tipTitle.show();
33789         }else{
33790             tipTitle.update('');
33791             tipTitle.hide();
33792         }
33793         el.dom.style.width  = tm.maxWidth+'px';
33794         //tipBody.dom.style.width = '';
33795         tipBodyText.update(o.text);
33796         var p = getPad(), w = ce.width;
33797         if(!w){
33798             var td = tipBodyText.dom;
33799             var aw = Math.max(td.offsetWidth, td.clientWidth, td.scrollWidth);
33800             if(aw > tm.maxWidth){
33801                 w = tm.maxWidth;
33802             }else if(aw < tm.minWidth){
33803                 w = tm.minWidth;
33804             }else{
33805                 w = aw;
33806             }
33807         }
33808         //tipBody.setWidth(w);
33809         el.setWidth(parseInt(w, 10) + p);
33810         if(ce.autoHide === false){
33811             close.setDisplayed(true);
33812             if(dd){
33813                 dd.unlock();
33814             }
33815         }else{
33816             close.setDisplayed(false);
33817             if(dd){
33818                 dd.lock();
33819             }
33820         }
33821         if(xy){
33822             el.avoidY = xy[1]-18;
33823             el.setXY(xy);
33824         }
33825         if(tm.animate){
33826             el.setOpacity(.1);
33827             el.setStyle("visibility", "visible");
33828             el.fadeIn({callback: afterShow});
33829         }else{
33830             afterShow();
33831         }
33832     };
33833     
33834     var afterShow = function(){
33835         if(ce){
33836             el.show();
33837             esc.enable();
33838             if(tm.autoDismiss && ce.autoHide !== false){
33839                 dismissProc = setTimeout(hide, tm.autoDismissDelay);
33840             }
33841         }
33842     };
33843     
33844     var hide = function(noanim){
33845         clearTimeout(dismissProc);
33846         clearTimeout(hideProc);
33847         ce = null;
33848         if(el.isVisible()){
33849             esc.disable();
33850             if(noanim !== true && tm.animate){
33851                 el.fadeOut({callback: afterHide});
33852             }else{
33853                 afterHide();
33854             } 
33855         }
33856     };
33857     
33858     var afterHide = function(){
33859         el.hide();
33860         if(removeCls){
33861             el.removeClass(removeCls);
33862             removeCls = null;
33863         }
33864     };
33865     
33866     return {
33867         /**
33868         * @cfg {Number} minWidth
33869         * The minimum width of the quick tip (defaults to 40)
33870         */
33871        minWidth : 40,
33872         /**
33873         * @cfg {Number} maxWidth
33874         * The maximum width of the quick tip (defaults to 300)
33875         */
33876        maxWidth : 300,
33877         /**
33878         * @cfg {Boolean} interceptTitles
33879         * True to automatically use the element's DOM title value if available (defaults to false)
33880         */
33881        interceptTitles : false,
33882         /**
33883         * @cfg {Boolean} trackMouse
33884         * True to have the quick tip follow the mouse as it moves over the target element (defaults to false)
33885         */
33886        trackMouse : false,
33887         /**
33888         * @cfg {Boolean} hideOnClick
33889         * True to hide the quick tip if the user clicks anywhere in the document (defaults to true)
33890         */
33891        hideOnClick : true,
33892         /**
33893         * @cfg {Number} showDelay
33894         * Delay in milliseconds before the quick tip displays after the mouse enters the target element (defaults to 500)
33895         */
33896        showDelay : 500,
33897         /**
33898         * @cfg {Number} hideDelay
33899         * Delay in milliseconds before the quick tip hides when autoHide = true (defaults to 200)
33900         */
33901        hideDelay : 200,
33902         /**
33903         * @cfg {Boolean} autoHide
33904         * True to automatically hide the quick tip after the mouse exits the target element (defaults to true).
33905         * Used in conjunction with hideDelay.
33906         */
33907        autoHide : true,
33908         /**
33909         * @cfg {Boolean}
33910         * True to automatically hide the quick tip after a set period of time, regardless of the user's actions
33911         * (defaults to true).  Used in conjunction with autoDismissDelay.
33912         */
33913        autoDismiss : true,
33914         /**
33915         * @cfg {Number}
33916         * Delay in milliseconds before the quick tip hides when autoDismiss = true (defaults to 5000)
33917         */
33918        autoDismissDelay : 5000,
33919        /**
33920         * @cfg {Boolean} animate
33921         * True to turn on fade animation. Defaults to false (ClearType/scrollbar flicker issues in IE7).
33922         */
33923        animate : false,
33924
33925        /**
33926         * @cfg {String} title
33927         * Title text to display (defaults to '').  This can be any valid HTML markup.
33928         */
33929         title: '',
33930        /**
33931         * @cfg {String} text
33932         * Body text to display (defaults to '').  This can be any valid HTML markup.
33933         */
33934         text : '',
33935        /**
33936         * @cfg {String} cls
33937         * A CSS class to apply to the base quick tip element (defaults to '').
33938         */
33939         cls : '',
33940        /**
33941         * @cfg {Number} width
33942         * Width in pixels of the quick tip (defaults to auto).  Width will be ignored if it exceeds the bounds of
33943         * minWidth or maxWidth.
33944         */
33945         width : null,
33946
33947     /**
33948      * Initialize and enable QuickTips for first use.  This should be called once before the first attempt to access
33949      * or display QuickTips in a page.
33950      */
33951        init : function(){
33952           tm = Roo.QuickTips;
33953           cfg = tm.tagConfig;
33954           if(!inited){
33955               if(!Roo.isReady){ // allow calling of init() before onReady
33956                   Roo.onReady(Roo.QuickTips.init, Roo.QuickTips);
33957                   return;
33958               }
33959               el = new Roo.Layer({cls:"x-tip", shadow:"drop", shim: true, constrain:true, shadowOffset:4});
33960               el.fxDefaults = {stopFx: true};
33961               // maximum custom styling
33962               //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>');
33963               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>');              
33964               tipTitle = el.child('h3');
33965               tipTitle.enableDisplayMode("block");
33966               tipBody = el.child('div.x-tip-bd');
33967               tipBodyText = el.child('div.x-tip-bd-inner');
33968               //bdLeft = el.child('div.x-tip-bd-left');
33969               //bdRight = el.child('div.x-tip-bd-right');
33970               close = el.child('div.x-tip-close');
33971               close.enableDisplayMode("block");
33972               close.on("click", hide);
33973               var d = Roo.get(document);
33974               d.on("mousedown", onDown);
33975               d.on("mouseover", onOver);
33976               d.on("mouseout", onOut);
33977               d.on("mousemove", onMove);
33978               esc = d.addKeyListener(27, hide);
33979               esc.disable();
33980               if(Roo.dd.DD){
33981                   dd = el.initDD("default", null, {
33982                       onDrag : function(){
33983                           el.sync();  
33984                       }
33985                   });
33986                   dd.setHandleElId(tipTitle.id);
33987                   dd.lock();
33988               }
33989               inited = true;
33990           }
33991           this.enable(); 
33992        },
33993
33994     /**
33995      * Configures a new quick tip instance and assigns it to a target element.  The following config options
33996      * are supported:
33997      * <pre>
33998 Property    Type                   Description
33999 ----------  ---------------------  ------------------------------------------------------------------------
34000 target      Element/String/Array   An Element, id or array of ids that this quick tip should be tied to
34001      * </ul>
34002      * @param {Object} config The config object
34003      */
34004        register : function(config){
34005            var cs = config instanceof Array ? config : arguments;
34006            for(var i = 0, len = cs.length; i < len; i++) {
34007                var c = cs[i];
34008                var target = c.target;
34009                if(target){
34010                    if(target instanceof Array){
34011                        for(var j = 0, jlen = target.length; j < jlen; j++){
34012                            tagEls[target[j]] = c;
34013                        }
34014                    }else{
34015                        tagEls[typeof target == 'string' ? target : Roo.id(target)] = c;
34016                    }
34017                }
34018            }
34019        },
34020
34021     /**
34022      * Removes this quick tip from its element and destroys it.
34023      * @param {String/HTMLElement/Element} el The element from which the quick tip is to be removed.
34024      */
34025        unregister : function(el){
34026            delete tagEls[Roo.id(el)];
34027        },
34028
34029     /**
34030      * Enable this quick tip.
34031      */
34032        enable : function(){
34033            if(inited && disabled){
34034                locks.pop();
34035                if(locks.length < 1){
34036                    disabled = false;
34037                }
34038            }
34039        },
34040
34041     /**
34042      * Disable this quick tip.
34043      */
34044        disable : function(){
34045           disabled = true;
34046           clearTimeout(showProc);
34047           clearTimeout(hideProc);
34048           clearTimeout(dismissProc);
34049           if(ce){
34050               hide(true);
34051           }
34052           locks.push(1);
34053        },
34054
34055     /**
34056      * Returns true if the quick tip is enabled, else false.
34057      */
34058        isEnabled : function(){
34059             return !disabled;
34060        },
34061
34062         // private
34063        tagConfig : {
34064            namespace : "roo", // was ext?? this may break..
34065            alt_namespace : "ext",
34066            attribute : "qtip",
34067            width : "width",
34068            target : "target",
34069            title : "qtitle",
34070            hide : "hide",
34071            cls : "qclass"
34072        }
34073    };
34074 }();
34075
34076 // backwards compat
34077 Roo.QuickTips.tips = Roo.QuickTips.register;/*
34078  * Based on:
34079  * Ext JS Library 1.1.1
34080  * Copyright(c) 2006-2007, Ext JS, LLC.
34081  *
34082  * Originally Released Under LGPL - original licence link has changed is not relivant.
34083  *
34084  * Fork - LGPL
34085  * <script type="text/javascript">
34086  */
34087  
34088
34089 /**
34090  * @class Roo.tree.TreePanel
34091  * @extends Roo.data.Tree
34092
34093  * @cfg {Boolean} rootVisible false to hide the root node (defaults to true)
34094  * @cfg {Boolean} lines false to disable tree lines (defaults to true)
34095  * @cfg {Boolean} enableDD true to enable drag and drop
34096  * @cfg {Boolean} enableDrag true to enable just drag
34097  * @cfg {Boolean} enableDrop true to enable just drop
34098  * @cfg {Object} dragConfig Custom config to pass to the {@link Roo.tree.TreeDragZone} instance
34099  * @cfg {Object} dropConfig Custom config to pass to the {@link Roo.tree.TreeDropZone} instance
34100  * @cfg {String} ddGroup The DD group this TreePanel belongs to
34101  * @cfg {String} ddAppendOnly True if the tree should only allow append drops (use for trees which are sorted)
34102  * @cfg {Boolean} ddScroll true to enable YUI body scrolling
34103  * @cfg {Boolean} containerScroll true to register this container with ScrollManager
34104  * @cfg {Boolean} hlDrop false to disable node highlight on drop (defaults to the value of Roo.enableFx)
34105  * @cfg {String} hlColor The color of the node highlight (defaults to C3DAF9)
34106  * @cfg {Boolean} animate true to enable animated expand/collapse (defaults to the value of Roo.enableFx)
34107  * @cfg {Boolean} singleExpand true if only 1 node per branch may be expanded
34108  * @cfg {Boolean} selModel A tree selection model to use with this TreePanel (defaults to a {@link Roo.tree.DefaultSelectionModel})
34109  * @cfg {Boolean} loader A TreeLoader for use with this TreePanel
34110  * @cfg {Object|Roo.tree.TreeEditor} editor The TreeEditor or xtype data to display when clicked.
34111  * @cfg {String} pathSeparator The token used to separate sub-paths in path strings (defaults to '/')
34112  * @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>
34113  * @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>
34114  * 
34115  * @constructor
34116  * @param {String/HTMLElement/Element} el The container element
34117  * @param {Object} config
34118  */
34119 Roo.tree.TreePanel = function(el, config){
34120     var root = false;
34121     var loader = false;
34122     if (config.root) {
34123         root = config.root;
34124         delete config.root;
34125     }
34126     if (config.loader) {
34127         loader = config.loader;
34128         delete config.loader;
34129     }
34130     
34131     Roo.apply(this, config);
34132     Roo.tree.TreePanel.superclass.constructor.call(this);
34133     this.el = Roo.get(el);
34134     this.el.addClass('x-tree');
34135     //console.log(root);
34136     if (root) {
34137         this.setRootNode( Roo.factory(root, Roo.tree));
34138     }
34139     if (loader) {
34140         this.loader = Roo.factory(loader, Roo.tree);
34141     }
34142    /**
34143     * Read-only. The id of the container element becomes this TreePanel's id.
34144     */
34145     this.id = this.el.id;
34146     this.addEvents({
34147         /**
34148         * @event beforeload
34149         * Fires before a node is loaded, return false to cancel
34150         * @param {Node} node The node being loaded
34151         */
34152         "beforeload" : true,
34153         /**
34154         * @event load
34155         * Fires when a node is loaded
34156         * @param {Node} node The node that was loaded
34157         */
34158         "load" : true,
34159         /**
34160         * @event textchange
34161         * Fires when the text for a node is changed
34162         * @param {Node} node The node
34163         * @param {String} text The new text
34164         * @param {String} oldText The old text
34165         */
34166         "textchange" : true,
34167         /**
34168         * @event beforeexpand
34169         * Fires before a node is expanded, return false to cancel.
34170         * @param {Node} node The node
34171         * @param {Boolean} deep
34172         * @param {Boolean} anim
34173         */
34174         "beforeexpand" : true,
34175         /**
34176         * @event beforecollapse
34177         * Fires before a node is collapsed, return false to cancel.
34178         * @param {Node} node The node
34179         * @param {Boolean} deep
34180         * @param {Boolean} anim
34181         */
34182         "beforecollapse" : true,
34183         /**
34184         * @event expand
34185         * Fires when a node is expanded
34186         * @param {Node} node The node
34187         */
34188         "expand" : true,
34189         /**
34190         * @event disabledchange
34191         * Fires when the disabled status of a node changes
34192         * @param {Node} node The node
34193         * @param {Boolean} disabled
34194         */
34195         "disabledchange" : true,
34196         /**
34197         * @event collapse
34198         * Fires when a node is collapsed
34199         * @param {Node} node The node
34200         */
34201         "collapse" : true,
34202         /**
34203         * @event beforeclick
34204         * Fires before click processing on a node. Return false to cancel the default action.
34205         * @param {Node} node The node
34206         * @param {Roo.EventObject} e The event object
34207         */
34208         "beforeclick":true,
34209         /**
34210         * @event checkchange
34211         * Fires when a node with a checkbox's checked property changes
34212         * @param {Node} this This node
34213         * @param {Boolean} checked
34214         */
34215         "checkchange":true,
34216         /**
34217         * @event click
34218         * Fires when a node is clicked
34219         * @param {Node} node The node
34220         * @param {Roo.EventObject} e The event object
34221         */
34222         "click":true,
34223         /**
34224         * @event dblclick
34225         * Fires when a node is double clicked
34226         * @param {Node} node The node
34227         * @param {Roo.EventObject} e The event object
34228         */
34229         "dblclick":true,
34230         /**
34231         * @event contextmenu
34232         * Fires when a node is right clicked
34233         * @param {Node} node The node
34234         * @param {Roo.EventObject} e The event object
34235         */
34236         "contextmenu":true,
34237         /**
34238         * @event beforechildrenrendered
34239         * Fires right before the child nodes for a node are rendered
34240         * @param {Node} node The node
34241         */
34242         "beforechildrenrendered":true,
34243         /**
34244         * @event startdrag
34245         * Fires when a node starts being dragged
34246         * @param {Roo.tree.TreePanel} this
34247         * @param {Roo.tree.TreeNode} node
34248         * @param {event} e The raw browser event
34249         */ 
34250        "startdrag" : true,
34251        /**
34252         * @event enddrag
34253         * Fires when a drag operation is complete
34254         * @param {Roo.tree.TreePanel} this
34255         * @param {Roo.tree.TreeNode} node
34256         * @param {event} e The raw browser event
34257         */
34258        "enddrag" : true,
34259        /**
34260         * @event dragdrop
34261         * Fires when a dragged node is dropped on a valid DD target
34262         * @param {Roo.tree.TreePanel} this
34263         * @param {Roo.tree.TreeNode} node
34264         * @param {DD} dd The dd it was dropped on
34265         * @param {event} e The raw browser event
34266         */
34267        "dragdrop" : true,
34268        /**
34269         * @event beforenodedrop
34270         * Fires when a DD object is dropped on a node in this tree for preprocessing. Return false to cancel the drop. The dropEvent
34271         * passed to handlers has the following properties:<br />
34272         * <ul style="padding:5px;padding-left:16px;">
34273         * <li>tree - The TreePanel</li>
34274         * <li>target - The node being targeted for the drop</li>
34275         * <li>data - The drag data from the drag source</li>
34276         * <li>point - The point of the drop - append, above or below</li>
34277         * <li>source - The drag source</li>
34278         * <li>rawEvent - Raw mouse event</li>
34279         * <li>dropNode - Drop node(s) provided by the source <b>OR</b> you can supply node(s)
34280         * to be inserted by setting them on this object.</li>
34281         * <li>cancel - Set this to true to cancel the drop.</li>
34282         * </ul>
34283         * @param {Object} dropEvent
34284         */
34285        "beforenodedrop" : true,
34286        /**
34287         * @event nodedrop
34288         * Fires after a DD object is dropped on a node in this tree. The dropEvent
34289         * passed to handlers has the following properties:<br />
34290         * <ul style="padding:5px;padding-left:16px;">
34291         * <li>tree - The TreePanel</li>
34292         * <li>target - The node being targeted for the drop</li>
34293         * <li>data - The drag data from the drag source</li>
34294         * <li>point - The point of the drop - append, above or below</li>
34295         * <li>source - The drag source</li>
34296         * <li>rawEvent - Raw mouse event</li>
34297         * <li>dropNode - Dropped node(s).</li>
34298         * </ul>
34299         * @param {Object} dropEvent
34300         */
34301        "nodedrop" : true,
34302         /**
34303         * @event nodedragover
34304         * Fires when a tree node is being targeted for a drag drop, return false to signal drop not allowed. The dragOverEvent
34305         * passed to handlers has the following properties:<br />
34306         * <ul style="padding:5px;padding-left:16px;">
34307         * <li>tree - The TreePanel</li>
34308         * <li>target - The node being targeted for the drop</li>
34309         * <li>data - The drag data from the drag source</li>
34310         * <li>point - The point of the drop - append, above or below</li>
34311         * <li>source - The drag source</li>
34312         * <li>rawEvent - Raw mouse event</li>
34313         * <li>dropNode - Drop node(s) provided by the source.</li>
34314         * <li>cancel - Set this to true to signal drop not allowed.</li>
34315         * </ul>
34316         * @param {Object} dragOverEvent
34317         */
34318        "nodedragover" : true,
34319        /**
34320         * @event appendnode
34321         * Fires when append node to the tree
34322         * @param {Roo.tree.TreePanel} this
34323         * @param {Roo.tree.TreeNode} node
34324         * @param {Number} index The index of the newly appended node
34325         */
34326        "appendnode" : true
34327         
34328     });
34329     if(this.singleExpand){
34330        this.on("beforeexpand", this.restrictExpand, this);
34331     }
34332     if (this.editor) {
34333         this.editor.tree = this;
34334         this.editor = Roo.factory(this.editor, Roo.tree);
34335     }
34336     
34337     if (this.selModel) {
34338         this.selModel = Roo.factory(this.selModel, Roo.tree);
34339     }
34340    
34341 };
34342 Roo.extend(Roo.tree.TreePanel, Roo.data.Tree, {
34343     rootVisible : true,
34344     animate: Roo.enableFx,
34345     lines : true,
34346     enableDD : false,
34347     hlDrop : Roo.enableFx,
34348   
34349     renderer: false,
34350     
34351     rendererTip: false,
34352     // private
34353     restrictExpand : function(node){
34354         var p = node.parentNode;
34355         if(p){
34356             if(p.expandedChild && p.expandedChild.parentNode == p){
34357                 p.expandedChild.collapse();
34358             }
34359             p.expandedChild = node;
34360         }
34361     },
34362
34363     // private override
34364     setRootNode : function(node){
34365         Roo.tree.TreePanel.superclass.setRootNode.call(this, node);
34366         if(!this.rootVisible){
34367             node.ui = new Roo.tree.RootTreeNodeUI(node);
34368         }
34369         return node;
34370     },
34371
34372     /**
34373      * Returns the container element for this TreePanel
34374      */
34375     getEl : function(){
34376         return this.el;
34377     },
34378
34379     /**
34380      * Returns the default TreeLoader for this TreePanel
34381      */
34382     getLoader : function(){
34383         return this.loader;
34384     },
34385
34386     /**
34387      * Expand all nodes
34388      */
34389     expandAll : function(){
34390         this.root.expand(true);
34391     },
34392
34393     /**
34394      * Collapse all nodes
34395      */
34396     collapseAll : function(){
34397         this.root.collapse(true);
34398     },
34399
34400     /**
34401      * Returns the selection model used by this TreePanel
34402      */
34403     getSelectionModel : function(){
34404         if(!this.selModel){
34405             this.selModel = new Roo.tree.DefaultSelectionModel();
34406         }
34407         return this.selModel;
34408     },
34409
34410     /**
34411      * Retrieve an array of checked nodes, or an array of a specific attribute of checked nodes (e.g. "id")
34412      * @param {String} attribute (optional) Defaults to null (return the actual nodes)
34413      * @param {TreeNode} startNode (optional) The node to start from, defaults to the root
34414      * @return {Array}
34415      */
34416     getChecked : function(a, startNode){
34417         startNode = startNode || this.root;
34418         var r = [];
34419         var f = function(){
34420             if(this.attributes.checked){
34421                 r.push(!a ? this : (a == 'id' ? this.id : this.attributes[a]));
34422             }
34423         }
34424         startNode.cascade(f);
34425         return r;
34426     },
34427
34428     /**
34429      * Expands a specified path in this TreePanel. A path can be retrieved from a node with {@link Roo.data.Node#getPath}
34430      * @param {String} path
34431      * @param {String} attr (optional) The attribute used in the path (see {@link Roo.data.Node#getPath} for more info)
34432      * @param {Function} callback (optional) The callback to call when the expand is complete. The callback will be called with
34433      * (bSuccess, oLastNode) where bSuccess is if the expand was successful and oLastNode is the last node that was expanded.
34434      */
34435     expandPath : function(path, attr, callback){
34436         attr = attr || "id";
34437         var keys = path.split(this.pathSeparator);
34438         var curNode = this.root;
34439         if(curNode.attributes[attr] != keys[1]){ // invalid root
34440             if(callback){
34441                 callback(false, null);
34442             }
34443             return;
34444         }
34445         var index = 1;
34446         var f = function(){
34447             if(++index == keys.length){
34448                 if(callback){
34449                     callback(true, curNode);
34450                 }
34451                 return;
34452             }
34453             var c = curNode.findChild(attr, keys[index]);
34454             if(!c){
34455                 if(callback){
34456                     callback(false, curNode);
34457                 }
34458                 return;
34459             }
34460             curNode = c;
34461             c.expand(false, false, f);
34462         };
34463         curNode.expand(false, false, f);
34464     },
34465
34466     /**
34467      * Selects the node in this tree at the specified path. A path can be retrieved from a node with {@link Roo.data.Node#getPath}
34468      * @param {String} path
34469      * @param {String} attr (optional) The attribute used in the path (see {@link Roo.data.Node#getPath} for more info)
34470      * @param {Function} callback (optional) The callback to call when the selection is complete. The callback will be called with
34471      * (bSuccess, oSelNode) where bSuccess is if the selection was successful and oSelNode is the selected node.
34472      */
34473     selectPath : function(path, attr, callback){
34474         attr = attr || "id";
34475         var keys = path.split(this.pathSeparator);
34476         var v = keys.pop();
34477         if(keys.length > 0){
34478             var f = function(success, node){
34479                 if(success && node){
34480                     var n = node.findChild(attr, v);
34481                     if(n){
34482                         n.select();
34483                         if(callback){
34484                             callback(true, n);
34485                         }
34486                     }else if(callback){
34487                         callback(false, n);
34488                     }
34489                 }else{
34490                     if(callback){
34491                         callback(false, n);
34492                     }
34493                 }
34494             };
34495             this.expandPath(keys.join(this.pathSeparator), attr, f);
34496         }else{
34497             this.root.select();
34498             if(callback){
34499                 callback(true, this.root);
34500             }
34501         }
34502     },
34503
34504     getTreeEl : function(){
34505         return this.el;
34506     },
34507
34508     /**
34509      * Trigger rendering of this TreePanel
34510      */
34511     render : function(){
34512         if (this.innerCt) {
34513             return this; // stop it rendering more than once!!
34514         }
34515         
34516         this.innerCt = this.el.createChild({tag:"ul",
34517                cls:"x-tree-root-ct " +
34518                (this.lines ? "x-tree-lines" : "x-tree-no-lines")});
34519
34520         if(this.containerScroll){
34521             Roo.dd.ScrollManager.register(this.el);
34522         }
34523         if((this.enableDD || this.enableDrop) && !this.dropZone){
34524            /**
34525             * The dropZone used by this tree if drop is enabled
34526             * @type Roo.tree.TreeDropZone
34527             */
34528              this.dropZone = new Roo.tree.TreeDropZone(this, this.dropConfig || {
34529                ddGroup: this.ddGroup || "TreeDD", appendOnly: this.ddAppendOnly === true
34530            });
34531         }
34532         if((this.enableDD || this.enableDrag) && !this.dragZone){
34533            /**
34534             * The dragZone used by this tree if drag is enabled
34535             * @type Roo.tree.TreeDragZone
34536             */
34537             this.dragZone = new Roo.tree.TreeDragZone(this, this.dragConfig || {
34538                ddGroup: this.ddGroup || "TreeDD",
34539                scroll: this.ddScroll
34540            });
34541         }
34542         this.getSelectionModel().init(this);
34543         if (!this.root) {
34544             Roo.log("ROOT not set in tree");
34545             return this;
34546         }
34547         this.root.render();
34548         if(!this.rootVisible){
34549             this.root.renderChildren();
34550         }
34551         return this;
34552     }
34553 });/*
34554  * Based on:
34555  * Ext JS Library 1.1.1
34556  * Copyright(c) 2006-2007, Ext JS, LLC.
34557  *
34558  * Originally Released Under LGPL - original licence link has changed is not relivant.
34559  *
34560  * Fork - LGPL
34561  * <script type="text/javascript">
34562  */
34563  
34564
34565 /**
34566  * @class Roo.tree.DefaultSelectionModel
34567  * @extends Roo.util.Observable
34568  * The default single selection for a TreePanel.
34569  * @param {Object} cfg Configuration
34570  */
34571 Roo.tree.DefaultSelectionModel = function(cfg){
34572    this.selNode = null;
34573    
34574    
34575    
34576    this.addEvents({
34577        /**
34578         * @event selectionchange
34579         * Fires when the selected node changes
34580         * @param {DefaultSelectionModel} this
34581         * @param {TreeNode} node the new selection
34582         */
34583        "selectionchange" : true,
34584
34585        /**
34586         * @event beforeselect
34587         * Fires before the selected node changes, return false to cancel the change
34588         * @param {DefaultSelectionModel} this
34589         * @param {TreeNode} node the new selection
34590         * @param {TreeNode} node the old selection
34591         */
34592        "beforeselect" : true
34593    });
34594    
34595     Roo.tree.DefaultSelectionModel.superclass.constructor.call(this,cfg);
34596 };
34597
34598 Roo.extend(Roo.tree.DefaultSelectionModel, Roo.util.Observable, {
34599     init : function(tree){
34600         this.tree = tree;
34601         tree.getTreeEl().on("keydown", this.onKeyDown, this);
34602         tree.on("click", this.onNodeClick, this);
34603     },
34604     
34605     onNodeClick : function(node, e){
34606         if (e.ctrlKey && this.selNode == node)  {
34607             this.unselect(node);
34608             return;
34609         }
34610         this.select(node);
34611     },
34612     
34613     /**
34614      * Select a node.
34615      * @param {TreeNode} node The node to select
34616      * @return {TreeNode} The selected node
34617      */
34618     select : function(node){
34619         var last = this.selNode;
34620         if(last != node && this.fireEvent('beforeselect', this, node, last) !== false){
34621             if(last){
34622                 last.ui.onSelectedChange(false);
34623             }
34624             this.selNode = node;
34625             node.ui.onSelectedChange(true);
34626             this.fireEvent("selectionchange", this, node, last);
34627         }
34628         return node;
34629     },
34630     
34631     /**
34632      * Deselect a node.
34633      * @param {TreeNode} node The node to unselect
34634      */
34635     unselect : function(node){
34636         if(this.selNode == node){
34637             this.clearSelections();
34638         }    
34639     },
34640     
34641     /**
34642      * Clear all selections
34643      */
34644     clearSelections : function(){
34645         var n = this.selNode;
34646         if(n){
34647             n.ui.onSelectedChange(false);
34648             this.selNode = null;
34649             this.fireEvent("selectionchange", this, null);
34650         }
34651         return n;
34652     },
34653     
34654     /**
34655      * Get the selected node
34656      * @return {TreeNode} The selected node
34657      */
34658     getSelectedNode : function(){
34659         return this.selNode;    
34660     },
34661     
34662     /**
34663      * Returns true if the node is selected
34664      * @param {TreeNode} node The node to check
34665      * @return {Boolean}
34666      */
34667     isSelected : function(node){
34668         return this.selNode == node;  
34669     },
34670
34671     /**
34672      * Selects the node above the selected node in the tree, intelligently walking the nodes
34673      * @return TreeNode The new selection
34674      */
34675     selectPrevious : function(){
34676         var s = this.selNode || this.lastSelNode;
34677         if(!s){
34678             return null;
34679         }
34680         var ps = s.previousSibling;
34681         if(ps){
34682             if(!ps.isExpanded() || ps.childNodes.length < 1){
34683                 return this.select(ps);
34684             } else{
34685                 var lc = ps.lastChild;
34686                 while(lc && lc.isExpanded() && lc.childNodes.length > 0){
34687                     lc = lc.lastChild;
34688                 }
34689                 return this.select(lc);
34690             }
34691         } else if(s.parentNode && (this.tree.rootVisible || !s.parentNode.isRoot)){
34692             return this.select(s.parentNode);
34693         }
34694         return null;
34695     },
34696
34697     /**
34698      * Selects the node above the selected node in the tree, intelligently walking the nodes
34699      * @return TreeNode The new selection
34700      */
34701     selectNext : function(){
34702         var s = this.selNode || this.lastSelNode;
34703         if(!s){
34704             return null;
34705         }
34706         if(s.firstChild && s.isExpanded()){
34707              return this.select(s.firstChild);
34708          }else if(s.nextSibling){
34709              return this.select(s.nextSibling);
34710          }else if(s.parentNode){
34711             var newS = null;
34712             s.parentNode.bubble(function(){
34713                 if(this.nextSibling){
34714                     newS = this.getOwnerTree().selModel.select(this.nextSibling);
34715                     return false;
34716                 }
34717             });
34718             return newS;
34719          }
34720         return null;
34721     },
34722
34723     onKeyDown : function(e){
34724         var s = this.selNode || this.lastSelNode;
34725         // undesirable, but required
34726         var sm = this;
34727         if(!s){
34728             return;
34729         }
34730         var k = e.getKey();
34731         switch(k){
34732              case e.DOWN:
34733                  e.stopEvent();
34734                  this.selectNext();
34735              break;
34736              case e.UP:
34737                  e.stopEvent();
34738                  this.selectPrevious();
34739              break;
34740              case e.RIGHT:
34741                  e.preventDefault();
34742                  if(s.hasChildNodes()){
34743                      if(!s.isExpanded()){
34744                          s.expand();
34745                      }else if(s.firstChild){
34746                          this.select(s.firstChild, e);
34747                      }
34748                  }
34749              break;
34750              case e.LEFT:
34751                  e.preventDefault();
34752                  if(s.hasChildNodes() && s.isExpanded()){
34753                      s.collapse();
34754                  }else if(s.parentNode && (this.tree.rootVisible || s.parentNode != this.tree.getRootNode())){
34755                      this.select(s.parentNode, e);
34756                  }
34757              break;
34758         };
34759     }
34760 });
34761
34762 /**
34763  * @class Roo.tree.MultiSelectionModel
34764  * @extends Roo.util.Observable
34765  * Multi selection for a TreePanel.
34766  * @param {Object} cfg Configuration
34767  */
34768 Roo.tree.MultiSelectionModel = function(){
34769    this.selNodes = [];
34770    this.selMap = {};
34771    this.addEvents({
34772        /**
34773         * @event selectionchange
34774         * Fires when the selected nodes change
34775         * @param {MultiSelectionModel} this
34776         * @param {Array} nodes Array of the selected nodes
34777         */
34778        "selectionchange" : true
34779    });
34780    Roo.tree.MultiSelectionModel.superclass.constructor.call(this,cfg);
34781    
34782 };
34783
34784 Roo.extend(Roo.tree.MultiSelectionModel, Roo.util.Observable, {
34785     init : function(tree){
34786         this.tree = tree;
34787         tree.getTreeEl().on("keydown", this.onKeyDown, this);
34788         tree.on("click", this.onNodeClick, this);
34789     },
34790     
34791     onNodeClick : function(node, e){
34792         this.select(node, e, e.ctrlKey);
34793     },
34794     
34795     /**
34796      * Select a node.
34797      * @param {TreeNode} node The node to select
34798      * @param {EventObject} e (optional) An event associated with the selection
34799      * @param {Boolean} keepExisting True to retain existing selections
34800      * @return {TreeNode} The selected node
34801      */
34802     select : function(node, e, keepExisting){
34803         if(keepExisting !== true){
34804             this.clearSelections(true);
34805         }
34806         if(this.isSelected(node)){
34807             this.lastSelNode = node;
34808             return node;
34809         }
34810         this.selNodes.push(node);
34811         this.selMap[node.id] = node;
34812         this.lastSelNode = node;
34813         node.ui.onSelectedChange(true);
34814         this.fireEvent("selectionchange", this, this.selNodes);
34815         return node;
34816     },
34817     
34818     /**
34819      * Deselect a node.
34820      * @param {TreeNode} node The node to unselect
34821      */
34822     unselect : function(node){
34823         if(this.selMap[node.id]){
34824             node.ui.onSelectedChange(false);
34825             var sn = this.selNodes;
34826             var index = -1;
34827             if(sn.indexOf){
34828                 index = sn.indexOf(node);
34829             }else{
34830                 for(var i = 0, len = sn.length; i < len; i++){
34831                     if(sn[i] == node){
34832                         index = i;
34833                         break;
34834                     }
34835                 }
34836             }
34837             if(index != -1){
34838                 this.selNodes.splice(index, 1);
34839             }
34840             delete this.selMap[node.id];
34841             this.fireEvent("selectionchange", this, this.selNodes);
34842         }
34843     },
34844     
34845     /**
34846      * Clear all selections
34847      */
34848     clearSelections : function(suppressEvent){
34849         var sn = this.selNodes;
34850         if(sn.length > 0){
34851             for(var i = 0, len = sn.length; i < len; i++){
34852                 sn[i].ui.onSelectedChange(false);
34853             }
34854             this.selNodes = [];
34855             this.selMap = {};
34856             if(suppressEvent !== true){
34857                 this.fireEvent("selectionchange", this, this.selNodes);
34858             }
34859         }
34860     },
34861     
34862     /**
34863      * Returns true if the node is selected
34864      * @param {TreeNode} node The node to check
34865      * @return {Boolean}
34866      */
34867     isSelected : function(node){
34868         return this.selMap[node.id] ? true : false;  
34869     },
34870     
34871     /**
34872      * Returns an array of the selected nodes
34873      * @return {Array}
34874      */
34875     getSelectedNodes : function(){
34876         return this.selNodes;    
34877     },
34878
34879     onKeyDown : Roo.tree.DefaultSelectionModel.prototype.onKeyDown,
34880
34881     selectNext : Roo.tree.DefaultSelectionModel.prototype.selectNext,
34882
34883     selectPrevious : Roo.tree.DefaultSelectionModel.prototype.selectPrevious
34884 });/*
34885  * Based on:
34886  * Ext JS Library 1.1.1
34887  * Copyright(c) 2006-2007, Ext JS, LLC.
34888  *
34889  * Originally Released Under LGPL - original licence link has changed is not relivant.
34890  *
34891  * Fork - LGPL
34892  * <script type="text/javascript">
34893  */
34894  
34895 /**
34896  * @class Roo.tree.TreeNode
34897  * @extends Roo.data.Node
34898  * @cfg {String} text The text for this node
34899  * @cfg {Boolean} expanded true to start the node expanded
34900  * @cfg {Boolean} allowDrag false to make this node undraggable if DD is on (defaults to true)
34901  * @cfg {Boolean} allowDrop false if this node cannot be drop on
34902  * @cfg {Boolean} disabled true to start the node disabled
34903  * @cfg {String} icon The path to an icon for the node. The preferred way to do this
34904  *    is to use the cls or iconCls attributes and add the icon via a CSS background image.
34905  * @cfg {String} cls A css class to be added to the node
34906  * @cfg {String} iconCls A css class to be added to the nodes icon element for applying css background images
34907  * @cfg {String} href URL of the link used for the node (defaults to #)
34908  * @cfg {String} hrefTarget target frame for the link
34909  * @cfg {String} qtip An Ext QuickTip for the node
34910  * @cfg {String} qtipCfg An Ext QuickTip config for the node (used instead of qtip)
34911  * @cfg {Boolean} singleClickExpand True for single click expand on this node
34912  * @cfg {Function} uiProvider A UI <b>class</b> to use for this node (defaults to Roo.tree.TreeNodeUI)
34913  * @cfg {Boolean} checked True to render a checked checkbox for this node, false to render an unchecked checkbox
34914  * (defaults to undefined with no checkbox rendered)
34915  * @constructor
34916  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node
34917  */
34918 Roo.tree.TreeNode = function(attributes){
34919     attributes = attributes || {};
34920     if(typeof attributes == "string"){
34921         attributes = {text: attributes};
34922     }
34923     this.childrenRendered = false;
34924     this.rendered = false;
34925     Roo.tree.TreeNode.superclass.constructor.call(this, attributes);
34926     this.expanded = attributes.expanded === true;
34927     this.isTarget = attributes.isTarget !== false;
34928     this.draggable = attributes.draggable !== false && attributes.allowDrag !== false;
34929     this.allowChildren = attributes.allowChildren !== false && attributes.allowDrop !== false;
34930
34931     /**
34932      * Read-only. The text for this node. To change it use setText().
34933      * @type String
34934      */
34935     this.text = attributes.text;
34936     /**
34937      * True if this node is disabled.
34938      * @type Boolean
34939      */
34940     this.disabled = attributes.disabled === true;
34941
34942     this.addEvents({
34943         /**
34944         * @event textchange
34945         * Fires when the text for this node is changed
34946         * @param {Node} this This node
34947         * @param {String} text The new text
34948         * @param {String} oldText The old text
34949         */
34950         "textchange" : true,
34951         /**
34952         * @event beforeexpand
34953         * Fires before this node is expanded, return false to cancel.
34954         * @param {Node} this This node
34955         * @param {Boolean} deep
34956         * @param {Boolean} anim
34957         */
34958         "beforeexpand" : true,
34959         /**
34960         * @event beforecollapse
34961         * Fires before this node is collapsed, return false to cancel.
34962         * @param {Node} this This node
34963         * @param {Boolean} deep
34964         * @param {Boolean} anim
34965         */
34966         "beforecollapse" : true,
34967         /**
34968         * @event expand
34969         * Fires when this node is expanded
34970         * @param {Node} this This node
34971         */
34972         "expand" : true,
34973         /**
34974         * @event disabledchange
34975         * Fires when the disabled status of this node changes
34976         * @param {Node} this This node
34977         * @param {Boolean} disabled
34978         */
34979         "disabledchange" : true,
34980         /**
34981         * @event collapse
34982         * Fires when this node is collapsed
34983         * @param {Node} this This node
34984         */
34985         "collapse" : true,
34986         /**
34987         * @event beforeclick
34988         * Fires before click processing. Return false to cancel the default action.
34989         * @param {Node} this This node
34990         * @param {Roo.EventObject} e The event object
34991         */
34992         "beforeclick":true,
34993         /**
34994         * @event checkchange
34995         * Fires when a node with a checkbox's checked property changes
34996         * @param {Node} this This node
34997         * @param {Boolean} checked
34998         */
34999         "checkchange":true,
35000         /**
35001         * @event click
35002         * Fires when this node is clicked
35003         * @param {Node} this This node
35004         * @param {Roo.EventObject} e The event object
35005         */
35006         "click":true,
35007         /**
35008         * @event dblclick
35009         * Fires when this node is double clicked
35010         * @param {Node} this This node
35011         * @param {Roo.EventObject} e The event object
35012         */
35013         "dblclick":true,
35014         /**
35015         * @event contextmenu
35016         * Fires when this node is right clicked
35017         * @param {Node} this This node
35018         * @param {Roo.EventObject} e The event object
35019         */
35020         "contextmenu":true,
35021         /**
35022         * @event beforechildrenrendered
35023         * Fires right before the child nodes for this node are rendered
35024         * @param {Node} this This node
35025         */
35026         "beforechildrenrendered":true
35027     });
35028
35029     var uiClass = this.attributes.uiProvider || Roo.tree.TreeNodeUI;
35030
35031     /**
35032      * Read-only. The UI for this node
35033      * @type TreeNodeUI
35034      */
35035     this.ui = new uiClass(this);
35036     
35037     // finally support items[]
35038     if (typeof(this.attributes.items) == 'undefined' || !this.attributes.items) {
35039         return;
35040     }
35041     
35042     
35043     Roo.each(this.attributes.items, function(c) {
35044         this.appendChild(Roo.factory(c,Roo.Tree));
35045     }, this);
35046     delete this.attributes.items;
35047     
35048     
35049     
35050 };
35051 Roo.extend(Roo.tree.TreeNode, Roo.data.Node, {
35052     preventHScroll: true,
35053     /**
35054      * Returns true if this node is expanded
35055      * @return {Boolean}
35056      */
35057     isExpanded : function(){
35058         return this.expanded;
35059     },
35060
35061     /**
35062      * Returns the UI object for this node
35063      * @return {TreeNodeUI}
35064      */
35065     getUI : function(){
35066         return this.ui;
35067     },
35068
35069     // private override
35070     setFirstChild : function(node){
35071         var of = this.firstChild;
35072         Roo.tree.TreeNode.superclass.setFirstChild.call(this, node);
35073         if(this.childrenRendered && of && node != of){
35074             of.renderIndent(true, true);
35075         }
35076         if(this.rendered){
35077             this.renderIndent(true, true);
35078         }
35079     },
35080
35081     // private override
35082     setLastChild : function(node){
35083         var ol = this.lastChild;
35084         Roo.tree.TreeNode.superclass.setLastChild.call(this, node);
35085         if(this.childrenRendered && ol && node != ol){
35086             ol.renderIndent(true, true);
35087         }
35088         if(this.rendered){
35089             this.renderIndent(true, true);
35090         }
35091     },
35092
35093     // these methods are overridden to provide lazy rendering support
35094     // private override
35095     appendChild : function()
35096     {
35097         var node = Roo.tree.TreeNode.superclass.appendChild.apply(this, arguments);
35098         if(node && this.childrenRendered){
35099             node.render();
35100         }
35101         this.ui.updateExpandIcon();
35102         return node;
35103     },
35104
35105     // private override
35106     removeChild : function(node){
35107         this.ownerTree.getSelectionModel().unselect(node);
35108         Roo.tree.TreeNode.superclass.removeChild.apply(this, arguments);
35109         // if it's been rendered remove dom node
35110         if(this.childrenRendered){
35111             node.ui.remove();
35112         }
35113         if(this.childNodes.length < 1){
35114             this.collapse(false, false);
35115         }else{
35116             this.ui.updateExpandIcon();
35117         }
35118         if(!this.firstChild) {
35119             this.childrenRendered = false;
35120         }
35121         return node;
35122     },
35123
35124     // private override
35125     insertBefore : function(node, refNode){
35126         var newNode = Roo.tree.TreeNode.superclass.insertBefore.apply(this, arguments);
35127         if(newNode && refNode && this.childrenRendered){
35128             node.render();
35129         }
35130         this.ui.updateExpandIcon();
35131         return newNode;
35132     },
35133
35134     /**
35135      * Sets the text for this node
35136      * @param {String} text
35137      */
35138     setText : function(text){
35139         var oldText = this.text;
35140         this.text = text;
35141         this.attributes.text = text;
35142         if(this.rendered){ // event without subscribing
35143             this.ui.onTextChange(this, text, oldText);
35144         }
35145         this.fireEvent("textchange", this, text, oldText);
35146     },
35147
35148     /**
35149      * Triggers selection of this node
35150      */
35151     select : function(){
35152         this.getOwnerTree().getSelectionModel().select(this);
35153     },
35154
35155     /**
35156      * Triggers deselection of this node
35157      */
35158     unselect : function(){
35159         this.getOwnerTree().getSelectionModel().unselect(this);
35160     },
35161
35162     /**
35163      * Returns true if this node is selected
35164      * @return {Boolean}
35165      */
35166     isSelected : function(){
35167         return this.getOwnerTree().getSelectionModel().isSelected(this);
35168     },
35169
35170     /**
35171      * Expand this node.
35172      * @param {Boolean} deep (optional) True to expand all children as well
35173      * @param {Boolean} anim (optional) false to cancel the default animation
35174      * @param {Function} callback (optional) A callback to be called when
35175      * expanding this node completes (does not wait for deep expand to complete).
35176      * Called with 1 parameter, this node.
35177      */
35178     expand : function(deep, anim, callback){
35179         if(!this.expanded){
35180             if(this.fireEvent("beforeexpand", this, deep, anim) === false){
35181                 return;
35182             }
35183             if(!this.childrenRendered){
35184                 this.renderChildren();
35185             }
35186             this.expanded = true;
35187             
35188             if(!this.isHiddenRoot() && (this.getOwnerTree() && this.getOwnerTree().animate && anim !== false) || anim){
35189                 this.ui.animExpand(function(){
35190                     this.fireEvent("expand", this);
35191                     if(typeof callback == "function"){
35192                         callback(this);
35193                     }
35194                     if(deep === true){
35195                         this.expandChildNodes(true);
35196                     }
35197                 }.createDelegate(this));
35198                 return;
35199             }else{
35200                 this.ui.expand();
35201                 this.fireEvent("expand", this);
35202                 if(typeof callback == "function"){
35203                     callback(this);
35204                 }
35205             }
35206         }else{
35207            if(typeof callback == "function"){
35208                callback(this);
35209            }
35210         }
35211         if(deep === true){
35212             this.expandChildNodes(true);
35213         }
35214     },
35215
35216     isHiddenRoot : function(){
35217         return this.isRoot && !this.getOwnerTree().rootVisible;
35218     },
35219
35220     /**
35221      * Collapse this node.
35222      * @param {Boolean} deep (optional) True to collapse all children as well
35223      * @param {Boolean} anim (optional) false to cancel the default animation
35224      */
35225     collapse : function(deep, anim){
35226         if(this.expanded && !this.isHiddenRoot()){
35227             if(this.fireEvent("beforecollapse", this, deep, anim) === false){
35228                 return;
35229             }
35230             this.expanded = false;
35231             if((this.getOwnerTree().animate && anim !== false) || anim){
35232                 this.ui.animCollapse(function(){
35233                     this.fireEvent("collapse", this);
35234                     if(deep === true){
35235                         this.collapseChildNodes(true);
35236                     }
35237                 }.createDelegate(this));
35238                 return;
35239             }else{
35240                 this.ui.collapse();
35241                 this.fireEvent("collapse", this);
35242             }
35243         }
35244         if(deep === true){
35245             var cs = this.childNodes;
35246             for(var i = 0, len = cs.length; i < len; i++) {
35247                 cs[i].collapse(true, false);
35248             }
35249         }
35250     },
35251
35252     // private
35253     delayedExpand : function(delay){
35254         if(!this.expandProcId){
35255             this.expandProcId = this.expand.defer(delay, this);
35256         }
35257     },
35258
35259     // private
35260     cancelExpand : function(){
35261         if(this.expandProcId){
35262             clearTimeout(this.expandProcId);
35263         }
35264         this.expandProcId = false;
35265     },
35266
35267     /**
35268      * Toggles expanded/collapsed state of the node
35269      */
35270     toggle : function(){
35271         if(this.expanded){
35272             this.collapse();
35273         }else{
35274             this.expand();
35275         }
35276     },
35277
35278     /**
35279      * Ensures all parent nodes are expanded
35280      */
35281     ensureVisible : function(callback){
35282         var tree = this.getOwnerTree();
35283         tree.expandPath(this.parentNode.getPath(), false, function(){
35284             tree.getTreeEl().scrollChildIntoView(this.ui.anchor);
35285             Roo.callback(callback);
35286         }.createDelegate(this));
35287     },
35288
35289     /**
35290      * Expand all child nodes
35291      * @param {Boolean} deep (optional) true if the child nodes should also expand their child nodes
35292      */
35293     expandChildNodes : function(deep){
35294         var cs = this.childNodes;
35295         for(var i = 0, len = cs.length; i < len; i++) {
35296                 cs[i].expand(deep);
35297         }
35298     },
35299
35300     /**
35301      * Collapse all child nodes
35302      * @param {Boolean} deep (optional) true if the child nodes should also collapse their child nodes
35303      */
35304     collapseChildNodes : function(deep){
35305         var cs = this.childNodes;
35306         for(var i = 0, len = cs.length; i < len; i++) {
35307                 cs[i].collapse(deep);
35308         }
35309     },
35310
35311     /**
35312      * Disables this node
35313      */
35314     disable : function(){
35315         this.disabled = true;
35316         this.unselect();
35317         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
35318             this.ui.onDisableChange(this, true);
35319         }
35320         this.fireEvent("disabledchange", this, true);
35321     },
35322
35323     /**
35324      * Enables this node
35325      */
35326     enable : function(){
35327         this.disabled = false;
35328         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
35329             this.ui.onDisableChange(this, false);
35330         }
35331         this.fireEvent("disabledchange", this, false);
35332     },
35333
35334     // private
35335     renderChildren : function(suppressEvent){
35336         if(suppressEvent !== false){
35337             this.fireEvent("beforechildrenrendered", this);
35338         }
35339         var cs = this.childNodes;
35340         for(var i = 0, len = cs.length; i < len; i++){
35341             cs[i].render(true);
35342         }
35343         this.childrenRendered = true;
35344     },
35345
35346     // private
35347     sort : function(fn, scope){
35348         Roo.tree.TreeNode.superclass.sort.apply(this, arguments);
35349         if(this.childrenRendered){
35350             var cs = this.childNodes;
35351             for(var i = 0, len = cs.length; i < len; i++){
35352                 cs[i].render(true);
35353             }
35354         }
35355     },
35356
35357     // private
35358     render : function(bulkRender){
35359         this.ui.render(bulkRender);
35360         if(!this.rendered){
35361             this.rendered = true;
35362             if(this.expanded){
35363                 this.expanded = false;
35364                 this.expand(false, false);
35365             }
35366         }
35367     },
35368
35369     // private
35370     renderIndent : function(deep, refresh){
35371         if(refresh){
35372             this.ui.childIndent = null;
35373         }
35374         this.ui.renderIndent();
35375         if(deep === true && this.childrenRendered){
35376             var cs = this.childNodes;
35377             for(var i = 0, len = cs.length; i < len; i++){
35378                 cs[i].renderIndent(true, refresh);
35379             }
35380         }
35381     }
35382 });/*
35383  * Based on:
35384  * Ext JS Library 1.1.1
35385  * Copyright(c) 2006-2007, Ext JS, LLC.
35386  *
35387  * Originally Released Under LGPL - original licence link has changed is not relivant.
35388  *
35389  * Fork - LGPL
35390  * <script type="text/javascript">
35391  */
35392  
35393 /**
35394  * @class Roo.tree.AsyncTreeNode
35395  * @extends Roo.tree.TreeNode
35396  * @cfg {TreeLoader} loader A TreeLoader to be used by this node (defaults to the loader defined on the tree)
35397  * @constructor
35398  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node 
35399  */
35400  Roo.tree.AsyncTreeNode = function(config){
35401     this.loaded = false;
35402     this.loading = false;
35403     Roo.tree.AsyncTreeNode.superclass.constructor.apply(this, arguments);
35404     /**
35405     * @event beforeload
35406     * Fires before this node is loaded, return false to cancel
35407     * @param {Node} this This node
35408     */
35409     this.addEvents({'beforeload':true, 'load': true});
35410     /**
35411     * @event load
35412     * Fires when this node is loaded
35413     * @param {Node} this This node
35414     */
35415     /**
35416      * The loader used by this node (defaults to using the tree's defined loader)
35417      * @type TreeLoader
35418      * @property loader
35419      */
35420 };
35421 Roo.extend(Roo.tree.AsyncTreeNode, Roo.tree.TreeNode, {
35422     expand : function(deep, anim, callback){
35423         if(this.loading){ // if an async load is already running, waiting til it's done
35424             var timer;
35425             var f = function(){
35426                 if(!this.loading){ // done loading
35427                     clearInterval(timer);
35428                     this.expand(deep, anim, callback);
35429                 }
35430             }.createDelegate(this);
35431             timer = setInterval(f, 200);
35432             return;
35433         }
35434         if(!this.loaded){
35435             if(this.fireEvent("beforeload", this) === false){
35436                 return;
35437             }
35438             this.loading = true;
35439             this.ui.beforeLoad(this);
35440             var loader = this.loader || this.attributes.loader || this.getOwnerTree().getLoader();
35441             if(loader){
35442                 loader.load(this, this.loadComplete.createDelegate(this, [deep, anim, callback]));
35443                 return;
35444             }
35445         }
35446         Roo.tree.AsyncTreeNode.superclass.expand.call(this, deep, anim, callback);
35447     },
35448     
35449     /**
35450      * Returns true if this node is currently loading
35451      * @return {Boolean}
35452      */
35453     isLoading : function(){
35454         return this.loading;  
35455     },
35456     
35457     loadComplete : function(deep, anim, callback){
35458         this.loading = false;
35459         this.loaded = true;
35460         this.ui.afterLoad(this);
35461         this.fireEvent("load", this);
35462         this.expand(deep, anim, callback);
35463     },
35464     
35465     /**
35466      * Returns true if this node has been loaded
35467      * @return {Boolean}
35468      */
35469     isLoaded : function(){
35470         return this.loaded;
35471     },
35472     
35473     hasChildNodes : function(){
35474         if(!this.isLeaf() && !this.loaded){
35475             return true;
35476         }else{
35477             return Roo.tree.AsyncTreeNode.superclass.hasChildNodes.call(this);
35478         }
35479     },
35480
35481     /**
35482      * Trigger a reload for this node
35483      * @param {Function} callback
35484      */
35485     reload : function(callback){
35486         this.collapse(false, false);
35487         while(this.firstChild){
35488             this.removeChild(this.firstChild);
35489         }
35490         this.childrenRendered = false;
35491         this.loaded = false;
35492         if(this.isHiddenRoot()){
35493             this.expanded = false;
35494         }
35495         this.expand(false, false, callback);
35496     }
35497 });/*
35498  * Based on:
35499  * Ext JS Library 1.1.1
35500  * Copyright(c) 2006-2007, Ext JS, LLC.
35501  *
35502  * Originally Released Under LGPL - original licence link has changed is not relivant.
35503  *
35504  * Fork - LGPL
35505  * <script type="text/javascript">
35506  */
35507  
35508 /**
35509  * @class Roo.tree.TreeNodeUI
35510  * @constructor
35511  * @param {Object} node The node to render
35512  * The TreeNode UI implementation is separate from the
35513  * tree implementation. Unless you are customizing the tree UI,
35514  * you should never have to use this directly.
35515  */
35516 Roo.tree.TreeNodeUI = function(node){
35517     this.node = node;
35518     this.rendered = false;
35519     this.animating = false;
35520     this.emptyIcon = Roo.BLANK_IMAGE_URL;
35521 };
35522
35523 Roo.tree.TreeNodeUI.prototype = {
35524     removeChild : function(node){
35525         if(this.rendered){
35526             this.ctNode.removeChild(node.ui.getEl());
35527         }
35528     },
35529
35530     beforeLoad : function(){
35531          this.addClass("x-tree-node-loading");
35532     },
35533
35534     afterLoad : function(){
35535          this.removeClass("x-tree-node-loading");
35536     },
35537
35538     onTextChange : function(node, text, oldText){
35539         if(this.rendered){
35540             this.textNode.innerHTML = text;
35541         }
35542     },
35543
35544     onDisableChange : function(node, state){
35545         this.disabled = state;
35546         if(state){
35547             this.addClass("x-tree-node-disabled");
35548         }else{
35549             this.removeClass("x-tree-node-disabled");
35550         }
35551     },
35552
35553     onSelectedChange : function(state){
35554         if(state){
35555             this.focus();
35556             this.addClass("x-tree-selected");
35557         }else{
35558             //this.blur();
35559             this.removeClass("x-tree-selected");
35560         }
35561     },
35562
35563     onMove : function(tree, node, oldParent, newParent, index, refNode){
35564         this.childIndent = null;
35565         if(this.rendered){
35566             var targetNode = newParent.ui.getContainer();
35567             if(!targetNode){//target not rendered
35568                 this.holder = document.createElement("div");
35569                 this.holder.appendChild(this.wrap);
35570                 return;
35571             }
35572             var insertBefore = refNode ? refNode.ui.getEl() : null;
35573             if(insertBefore){
35574                 targetNode.insertBefore(this.wrap, insertBefore);
35575             }else{
35576                 targetNode.appendChild(this.wrap);
35577             }
35578             this.node.renderIndent(true);
35579         }
35580     },
35581
35582     addClass : function(cls){
35583         if(this.elNode){
35584             Roo.fly(this.elNode).addClass(cls);
35585         }
35586     },
35587
35588     removeClass : function(cls){
35589         if(this.elNode){
35590             Roo.fly(this.elNode).removeClass(cls);
35591         }
35592     },
35593
35594     remove : function(){
35595         if(this.rendered){
35596             this.holder = document.createElement("div");
35597             this.holder.appendChild(this.wrap);
35598         }
35599     },
35600
35601     fireEvent : function(){
35602         return this.node.fireEvent.apply(this.node, arguments);
35603     },
35604
35605     initEvents : function(){
35606         this.node.on("move", this.onMove, this);
35607         var E = Roo.EventManager;
35608         var a = this.anchor;
35609
35610         var el = Roo.fly(a, '_treeui');
35611
35612         if(Roo.isOpera){ // opera render bug ignores the CSS
35613             el.setStyle("text-decoration", "none");
35614         }
35615
35616         el.on("click", this.onClick, this);
35617         el.on("dblclick", this.onDblClick, this);
35618
35619         if(this.checkbox){
35620             Roo.EventManager.on(this.checkbox,
35621                     Roo.isIE ? 'click' : 'change', this.onCheckChange, this);
35622         }
35623
35624         el.on("contextmenu", this.onContextMenu, this);
35625
35626         var icon = Roo.fly(this.iconNode);
35627         icon.on("click", this.onClick, this);
35628         icon.on("dblclick", this.onDblClick, this);
35629         icon.on("contextmenu", this.onContextMenu, this);
35630         E.on(this.ecNode, "click", this.ecClick, this, true);
35631
35632         if(this.node.disabled){
35633             this.addClass("x-tree-node-disabled");
35634         }
35635         if(this.node.hidden){
35636             this.addClass("x-tree-node-disabled");
35637         }
35638         var ot = this.node.getOwnerTree();
35639         var dd = ot ? (ot.enableDD || ot.enableDrag || ot.enableDrop) : false;
35640         if(dd && (!this.node.isRoot || ot.rootVisible)){
35641             Roo.dd.Registry.register(this.elNode, {
35642                 node: this.node,
35643                 handles: this.getDDHandles(),
35644                 isHandle: false
35645             });
35646         }
35647     },
35648
35649     getDDHandles : function(){
35650         return [this.iconNode, this.textNode];
35651     },
35652
35653     hide : function(){
35654         if(this.rendered){
35655             this.wrap.style.display = "none";
35656         }
35657     },
35658
35659     show : function(){
35660         if(this.rendered){
35661             this.wrap.style.display = "";
35662         }
35663     },
35664
35665     onContextMenu : function(e){
35666         if (this.node.hasListener("contextmenu") || this.node.getOwnerTree().hasListener("contextmenu")) {
35667             e.preventDefault();
35668             this.focus();
35669             this.fireEvent("contextmenu", this.node, e);
35670         }
35671     },
35672
35673     onClick : function(e){
35674         if(this.dropping){
35675             e.stopEvent();
35676             return;
35677         }
35678         if(this.fireEvent("beforeclick", this.node, e) !== false){
35679             if(!this.disabled && this.node.attributes.href){
35680                 this.fireEvent("click", this.node, e);
35681                 return;
35682             }
35683             e.preventDefault();
35684             if(this.disabled){
35685                 return;
35686             }
35687
35688             if(this.node.attributes.singleClickExpand && !this.animating && this.node.hasChildNodes()){
35689                 this.node.toggle();
35690             }
35691
35692             this.fireEvent("click", this.node, e);
35693         }else{
35694             e.stopEvent();
35695         }
35696     },
35697
35698     onDblClick : function(e){
35699         e.preventDefault();
35700         if(this.disabled){
35701             return;
35702         }
35703         if(this.checkbox){
35704             this.toggleCheck();
35705         }
35706         if(!this.animating && this.node.hasChildNodes()){
35707             this.node.toggle();
35708         }
35709         this.fireEvent("dblclick", this.node, e);
35710     },
35711
35712     onCheckChange : function(){
35713         var checked = this.checkbox.checked;
35714         this.node.attributes.checked = checked;
35715         this.fireEvent('checkchange', this.node, checked);
35716     },
35717
35718     ecClick : function(e){
35719         if(!this.animating && this.node.hasChildNodes()){
35720             this.node.toggle();
35721         }
35722     },
35723
35724     startDrop : function(){
35725         this.dropping = true;
35726     },
35727
35728     // delayed drop so the click event doesn't get fired on a drop
35729     endDrop : function(){
35730        setTimeout(function(){
35731            this.dropping = false;
35732        }.createDelegate(this), 50);
35733     },
35734
35735     expand : function(){
35736         this.updateExpandIcon();
35737         this.ctNode.style.display = "";
35738     },
35739
35740     focus : function(){
35741         if(!this.node.preventHScroll){
35742             try{this.anchor.focus();
35743             }catch(e){}
35744         }else if(!Roo.isIE){
35745             try{
35746                 var noscroll = this.node.getOwnerTree().getTreeEl().dom;
35747                 var l = noscroll.scrollLeft;
35748                 this.anchor.focus();
35749                 noscroll.scrollLeft = l;
35750             }catch(e){}
35751         }
35752     },
35753
35754     toggleCheck : function(value){
35755         var cb = this.checkbox;
35756         if(cb){
35757             cb.checked = (value === undefined ? !cb.checked : value);
35758         }
35759     },
35760
35761     blur : function(){
35762         try{
35763             this.anchor.blur();
35764         }catch(e){}
35765     },
35766
35767     animExpand : function(callback){
35768         var ct = Roo.get(this.ctNode);
35769         ct.stopFx();
35770         if(!this.node.hasChildNodes()){
35771             this.updateExpandIcon();
35772             this.ctNode.style.display = "";
35773             Roo.callback(callback);
35774             return;
35775         }
35776         this.animating = true;
35777         this.updateExpandIcon();
35778
35779         ct.slideIn('t', {
35780            callback : function(){
35781                this.animating = false;
35782                Roo.callback(callback);
35783             },
35784             scope: this,
35785             duration: this.node.ownerTree.duration || .25
35786         });
35787     },
35788
35789     highlight : function(){
35790         var tree = this.node.getOwnerTree();
35791         Roo.fly(this.wrap).highlight(
35792             tree.hlColor || "C3DAF9",
35793             {endColor: tree.hlBaseColor}
35794         );
35795     },
35796
35797     collapse : function(){
35798         this.updateExpandIcon();
35799         this.ctNode.style.display = "none";
35800     },
35801
35802     animCollapse : function(callback){
35803         var ct = Roo.get(this.ctNode);
35804         ct.enableDisplayMode('block');
35805         ct.stopFx();
35806
35807         this.animating = true;
35808         this.updateExpandIcon();
35809
35810         ct.slideOut('t', {
35811             callback : function(){
35812                this.animating = false;
35813                Roo.callback(callback);
35814             },
35815             scope: this,
35816             duration: this.node.ownerTree.duration || .25
35817         });
35818     },
35819
35820     getContainer : function(){
35821         return this.ctNode;
35822     },
35823
35824     getEl : function(){
35825         return this.wrap;
35826     },
35827
35828     appendDDGhost : function(ghostNode){
35829         ghostNode.appendChild(this.elNode.cloneNode(true));
35830     },
35831
35832     getDDRepairXY : function(){
35833         return Roo.lib.Dom.getXY(this.iconNode);
35834     },
35835
35836     onRender : function(){
35837         this.render();
35838     },
35839
35840     render : function(bulkRender){
35841         var n = this.node, a = n.attributes;
35842         var targetNode = n.parentNode ?
35843               n.parentNode.ui.getContainer() : n.ownerTree.innerCt.dom;
35844
35845         if(!this.rendered){
35846             this.rendered = true;
35847
35848             this.renderElements(n, a, targetNode, bulkRender);
35849
35850             if(a.qtip){
35851                if(this.textNode.setAttributeNS){
35852                    this.textNode.setAttributeNS("ext", "qtip", a.qtip);
35853                    if(a.qtipTitle){
35854                        this.textNode.setAttributeNS("ext", "qtitle", a.qtipTitle);
35855                    }
35856                }else{
35857                    this.textNode.setAttribute("ext:qtip", a.qtip);
35858                    if(a.qtipTitle){
35859                        this.textNode.setAttribute("ext:qtitle", a.qtipTitle);
35860                    }
35861                }
35862             }else if(a.qtipCfg){
35863                 a.qtipCfg.target = Roo.id(this.textNode);
35864                 Roo.QuickTips.register(a.qtipCfg);
35865             }
35866             this.initEvents();
35867             if(!this.node.expanded){
35868                 this.updateExpandIcon();
35869             }
35870         }else{
35871             if(bulkRender === true) {
35872                 targetNode.appendChild(this.wrap);
35873             }
35874         }
35875     },
35876
35877     renderElements : function(n, a, targetNode, bulkRender)
35878     {
35879         // add some indent caching, this helps performance when rendering a large tree
35880         this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
35881         var t = n.getOwnerTree();
35882         var txt = t && t.renderer ? t.renderer(n.attributes) : Roo.util.Format.htmlEncode(n.text);
35883         if (typeof(n.attributes.html) != 'undefined') {
35884             txt = n.attributes.html;
35885         }
35886         var tip = t && t.rendererTip ? t.rendererTip(n.attributes) : txt;
35887         var cb = typeof a.checked == 'boolean';
35888         var href = a.href ? a.href : Roo.isGecko ? "" : "#";
35889         var buf = ['<li class="x-tree-node"><div class="x-tree-node-el ', a.cls,'">',
35890             '<span class="x-tree-node-indent">',this.indentMarkup,"</span>",
35891             '<img src="', this.emptyIcon, '" class="x-tree-ec-icon" />',
35892             '<img src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',(a.icon ? " x-tree-node-inline-icon" : ""),(a.iconCls ? " "+a.iconCls : ""),'" unselectable="on" />',
35893             cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + (a.checked ? 'checked="checked" />' : ' />')) : '',
35894             '<a hidefocus="on" href="',href,'" tabIndex="1" ',
35895              a.hrefTarget ? ' target="'+a.hrefTarget+'"' : "", 
35896                 '><span unselectable="on" qtip="' , tip ,'">',txt,"</span></a></div>",
35897             '<ul class="x-tree-node-ct" style="display:none;"></ul>',
35898             "</li>"];
35899
35900         if(bulkRender !== true && n.nextSibling && n.nextSibling.ui.getEl()){
35901             this.wrap = Roo.DomHelper.insertHtml("beforeBegin",
35902                                 n.nextSibling.ui.getEl(), buf.join(""));
35903         }else{
35904             this.wrap = Roo.DomHelper.insertHtml("beforeEnd", targetNode, buf.join(""));
35905         }
35906
35907         this.elNode = this.wrap.childNodes[0];
35908         this.ctNode = this.wrap.childNodes[1];
35909         var cs = this.elNode.childNodes;
35910         this.indentNode = cs[0];
35911         this.ecNode = cs[1];
35912         this.iconNode = cs[2];
35913         var index = 3;
35914         if(cb){
35915             this.checkbox = cs[3];
35916             index++;
35917         }
35918         this.anchor = cs[index];
35919         this.textNode = cs[index].firstChild;
35920     },
35921
35922     getAnchor : function(){
35923         return this.anchor;
35924     },
35925
35926     getTextEl : function(){
35927         return this.textNode;
35928     },
35929
35930     getIconEl : function(){
35931         return this.iconNode;
35932     },
35933
35934     isChecked : function(){
35935         return this.checkbox ? this.checkbox.checked : false;
35936     },
35937
35938     updateExpandIcon : function(){
35939         if(this.rendered){
35940             var n = this.node, c1, c2;
35941             var cls = n.isLast() ? "x-tree-elbow-end" : "x-tree-elbow";
35942             var hasChild = n.hasChildNodes();
35943             if(hasChild){
35944                 if(n.expanded){
35945                     cls += "-minus";
35946                     c1 = "x-tree-node-collapsed";
35947                     c2 = "x-tree-node-expanded";
35948                 }else{
35949                     cls += "-plus";
35950                     c1 = "x-tree-node-expanded";
35951                     c2 = "x-tree-node-collapsed";
35952                 }
35953                 if(this.wasLeaf){
35954                     this.removeClass("x-tree-node-leaf");
35955                     this.wasLeaf = false;
35956                 }
35957                 if(this.c1 != c1 || this.c2 != c2){
35958                     Roo.fly(this.elNode).replaceClass(c1, c2);
35959                     this.c1 = c1; this.c2 = c2;
35960                 }
35961             }else{
35962                 // this changes non-leafs into leafs if they have no children.
35963                 // it's not very rational behaviour..
35964                 
35965                 if(!this.wasLeaf && this.node.leaf){
35966                     Roo.fly(this.elNode).replaceClass("x-tree-node-expanded", "x-tree-node-leaf");
35967                     delete this.c1;
35968                     delete this.c2;
35969                     this.wasLeaf = true;
35970                 }
35971             }
35972             var ecc = "x-tree-ec-icon "+cls;
35973             if(this.ecc != ecc){
35974                 this.ecNode.className = ecc;
35975                 this.ecc = ecc;
35976             }
35977         }
35978     },
35979
35980     getChildIndent : function(){
35981         if(!this.childIndent){
35982             var buf = [];
35983             var p = this.node;
35984             while(p){
35985                 if(!p.isRoot || (p.isRoot && p.ownerTree.rootVisible)){
35986                     if(!p.isLast()) {
35987                         buf.unshift('<img src="'+this.emptyIcon+'" class="x-tree-elbow-line" />');
35988                     } else {
35989                         buf.unshift('<img src="'+this.emptyIcon+'" class="x-tree-icon" />');
35990                     }
35991                 }
35992                 p = p.parentNode;
35993             }
35994             this.childIndent = buf.join("");
35995         }
35996         return this.childIndent;
35997     },
35998
35999     renderIndent : function(){
36000         if(this.rendered){
36001             var indent = "";
36002             var p = this.node.parentNode;
36003             if(p){
36004                 indent = p.ui.getChildIndent();
36005             }
36006             if(this.indentMarkup != indent){ // don't rerender if not required
36007                 this.indentNode.innerHTML = indent;
36008                 this.indentMarkup = indent;
36009             }
36010             this.updateExpandIcon();
36011         }
36012     }
36013 };
36014
36015 Roo.tree.RootTreeNodeUI = function(){
36016     Roo.tree.RootTreeNodeUI.superclass.constructor.apply(this, arguments);
36017 };
36018 Roo.extend(Roo.tree.RootTreeNodeUI, Roo.tree.TreeNodeUI, {
36019     render : function(){
36020         if(!this.rendered){
36021             var targetNode = this.node.ownerTree.innerCt.dom;
36022             this.node.expanded = true;
36023             targetNode.innerHTML = '<div class="x-tree-root-node"></div>';
36024             this.wrap = this.ctNode = targetNode.firstChild;
36025         }
36026     },
36027     collapse : function(){
36028     },
36029     expand : function(){
36030     }
36031 });/*
36032  * Based on:
36033  * Ext JS Library 1.1.1
36034  * Copyright(c) 2006-2007, Ext JS, LLC.
36035  *
36036  * Originally Released Under LGPL - original licence link has changed is not relivant.
36037  *
36038  * Fork - LGPL
36039  * <script type="text/javascript">
36040  */
36041 /**
36042  * @class Roo.tree.TreeLoader
36043  * @extends Roo.util.Observable
36044  * A TreeLoader provides for lazy loading of an {@link Roo.tree.TreeNode}'s child
36045  * nodes from a specified URL. The response must be a javascript Array definition
36046  * who's elements are node definition objects. eg:
36047  * <pre><code>
36048 {  success : true,
36049    data :      [
36050    
36051     { 'id': 1, 'text': 'A folder Node', 'leaf': false },
36052     { 'id': 2, 'text': 'A leaf Node', 'leaf': true }
36053     ]
36054 }
36055
36056
36057 </code></pre>
36058  * <br><br>
36059  * The old style respose with just an array is still supported, but not recommended.
36060  * <br><br>
36061  *
36062  * A server request is sent, and child nodes are loaded only when a node is expanded.
36063  * The loading node's id is passed to the server under the parameter name "node" to
36064  * enable the server to produce the correct child nodes.
36065  * <br><br>
36066  * To pass extra parameters, an event handler may be attached to the "beforeload"
36067  * event, and the parameters specified in the TreeLoader's baseParams property:
36068  * <pre><code>
36069     myTreeLoader.on("beforeload", function(treeLoader, node) {
36070         this.baseParams.category = node.attributes.category;
36071     }, this);
36072     
36073 </code></pre>
36074  *
36075  * This would pass an HTTP parameter called "category" to the server containing
36076  * the value of the Node's "category" attribute.
36077  * @constructor
36078  * Creates a new Treeloader.
36079  * @param {Object} config A config object containing config properties.
36080  */
36081 Roo.tree.TreeLoader = function(config){
36082     this.baseParams = {};
36083     this.requestMethod = "POST";
36084     Roo.apply(this, config);
36085
36086     this.addEvents({
36087     
36088         /**
36089          * @event beforeload
36090          * Fires before a network request is made to retrieve the Json text which specifies a node's children.
36091          * @param {Object} This TreeLoader object.
36092          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36093          * @param {Object} callback The callback function specified in the {@link #load} call.
36094          */
36095         beforeload : true,
36096         /**
36097          * @event load
36098          * Fires when the node has been successfuly loaded.
36099          * @param {Object} This TreeLoader object.
36100          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36101          * @param {Object} response The response object containing the data from the server.
36102          */
36103         load : true,
36104         /**
36105          * @event loadexception
36106          * Fires if the network request failed.
36107          * @param {Object} This TreeLoader object.
36108          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36109          * @param {Object} response The response object containing the data from the server.
36110          */
36111         loadexception : true,
36112         /**
36113          * @event create
36114          * Fires before a node is created, enabling you to return custom Node types 
36115          * @param {Object} This TreeLoader object.
36116          * @param {Object} attr - the data returned from the AJAX call (modify it to suit)
36117          */
36118         create : true
36119     });
36120
36121     Roo.tree.TreeLoader.superclass.constructor.call(this);
36122 };
36123
36124 Roo.extend(Roo.tree.TreeLoader, Roo.util.Observable, {
36125     /**
36126     * @cfg {String} dataUrl The URL from which to request a Json string which
36127     * specifies an array of node definition object representing the child nodes
36128     * to be loaded.
36129     */
36130     /**
36131     * @cfg {String} requestMethod either GET or POST
36132     * defaults to POST (due to BC)
36133     * to be loaded.
36134     */
36135     /**
36136     * @cfg {Object} baseParams (optional) An object containing properties which
36137     * specify HTTP parameters to be passed to each request for child nodes.
36138     */
36139     /**
36140     * @cfg {Object} baseAttrs (optional) An object containing attributes to be added to all nodes
36141     * created by this loader. If the attributes sent by the server have an attribute in this object,
36142     * they take priority.
36143     */
36144     /**
36145     * @cfg {Object} uiProviders (optional) An object containing properties which
36146     * 
36147     * DEPRECATED - use 'create' event handler to modify attributes - which affect creation.
36148     * specify custom {@link Roo.tree.TreeNodeUI} implementations. If the optional
36149     * <i>uiProvider</i> attribute of a returned child node is a string rather
36150     * than a reference to a TreeNodeUI implementation, this that string value
36151     * is used as a property name in the uiProviders object. You can define the provider named
36152     * 'default' , and this will be used for all nodes (if no uiProvider is delivered by the node data)
36153     */
36154     uiProviders : {},
36155
36156     /**
36157     * @cfg {Boolean} clearOnLoad (optional) Default to true. Remove previously existing
36158     * child nodes before loading.
36159     */
36160     clearOnLoad : true,
36161
36162     /**
36163     * @cfg {String} root (optional) Default to false. Use this to read data from an object 
36164     * property on loading, rather than expecting an array. (eg. more compatible to a standard
36165     * Grid query { data : [ .....] }
36166     */
36167     
36168     root : false,
36169      /**
36170     * @cfg {String} queryParam (optional) 
36171     * Name of the query as it will be passed on the querystring (defaults to 'node')
36172     * eg. the request will be ?node=[id]
36173     */
36174     
36175     
36176     queryParam: false,
36177     
36178     /**
36179      * Load an {@link Roo.tree.TreeNode} from the URL specified in the constructor.
36180      * This is called automatically when a node is expanded, but may be used to reload
36181      * a node (or append new children if the {@link #clearOnLoad} option is false.)
36182      * @param {Roo.tree.TreeNode} node
36183      * @param {Function} callback
36184      */
36185     load : function(node, callback){
36186         if(this.clearOnLoad){
36187             while(node.firstChild){
36188                 node.removeChild(node.firstChild);
36189             }
36190         }
36191         if(node.attributes.children){ // preloaded json children
36192             var cs = node.attributes.children;
36193             for(var i = 0, len = cs.length; i < len; i++){
36194                 node.appendChild(this.createNode(cs[i]));
36195             }
36196             if(typeof callback == "function"){
36197                 callback();
36198             }
36199         }else if(this.dataUrl){
36200             this.requestData(node, callback);
36201         }
36202     },
36203
36204     getParams: function(node){
36205         var buf = [], bp = this.baseParams;
36206         for(var key in bp){
36207             if(typeof bp[key] != "function"){
36208                 buf.push(encodeURIComponent(key), "=", encodeURIComponent(bp[key]), "&");
36209             }
36210         }
36211         var n = this.queryParam === false ? 'node' : this.queryParam;
36212         buf.push(n + "=", encodeURIComponent(node.id));
36213         return buf.join("");
36214     },
36215
36216     requestData : function(node, callback){
36217         if(this.fireEvent("beforeload", this, node, callback) !== false){
36218             this.transId = Roo.Ajax.request({
36219                 method:this.requestMethod,
36220                 url: this.dataUrl||this.url,
36221                 success: this.handleResponse,
36222                 failure: this.handleFailure,
36223                 scope: this,
36224                 argument: {callback: callback, node: node},
36225                 params: this.getParams(node)
36226             });
36227         }else{
36228             // if the load is cancelled, make sure we notify
36229             // the node that we are done
36230             if(typeof callback == "function"){
36231                 callback();
36232             }
36233         }
36234     },
36235
36236     isLoading : function(){
36237         return this.transId ? true : false;
36238     },
36239
36240     abort : function(){
36241         if(this.isLoading()){
36242             Roo.Ajax.abort(this.transId);
36243         }
36244     },
36245
36246     // private
36247     createNode : function(attr)
36248     {
36249         // apply baseAttrs, nice idea Corey!
36250         if(this.baseAttrs){
36251             Roo.applyIf(attr, this.baseAttrs);
36252         }
36253         if(this.applyLoader !== false){
36254             attr.loader = this;
36255         }
36256         // uiProvider = depreciated..
36257         
36258         if(typeof(attr.uiProvider) == 'string'){
36259            attr.uiProvider = this.uiProviders[attr.uiProvider] || 
36260                 /**  eval:var:attr */ eval(attr.uiProvider);
36261         }
36262         if(typeof(this.uiProviders['default']) != 'undefined') {
36263             attr.uiProvider = this.uiProviders['default'];
36264         }
36265         
36266         this.fireEvent('create', this, attr);
36267         
36268         attr.leaf  = typeof(attr.leaf) == 'string' ? attr.leaf * 1 : attr.leaf;
36269         return(attr.leaf ?
36270                         new Roo.tree.TreeNode(attr) :
36271                         new Roo.tree.AsyncTreeNode(attr));
36272     },
36273
36274     processResponse : function(response, node, callback)
36275     {
36276         var json = response.responseText;
36277         try {
36278             
36279             var o = Roo.decode(json);
36280             
36281             if (this.root === false && typeof(o.success) != undefined) {
36282                 this.root = 'data'; // the default behaviour for list like data..
36283                 }
36284                 
36285             if (this.root !== false &&  !o.success) {
36286                 // it's a failure condition.
36287                 var a = response.argument;
36288                 this.fireEvent("loadexception", this, a.node, response);
36289                 Roo.log("Load failed - should have a handler really");
36290                 return;
36291             }
36292             
36293             
36294             
36295             if (this.root !== false) {
36296                  o = o[this.root];
36297             }
36298             
36299             for(var i = 0, len = o.length; i < len; i++){
36300                 var n = this.createNode(o[i]);
36301                 if(n){
36302                     node.appendChild(n);
36303                 }
36304             }
36305             if(typeof callback == "function"){
36306                 callback(this, node);
36307             }
36308         }catch(e){
36309             this.handleFailure(response);
36310         }
36311     },
36312
36313     handleResponse : function(response){
36314         this.transId = false;
36315         var a = response.argument;
36316         this.processResponse(response, a.node, a.callback);
36317         this.fireEvent("load", this, a.node, response);
36318     },
36319
36320     handleFailure : function(response)
36321     {
36322         // should handle failure better..
36323         this.transId = false;
36324         var a = response.argument;
36325         this.fireEvent("loadexception", this, a.node, response);
36326         if(typeof a.callback == "function"){
36327             a.callback(this, a.node);
36328         }
36329     }
36330 });/*
36331  * Based on:
36332  * Ext JS Library 1.1.1
36333  * Copyright(c) 2006-2007, Ext JS, LLC.
36334  *
36335  * Originally Released Under LGPL - original licence link has changed is not relivant.
36336  *
36337  * Fork - LGPL
36338  * <script type="text/javascript">
36339  */
36340
36341 /**
36342 * @class Roo.tree.TreeFilter
36343 * Note this class is experimental and doesn't update the indent (lines) or expand collapse icons of the nodes
36344 * @param {TreePanel} tree
36345 * @param {Object} config (optional)
36346  */
36347 Roo.tree.TreeFilter = function(tree, config){
36348     this.tree = tree;
36349     this.filtered = {};
36350     Roo.apply(this, config);
36351 };
36352
36353 Roo.tree.TreeFilter.prototype = {
36354     clearBlank:false,
36355     reverse:false,
36356     autoClear:false,
36357     remove:false,
36358
36359      /**
36360      * Filter the data by a specific attribute.
36361      * @param {String/RegExp} value Either string that the attribute value
36362      * should start with or a RegExp to test against the attribute
36363      * @param {String} attr (optional) The attribute passed in your node's attributes collection. Defaults to "text".
36364      * @param {TreeNode} startNode (optional) The node to start the filter at.
36365      */
36366     filter : function(value, attr, startNode){
36367         attr = attr || "text";
36368         var f;
36369         if(typeof value == "string"){
36370             var vlen = value.length;
36371             // auto clear empty filter
36372             if(vlen == 0 && this.clearBlank){
36373                 this.clear();
36374                 return;
36375             }
36376             value = value.toLowerCase();
36377             f = function(n){
36378                 return n.attributes[attr].substr(0, vlen).toLowerCase() == value;
36379             };
36380         }else if(value.exec){ // regex?
36381             f = function(n){
36382                 return value.test(n.attributes[attr]);
36383             };
36384         }else{
36385             throw 'Illegal filter type, must be string or regex';
36386         }
36387         this.filterBy(f, null, startNode);
36388         },
36389
36390     /**
36391      * Filter by a function. The passed function will be called with each
36392      * node in the tree (or from the startNode). If the function returns true, the node is kept
36393      * otherwise it is filtered. If a node is filtered, its children are also filtered.
36394      * @param {Function} fn The filter function
36395      * @param {Object} scope (optional) The scope of the function (defaults to the current node)
36396      */
36397     filterBy : function(fn, scope, startNode){
36398         startNode = startNode || this.tree.root;
36399         if(this.autoClear){
36400             this.clear();
36401         }
36402         var af = this.filtered, rv = this.reverse;
36403         var f = function(n){
36404             if(n == startNode){
36405                 return true;
36406             }
36407             if(af[n.id]){
36408                 return false;
36409             }
36410             var m = fn.call(scope || n, n);
36411             if(!m || rv){
36412                 af[n.id] = n;
36413                 n.ui.hide();
36414                 return false;
36415             }
36416             return true;
36417         };
36418         startNode.cascade(f);
36419         if(this.remove){
36420            for(var id in af){
36421                if(typeof id != "function"){
36422                    var n = af[id];
36423                    if(n && n.parentNode){
36424                        n.parentNode.removeChild(n);
36425                    }
36426                }
36427            }
36428         }
36429     },
36430
36431     /**
36432      * Clears the current filter. Note: with the "remove" option
36433      * set a filter cannot be cleared.
36434      */
36435     clear : function(){
36436         var t = this.tree;
36437         var af = this.filtered;
36438         for(var id in af){
36439             if(typeof id != "function"){
36440                 var n = af[id];
36441                 if(n){
36442                     n.ui.show();
36443                 }
36444             }
36445         }
36446         this.filtered = {};
36447     }
36448 };
36449 /*
36450  * Based on:
36451  * Ext JS Library 1.1.1
36452  * Copyright(c) 2006-2007, Ext JS, LLC.
36453  *
36454  * Originally Released Under LGPL - original licence link has changed is not relivant.
36455  *
36456  * Fork - LGPL
36457  * <script type="text/javascript">
36458  */
36459  
36460
36461 /**
36462  * @class Roo.tree.TreeSorter
36463  * Provides sorting of nodes in a TreePanel
36464  * 
36465  * @cfg {Boolean} folderSort True to sort leaf nodes under non leaf nodes
36466  * @cfg {String} property The named attribute on the node to sort by (defaults to text)
36467  * @cfg {String} dir The direction to sort (asc or desc) (defaults to asc)
36468  * @cfg {String} leafAttr The attribute used to determine leaf nodes in folder sort (defaults to "leaf")
36469  * @cfg {Boolean} caseSensitive true for case sensitive sort (defaults to false)
36470  * @cfg {Function} sortType A custom "casting" function used to convert node values before sorting
36471  * @constructor
36472  * @param {TreePanel} tree
36473  * @param {Object} config
36474  */
36475 Roo.tree.TreeSorter = function(tree, config){
36476     Roo.apply(this, config);
36477     tree.on("beforechildrenrendered", this.doSort, this);
36478     tree.on("append", this.updateSort, this);
36479     tree.on("insert", this.updateSort, this);
36480     
36481     var dsc = this.dir && this.dir.toLowerCase() == "desc";
36482     var p = this.property || "text";
36483     var sortType = this.sortType;
36484     var fs = this.folderSort;
36485     var cs = this.caseSensitive === true;
36486     var leafAttr = this.leafAttr || 'leaf';
36487
36488     this.sortFn = function(n1, n2){
36489         if(fs){
36490             if(n1.attributes[leafAttr] && !n2.attributes[leafAttr]){
36491                 return 1;
36492             }
36493             if(!n1.attributes[leafAttr] && n2.attributes[leafAttr]){
36494                 return -1;
36495             }
36496         }
36497         var v1 = sortType ? sortType(n1) : (cs ? n1.attributes[p] : n1.attributes[p].toUpperCase());
36498         var v2 = sortType ? sortType(n2) : (cs ? n2.attributes[p] : n2.attributes[p].toUpperCase());
36499         if(v1 < v2){
36500                         return dsc ? +1 : -1;
36501                 }else if(v1 > v2){
36502                         return dsc ? -1 : +1;
36503         }else{
36504                 return 0;
36505         }
36506     };
36507 };
36508
36509 Roo.tree.TreeSorter.prototype = {
36510     doSort : function(node){
36511         node.sort(this.sortFn);
36512     },
36513     
36514     compareNodes : function(n1, n2){
36515         return (n1.text.toUpperCase() > n2.text.toUpperCase() ? 1 : -1);
36516     },
36517     
36518     updateSort : function(tree, node){
36519         if(node.childrenRendered){
36520             this.doSort.defer(1, this, [node]);
36521         }
36522     }
36523 };/*
36524  * Based on:
36525  * Ext JS Library 1.1.1
36526  * Copyright(c) 2006-2007, Ext JS, LLC.
36527  *
36528  * Originally Released Under LGPL - original licence link has changed is not relivant.
36529  *
36530  * Fork - LGPL
36531  * <script type="text/javascript">
36532  */
36533
36534 if(Roo.dd.DropZone){
36535     
36536 Roo.tree.TreeDropZone = function(tree, config){
36537     this.allowParentInsert = false;
36538     this.allowContainerDrop = false;
36539     this.appendOnly = false;
36540     Roo.tree.TreeDropZone.superclass.constructor.call(this, tree.innerCt, config);
36541     this.tree = tree;
36542     this.lastInsertClass = "x-tree-no-status";
36543     this.dragOverData = {};
36544 };
36545
36546 Roo.extend(Roo.tree.TreeDropZone, Roo.dd.DropZone, {
36547     ddGroup : "TreeDD",
36548     scroll:  true,
36549     
36550     expandDelay : 1000,
36551     
36552     expandNode : function(node){
36553         if(node.hasChildNodes() && !node.isExpanded()){
36554             node.expand(false, null, this.triggerCacheRefresh.createDelegate(this));
36555         }
36556     },
36557     
36558     queueExpand : function(node){
36559         this.expandProcId = this.expandNode.defer(this.expandDelay, this, [node]);
36560     },
36561     
36562     cancelExpand : function(){
36563         if(this.expandProcId){
36564             clearTimeout(this.expandProcId);
36565             this.expandProcId = false;
36566         }
36567     },
36568     
36569     isValidDropPoint : function(n, pt, dd, e, data){
36570         if(!n || !data){ return false; }
36571         var targetNode = n.node;
36572         var dropNode = data.node;
36573         // default drop rules
36574         if(!(targetNode && targetNode.isTarget && pt)){
36575             return false;
36576         }
36577         if(pt == "append" && targetNode.allowChildren === false){
36578             return false;
36579         }
36580         if((pt == "above" || pt == "below") && (targetNode.parentNode && targetNode.parentNode.allowChildren === false)){
36581             return false;
36582         }
36583         if(dropNode && (targetNode == dropNode || dropNode.contains(targetNode))){
36584             return false;
36585         }
36586         // reuse the object
36587         var overEvent = this.dragOverData;
36588         overEvent.tree = this.tree;
36589         overEvent.target = targetNode;
36590         overEvent.data = data;
36591         overEvent.point = pt;
36592         overEvent.source = dd;
36593         overEvent.rawEvent = e;
36594         overEvent.dropNode = dropNode;
36595         overEvent.cancel = false;  
36596         var result = this.tree.fireEvent("nodedragover", overEvent);
36597         return overEvent.cancel === false && result !== false;
36598     },
36599     
36600     getDropPoint : function(e, n, dd)
36601     {
36602         var tn = n.node;
36603         if(tn.isRoot){
36604             return tn.allowChildren !== false ? "append" : false; // always append for root
36605         }
36606         var dragEl = n.ddel;
36607         var t = Roo.lib.Dom.getY(dragEl), b = t + dragEl.offsetHeight;
36608         var y = Roo.lib.Event.getPageY(e);
36609         //var noAppend = tn.allowChildren === false || tn.isLeaf();
36610         
36611         // we may drop nodes anywhere, as long as allowChildren has not been set to false..
36612         var noAppend = tn.allowChildren === false;
36613         if(this.appendOnly || tn.parentNode.allowChildren === false){
36614             return noAppend ? false : "append";
36615         }
36616         var noBelow = false;
36617         if(!this.allowParentInsert){
36618             noBelow = tn.hasChildNodes() && tn.isExpanded();
36619         }
36620         var q = (b - t) / (noAppend ? 2 : 3);
36621         if(y >= t && y < (t + q)){
36622             return "above";
36623         }else if(!noBelow && (noAppend || y >= b-q && y <= b)){
36624             return "below";
36625         }else{
36626             return "append";
36627         }
36628     },
36629     
36630     onNodeEnter : function(n, dd, e, data)
36631     {
36632         this.cancelExpand();
36633     },
36634     
36635     onNodeOver : function(n, dd, e, data)
36636     {
36637        
36638         var pt = this.getDropPoint(e, n, dd);
36639         var node = n.node;
36640         
36641         // auto node expand check
36642         if(!this.expandProcId && pt == "append" && node.hasChildNodes() && !n.node.isExpanded()){
36643             this.queueExpand(node);
36644         }else if(pt != "append"){
36645             this.cancelExpand();
36646         }
36647         
36648         // set the insert point style on the target node
36649         var returnCls = this.dropNotAllowed;
36650         if(this.isValidDropPoint(n, pt, dd, e, data)){
36651            if(pt){
36652                var el = n.ddel;
36653                var cls;
36654                if(pt == "above"){
36655                    returnCls = n.node.isFirst() ? "x-tree-drop-ok-above" : "x-tree-drop-ok-between";
36656                    cls = "x-tree-drag-insert-above";
36657                }else if(pt == "below"){
36658                    returnCls = n.node.isLast() ? "x-tree-drop-ok-below" : "x-tree-drop-ok-between";
36659                    cls = "x-tree-drag-insert-below";
36660                }else{
36661                    returnCls = "x-tree-drop-ok-append";
36662                    cls = "x-tree-drag-append";
36663                }
36664                if(this.lastInsertClass != cls){
36665                    Roo.fly(el).replaceClass(this.lastInsertClass, cls);
36666                    this.lastInsertClass = cls;
36667                }
36668            }
36669        }
36670        return returnCls;
36671     },
36672     
36673     onNodeOut : function(n, dd, e, data){
36674         
36675         this.cancelExpand();
36676         this.removeDropIndicators(n);
36677     },
36678     
36679     onNodeDrop : function(n, dd, e, data){
36680         var point = this.getDropPoint(e, n, dd);
36681         var targetNode = n.node;
36682         targetNode.ui.startDrop();
36683         if(!this.isValidDropPoint(n, point, dd, e, data)){
36684             targetNode.ui.endDrop();
36685             return false;
36686         }
36687         // first try to find the drop node
36688         var dropNode = data.node || (dd.getTreeNode ? dd.getTreeNode(data, targetNode, point, e) : null);
36689         var dropEvent = {
36690             tree : this.tree,
36691             target: targetNode,
36692             data: data,
36693             point: point,
36694             source: dd,
36695             rawEvent: e,
36696             dropNode: dropNode,
36697             cancel: !dropNode   
36698         };
36699         var retval = this.tree.fireEvent("beforenodedrop", dropEvent);
36700         if(retval === false || dropEvent.cancel === true || !dropEvent.dropNode){
36701             targetNode.ui.endDrop();
36702             return false;
36703         }
36704         // allow target changing
36705         targetNode = dropEvent.target;
36706         if(point == "append" && !targetNode.isExpanded()){
36707             targetNode.expand(false, null, function(){
36708                 this.completeDrop(dropEvent);
36709             }.createDelegate(this));
36710         }else{
36711             this.completeDrop(dropEvent);
36712         }
36713         return true;
36714     },
36715     
36716     completeDrop : function(de){
36717         var ns = de.dropNode, p = de.point, t = de.target;
36718         if(!(ns instanceof Array)){
36719             ns = [ns];
36720         }
36721         var n;
36722         for(var i = 0, len = ns.length; i < len; i++){
36723             n = ns[i];
36724             if(p == "above"){
36725                 t.parentNode.insertBefore(n, t);
36726             }else if(p == "below"){
36727                 t.parentNode.insertBefore(n, t.nextSibling);
36728             }else{
36729                 t.appendChild(n);
36730             }
36731         }
36732         n.ui.focus();
36733         if(this.tree.hlDrop){
36734             n.ui.highlight();
36735         }
36736         t.ui.endDrop();
36737         this.tree.fireEvent("nodedrop", de);
36738     },
36739     
36740     afterNodeMoved : function(dd, data, e, targetNode, dropNode){
36741         if(this.tree.hlDrop){
36742             dropNode.ui.focus();
36743             dropNode.ui.highlight();
36744         }
36745         this.tree.fireEvent("nodedrop", this.tree, targetNode, data, dd, e);
36746     },
36747     
36748     getTree : function(){
36749         return this.tree;
36750     },
36751     
36752     removeDropIndicators : function(n){
36753         if(n && n.ddel){
36754             var el = n.ddel;
36755             Roo.fly(el).removeClass([
36756                     "x-tree-drag-insert-above",
36757                     "x-tree-drag-insert-below",
36758                     "x-tree-drag-append"]);
36759             this.lastInsertClass = "_noclass";
36760         }
36761     },
36762     
36763     beforeDragDrop : function(target, e, id){
36764         this.cancelExpand();
36765         return true;
36766     },
36767     
36768     afterRepair : function(data){
36769         if(data && Roo.enableFx){
36770             data.node.ui.highlight();
36771         }
36772         this.hideProxy();
36773     } 
36774     
36775 });
36776
36777 }
36778 /*
36779  * Based on:
36780  * Ext JS Library 1.1.1
36781  * Copyright(c) 2006-2007, Ext JS, LLC.
36782  *
36783  * Originally Released Under LGPL - original licence link has changed is not relivant.
36784  *
36785  * Fork - LGPL
36786  * <script type="text/javascript">
36787  */
36788  
36789
36790 if(Roo.dd.DragZone){
36791 Roo.tree.TreeDragZone = function(tree, config){
36792     Roo.tree.TreeDragZone.superclass.constructor.call(this, tree.getTreeEl(), config);
36793     this.tree = tree;
36794 };
36795
36796 Roo.extend(Roo.tree.TreeDragZone, Roo.dd.DragZone, {
36797     ddGroup : "TreeDD",
36798    
36799     onBeforeDrag : function(data, e){
36800         var n = data.node;
36801         return n && n.draggable && !n.disabled;
36802     },
36803      
36804     
36805     onInitDrag : function(e){
36806         var data = this.dragData;
36807         this.tree.getSelectionModel().select(data.node);
36808         this.proxy.update("");
36809         data.node.ui.appendDDGhost(this.proxy.ghost.dom);
36810         this.tree.fireEvent("startdrag", this.tree, data.node, e);
36811     },
36812     
36813     getRepairXY : function(e, data){
36814         return data.node.ui.getDDRepairXY();
36815     },
36816     
36817     onEndDrag : function(data, e){
36818         this.tree.fireEvent("enddrag", this.tree, data.node, e);
36819         
36820         
36821     },
36822     
36823     onValidDrop : function(dd, e, id){
36824         this.tree.fireEvent("dragdrop", this.tree, this.dragData.node, dd, e);
36825         this.hideProxy();
36826     },
36827     
36828     beforeInvalidDrop : function(e, id){
36829         // this scrolls the original position back into view
36830         var sm = this.tree.getSelectionModel();
36831         sm.clearSelections();
36832         sm.select(this.dragData.node);
36833     }
36834 });
36835 }/*
36836  * Based on:
36837  * Ext JS Library 1.1.1
36838  * Copyright(c) 2006-2007, Ext JS, LLC.
36839  *
36840  * Originally Released Under LGPL - original licence link has changed is not relivant.
36841  *
36842  * Fork - LGPL
36843  * <script type="text/javascript">
36844  */
36845 /**
36846  * @class Roo.tree.TreeEditor
36847  * @extends Roo.Editor
36848  * Provides editor functionality for inline tree node editing.  Any valid {@link Roo.form.Field} can be used
36849  * as the editor field.
36850  * @constructor
36851  * @param {Object} config (used to be the tree panel.)
36852  * @param {Object} oldconfig DEPRECIATED Either a prebuilt {@link Roo.form.Field} instance or a Field config object
36853  * 
36854  * @cfg {Roo.tree.TreePanel} tree The tree to bind to.
36855  * @cfg {Roo.form.TextField|Object} field The field configuration
36856  *
36857  * 
36858  */
36859 Roo.tree.TreeEditor = function(config, oldconfig) { // was -- (tree, config){
36860     var tree = config;
36861     var field;
36862     if (oldconfig) { // old style..
36863         field = oldconfig.events ? oldconfig : new Roo.form.TextField(oldconfig);
36864     } else {
36865         // new style..
36866         tree = config.tree;
36867         config.field = config.field  || {};
36868         config.field.xtype = 'TextField';
36869         field = Roo.factory(config.field, Roo.form);
36870     }
36871     config = config || {};
36872     
36873     
36874     this.addEvents({
36875         /**
36876          * @event beforenodeedit
36877          * Fires when editing is initiated, but before the value changes.  Editing can be canceled by returning
36878          * false from the handler of this event.
36879          * @param {Editor} this
36880          * @param {Roo.tree.Node} node 
36881          */
36882         "beforenodeedit" : true
36883     });
36884     
36885     //Roo.log(config);
36886     Roo.tree.TreeEditor.superclass.constructor.call(this, field, config);
36887
36888     this.tree = tree;
36889
36890     tree.on('beforeclick', this.beforeNodeClick, this);
36891     tree.getTreeEl().on('mousedown', this.hide, this);
36892     this.on('complete', this.updateNode, this);
36893     this.on('beforestartedit', this.fitToTree, this);
36894     this.on('startedit', this.bindScroll, this, {delay:10});
36895     this.on('specialkey', this.onSpecialKey, this);
36896 };
36897
36898 Roo.extend(Roo.tree.TreeEditor, Roo.Editor, {
36899     /**
36900      * @cfg {String} alignment
36901      * The position to align to (see {@link Roo.Element#alignTo} for more details, defaults to "l-l").
36902      */
36903     alignment: "l-l",
36904     // inherit
36905     autoSize: false,
36906     /**
36907      * @cfg {Boolean} hideEl
36908      * True to hide the bound element while the editor is displayed (defaults to false)
36909      */
36910     hideEl : false,
36911     /**
36912      * @cfg {String} cls
36913      * CSS class to apply to the editor (defaults to "x-small-editor x-tree-editor")
36914      */
36915     cls: "x-small-editor x-tree-editor",
36916     /**
36917      * @cfg {Boolean} shim
36918      * True to shim the editor if selects/iframes could be displayed beneath it (defaults to false)
36919      */
36920     shim:false,
36921     // inherit
36922     shadow:"frame",
36923     /**
36924      * @cfg {Number} maxWidth
36925      * The maximum width in pixels of the editor field (defaults to 250).  Note that if the maxWidth would exceed
36926      * the containing tree element's size, it will be automatically limited for you to the container width, taking
36927      * scroll and client offsets into account prior to each edit.
36928      */
36929     maxWidth: 250,
36930
36931     editDelay : 350,
36932
36933     // private
36934     fitToTree : function(ed, el){
36935         var td = this.tree.getTreeEl().dom, nd = el.dom;
36936         if(td.scrollLeft >  nd.offsetLeft){ // ensure the node left point is visible
36937             td.scrollLeft = nd.offsetLeft;
36938         }
36939         var w = Math.min(
36940                 this.maxWidth,
36941                 (td.clientWidth > 20 ? td.clientWidth : td.offsetWidth) - Math.max(0, nd.offsetLeft-td.scrollLeft) - /*cushion*/5);
36942         this.setSize(w, '');
36943         
36944         return this.fireEvent('beforenodeedit', this, this.editNode);
36945         
36946     },
36947
36948     // private
36949     triggerEdit : function(node){
36950         this.completeEdit();
36951         this.editNode = node;
36952         this.startEdit(node.ui.textNode, node.text);
36953     },
36954
36955     // private
36956     bindScroll : function(){
36957         this.tree.getTreeEl().on('scroll', this.cancelEdit, this);
36958     },
36959
36960     // private
36961     beforeNodeClick : function(node, e){
36962         var sinceLast = (this.lastClick ? this.lastClick.getElapsed() : 0);
36963         this.lastClick = new Date();
36964         if(sinceLast > this.editDelay && this.tree.getSelectionModel().isSelected(node)){
36965             e.stopEvent();
36966             this.triggerEdit(node);
36967             return false;
36968         }
36969         return true;
36970     },
36971
36972     // private
36973     updateNode : function(ed, value){
36974         this.tree.getTreeEl().un('scroll', this.cancelEdit, this);
36975         this.editNode.setText(value);
36976     },
36977
36978     // private
36979     onHide : function(){
36980         Roo.tree.TreeEditor.superclass.onHide.call(this);
36981         if(this.editNode){
36982             this.editNode.ui.focus();
36983         }
36984     },
36985
36986     // private
36987     onSpecialKey : function(field, e){
36988         var k = e.getKey();
36989         if(k == e.ESC){
36990             e.stopEvent();
36991             this.cancelEdit();
36992         }else if(k == e.ENTER && !e.hasModifier()){
36993             e.stopEvent();
36994             this.completeEdit();
36995         }
36996     }
36997 });//<Script type="text/javascript">
36998 /*
36999  * Based on:
37000  * Ext JS Library 1.1.1
37001  * Copyright(c) 2006-2007, Ext JS, LLC.
37002  *
37003  * Originally Released Under LGPL - original licence link has changed is not relivant.
37004  *
37005  * Fork - LGPL
37006  * <script type="text/javascript">
37007  */
37008  
37009 /**
37010  * Not documented??? - probably should be...
37011  */
37012
37013 Roo.tree.ColumnNodeUI = Roo.extend(Roo.tree.TreeNodeUI, {
37014     //focus: Roo.emptyFn, // prevent odd scrolling behavior
37015     
37016     renderElements : function(n, a, targetNode, bulkRender){
37017         //consel.log("renderElements?");
37018         this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
37019
37020         var t = n.getOwnerTree();
37021         var tid = Pman.Tab.Document_TypesTree.tree.el.id;
37022         
37023         var cols = t.columns;
37024         var bw = t.borderWidth;
37025         var c = cols[0];
37026         var href = a.href ? a.href : Roo.isGecko ? "" : "#";
37027          var cb = typeof a.checked == "boolean";
37028         var tx = String.format('{0}',n.text || (c.renderer ? c.renderer(a[c.dataIndex], n, a) : a[c.dataIndex]));
37029         var colcls = 'x-t-' + tid + '-c0';
37030         var buf = [
37031             '<li class="x-tree-node">',
37032             
37033                 
37034                 '<div class="x-tree-node-el ', a.cls,'">',
37035                     // extran...
37036                     '<div class="x-tree-col ', colcls, '" style="width:', c.width-bw, 'px;">',
37037                 
37038                 
37039                         '<span class="x-tree-node-indent">',this.indentMarkup,'</span>',
37040                         '<img src="', this.emptyIcon, '" class="x-tree-ec-icon  " />',
37041                         '<img src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',
37042                            (a.icon ? ' x-tree-node-inline-icon' : ''),
37043                            (a.iconCls ? ' '+a.iconCls : ''),
37044                            '" unselectable="on" />',
37045                         (cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + 
37046                              (a.checked ? 'checked="checked" />' : ' />')) : ''),
37047                              
37048                         '<a class="x-tree-node-anchor" hidefocus="on" href="',href,'" tabIndex="1" ',
37049                             (a.hrefTarget ? ' target="' +a.hrefTarget + '"' : ''), '>',
37050                             '<span unselectable="on" qtip="' + tx + '">',
37051                              tx,
37052                              '</span></a>' ,
37053                     '</div>',
37054                      '<a class="x-tree-node-anchor" hidefocus="on" href="',href,'" tabIndex="1" ',
37055                             (a.hrefTarget ? ' target="' +a.hrefTarget + '"' : ''), '>'
37056                  ];
37057         for(var i = 1, len = cols.length; i < len; i++){
37058             c = cols[i];
37059             colcls = 'x-t-' + tid + '-c' +i;
37060             tx = String.format('{0}', (c.renderer ? c.renderer(a[c.dataIndex], n, a) : a[c.dataIndex]));
37061             buf.push('<div class="x-tree-col ', colcls, ' ' ,(c.cls?c.cls:''),'" style="width:',c.width-bw,'px;">',
37062                         '<div class="x-tree-col-text" qtip="' + tx +'">',tx,"</div>",
37063                       "</div>");
37064          }
37065          
37066          buf.push(
37067             '</a>',
37068             '<div class="x-clear"></div></div>',
37069             '<ul class="x-tree-node-ct" style="display:none;"></ul>',
37070             "</li>");
37071         
37072         if(bulkRender !== true && n.nextSibling && n.nextSibling.ui.getEl()){
37073             this.wrap = Roo.DomHelper.insertHtml("beforeBegin",
37074                                 n.nextSibling.ui.getEl(), buf.join(""));
37075         }else{
37076             this.wrap = Roo.DomHelper.insertHtml("beforeEnd", targetNode, buf.join(""));
37077         }
37078         var el = this.wrap.firstChild;
37079         this.elRow = el;
37080         this.elNode = el.firstChild;
37081         this.ranchor = el.childNodes[1];
37082         this.ctNode = this.wrap.childNodes[1];
37083         var cs = el.firstChild.childNodes;
37084         this.indentNode = cs[0];
37085         this.ecNode = cs[1];
37086         this.iconNode = cs[2];
37087         var index = 3;
37088         if(cb){
37089             this.checkbox = cs[3];
37090             index++;
37091         }
37092         this.anchor = cs[index];
37093         
37094         this.textNode = cs[index].firstChild;
37095         
37096         //el.on("click", this.onClick, this);
37097         //el.on("dblclick", this.onDblClick, this);
37098         
37099         
37100        // console.log(this);
37101     },
37102     initEvents : function(){
37103         Roo.tree.ColumnNodeUI.superclass.initEvents.call(this);
37104         
37105             
37106         var a = this.ranchor;
37107
37108         var el = Roo.get(a);
37109
37110         if(Roo.isOpera){ // opera render bug ignores the CSS
37111             el.setStyle("text-decoration", "none");
37112         }
37113
37114         el.on("click", this.onClick, this);
37115         el.on("dblclick", this.onDblClick, this);
37116         el.on("contextmenu", this.onContextMenu, this);
37117         
37118     },
37119     
37120     /*onSelectedChange : function(state){
37121         if(state){
37122             this.focus();
37123             this.addClass("x-tree-selected");
37124         }else{
37125             //this.blur();
37126             this.removeClass("x-tree-selected");
37127         }
37128     },*/
37129     addClass : function(cls){
37130         if(this.elRow){
37131             Roo.fly(this.elRow).addClass(cls);
37132         }
37133         
37134     },
37135     
37136     
37137     removeClass : function(cls){
37138         if(this.elRow){
37139             Roo.fly(this.elRow).removeClass(cls);
37140         }
37141     }
37142
37143     
37144     
37145 });//<Script type="text/javascript">
37146
37147 /*
37148  * Based on:
37149  * Ext JS Library 1.1.1
37150  * Copyright(c) 2006-2007, Ext JS, LLC.
37151  *
37152  * Originally Released Under LGPL - original licence link has changed is not relivant.
37153  *
37154  * Fork - LGPL
37155  * <script type="text/javascript">
37156  */
37157  
37158
37159 /**
37160  * @class Roo.tree.ColumnTree
37161  * @extends Roo.data.TreePanel
37162  * @cfg {Object} columns  Including width, header, renderer, cls, dataIndex 
37163  * @cfg {int} borderWidth  compined right/left border allowance
37164  * @constructor
37165  * @param {String/HTMLElement/Element} el The container element
37166  * @param {Object} config
37167  */
37168 Roo.tree.ColumnTree =  function(el, config)
37169 {
37170    Roo.tree.ColumnTree.superclass.constructor.call(this, el , config);
37171    this.addEvents({
37172         /**
37173         * @event resize
37174         * Fire this event on a container when it resizes
37175         * @param {int} w Width
37176         * @param {int} h Height
37177         */
37178        "resize" : true
37179     });
37180     this.on('resize', this.onResize, this);
37181 };
37182
37183 Roo.extend(Roo.tree.ColumnTree, Roo.tree.TreePanel, {
37184     //lines:false,
37185     
37186     
37187     borderWidth: Roo.isBorderBox ? 0 : 2, 
37188     headEls : false,
37189     
37190     render : function(){
37191         // add the header.....
37192        
37193         Roo.tree.ColumnTree.superclass.render.apply(this);
37194         
37195         this.el.addClass('x-column-tree');
37196         
37197         this.headers = this.el.createChild(
37198             {cls:'x-tree-headers'},this.innerCt.dom);
37199    
37200         var cols = this.columns, c;
37201         var totalWidth = 0;
37202         this.headEls = [];
37203         var  len = cols.length;
37204         for(var i = 0; i < len; i++){
37205              c = cols[i];
37206              totalWidth += c.width;
37207             this.headEls.push(this.headers.createChild({
37208                  cls:'x-tree-hd ' + (c.cls?c.cls+'-hd':''),
37209                  cn: {
37210                      cls:'x-tree-hd-text',
37211                      html: c.header
37212                  },
37213                  style:'width:'+(c.width-this.borderWidth)+'px;'
37214              }));
37215         }
37216         this.headers.createChild({cls:'x-clear'});
37217         // prevent floats from wrapping when clipped
37218         this.headers.setWidth(totalWidth);
37219         //this.innerCt.setWidth(totalWidth);
37220         this.innerCt.setStyle({ overflow: 'auto' });
37221         this.onResize(this.width, this.height);
37222              
37223         
37224     },
37225     onResize : function(w,h)
37226     {
37227         this.height = h;
37228         this.width = w;
37229         // resize cols..
37230         this.innerCt.setWidth(this.width);
37231         this.innerCt.setHeight(this.height-20);
37232         
37233         // headers...
37234         var cols = this.columns, c;
37235         var totalWidth = 0;
37236         var expEl = false;
37237         var len = cols.length;
37238         for(var i = 0; i < len; i++){
37239             c = cols[i];
37240             if (this.autoExpandColumn !== false && c.dataIndex == this.autoExpandColumn) {
37241                 // it's the expander..
37242                 expEl  = this.headEls[i];
37243                 continue;
37244             }
37245             totalWidth += c.width;
37246             
37247         }
37248         if (expEl) {
37249             expEl.setWidth(  ((w - totalWidth)-this.borderWidth - 20));
37250         }
37251         this.headers.setWidth(w-20);
37252
37253         
37254         
37255         
37256     }
37257 });
37258 /*
37259  * Based on:
37260  * Ext JS Library 1.1.1
37261  * Copyright(c) 2006-2007, Ext JS, LLC.
37262  *
37263  * Originally Released Under LGPL - original licence link has changed is not relivant.
37264  *
37265  * Fork - LGPL
37266  * <script type="text/javascript">
37267  */
37268  
37269 /**
37270  * @class Roo.menu.Menu
37271  * @extends Roo.util.Observable
37272  * A menu object.  This is the container to which you add all other menu items.  Menu can also serve a as a base class
37273  * when you want a specialzed menu based off of another component (like {@link Roo.menu.DateMenu} for example).
37274  * @constructor
37275  * Creates a new Menu
37276  * @param {Object} config Configuration options
37277  */
37278 Roo.menu.Menu = function(config){
37279     
37280     Roo.menu.Menu.superclass.constructor.call(this, config);
37281     
37282     this.id = this.id || Roo.id();
37283     this.addEvents({
37284         /**
37285          * @event beforeshow
37286          * Fires before this menu is displayed
37287          * @param {Roo.menu.Menu} this
37288          */
37289         beforeshow : true,
37290         /**
37291          * @event beforehide
37292          * Fires before this menu is hidden
37293          * @param {Roo.menu.Menu} this
37294          */
37295         beforehide : true,
37296         /**
37297          * @event show
37298          * Fires after this menu is displayed
37299          * @param {Roo.menu.Menu} this
37300          */
37301         show : true,
37302         /**
37303          * @event hide
37304          * Fires after this menu is hidden
37305          * @param {Roo.menu.Menu} this
37306          */
37307         hide : true,
37308         /**
37309          * @event click
37310          * Fires when this menu is clicked (or when the enter key is pressed while it is active)
37311          * @param {Roo.menu.Menu} this
37312          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37313          * @param {Roo.EventObject} e
37314          */
37315         click : true,
37316         /**
37317          * @event mouseover
37318          * Fires when the mouse is hovering over this menu
37319          * @param {Roo.menu.Menu} this
37320          * @param {Roo.EventObject} e
37321          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37322          */
37323         mouseover : true,
37324         /**
37325          * @event mouseout
37326          * Fires when the mouse exits this menu
37327          * @param {Roo.menu.Menu} this
37328          * @param {Roo.EventObject} e
37329          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37330          */
37331         mouseout : true,
37332         /**
37333          * @event itemclick
37334          * Fires when a menu item contained in this menu is clicked
37335          * @param {Roo.menu.BaseItem} baseItem The BaseItem that was clicked
37336          * @param {Roo.EventObject} e
37337          */
37338         itemclick: true
37339     });
37340     if (this.registerMenu) {
37341         Roo.menu.MenuMgr.register(this);
37342     }
37343     
37344     var mis = this.items;
37345     this.items = new Roo.util.MixedCollection();
37346     if(mis){
37347         this.add.apply(this, mis);
37348     }
37349 };
37350
37351 Roo.extend(Roo.menu.Menu, Roo.util.Observable, {
37352     /**
37353      * @cfg {Number} minWidth The minimum width of the menu in pixels (defaults to 120)
37354      */
37355     minWidth : 120,
37356     /**
37357      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop"
37358      * for bottom-right shadow (defaults to "sides")
37359      */
37360     shadow : "sides",
37361     /**
37362      * @cfg {String} subMenuAlign The {@link Roo.Element#alignTo} anchor position value to use for submenus of
37363      * this menu (defaults to "tl-tr?")
37364      */
37365     subMenuAlign : "tl-tr?",
37366     /**
37367      * @cfg {String} defaultAlign The default {@link Roo.Element#alignTo) anchor position value for this menu
37368      * relative to its element of origin (defaults to "tl-bl?")
37369      */
37370     defaultAlign : "tl-bl?",
37371     /**
37372      * @cfg {Boolean} allowOtherMenus True to allow multiple menus to be displayed at the same time (defaults to false)
37373      */
37374     allowOtherMenus : false,
37375     /**
37376      * @cfg {Boolean} registerMenu True (default) - means that clicking on screen etc. hides it.
37377      */
37378     registerMenu : true,
37379
37380     hidden:true,
37381
37382     // private
37383     render : function(){
37384         if(this.el){
37385             return;
37386         }
37387         var el = this.el = new Roo.Layer({
37388             cls: "x-menu",
37389             shadow:this.shadow,
37390             constrain: false,
37391             parentEl: this.parentEl || document.body,
37392             zindex:15000
37393         });
37394
37395         this.keyNav = new Roo.menu.MenuNav(this);
37396
37397         if(this.plain){
37398             el.addClass("x-menu-plain");
37399         }
37400         if(this.cls){
37401             el.addClass(this.cls);
37402         }
37403         // generic focus element
37404         this.focusEl = el.createChild({
37405             tag: "a", cls: "x-menu-focus", href: "#", onclick: "return false;", tabIndex:"-1"
37406         });
37407         var ul = el.createChild({tag: "ul", cls: "x-menu-list"});
37408         //disabling touch- as it's causing issues ..
37409         //ul.on(Roo.isTouch ? 'touchstart' : 'click'   , this.onClick, this);
37410         ul.on('click'   , this.onClick, this);
37411         
37412         
37413         ul.on("mouseover", this.onMouseOver, this);
37414         ul.on("mouseout", this.onMouseOut, this);
37415         this.items.each(function(item){
37416             if (item.hidden) {
37417                 return;
37418             }
37419             
37420             var li = document.createElement("li");
37421             li.className = "x-menu-list-item";
37422             ul.dom.appendChild(li);
37423             item.render(li, this);
37424         }, this);
37425         this.ul = ul;
37426         this.autoWidth();
37427     },
37428
37429     // private
37430     autoWidth : function(){
37431         var el = this.el, ul = this.ul;
37432         if(!el){
37433             return;
37434         }
37435         var w = this.width;
37436         if(w){
37437             el.setWidth(w);
37438         }else if(Roo.isIE){
37439             el.setWidth(this.minWidth);
37440             var t = el.dom.offsetWidth; // force recalc
37441             el.setWidth(ul.getWidth()+el.getFrameWidth("lr"));
37442         }
37443     },
37444
37445     // private
37446     delayAutoWidth : function(){
37447         if(this.rendered){
37448             if(!this.awTask){
37449                 this.awTask = new Roo.util.DelayedTask(this.autoWidth, this);
37450             }
37451             this.awTask.delay(20);
37452         }
37453     },
37454
37455     // private
37456     findTargetItem : function(e){
37457         var t = e.getTarget(".x-menu-list-item", this.ul,  true);
37458         if(t && t.menuItemId){
37459             return this.items.get(t.menuItemId);
37460         }
37461     },
37462
37463     // private
37464     onClick : function(e){
37465         Roo.log("menu.onClick");
37466         var t = this.findTargetItem(e);
37467         if(!t){
37468             return;
37469         }
37470         Roo.log(e);
37471         if (Roo.isTouch && e.type == 'touchstart' && t.menu  && !t.disabled) {
37472             if(t == this.activeItem && t.shouldDeactivate(e)){
37473                 this.activeItem.deactivate();
37474                 delete this.activeItem;
37475                 return;
37476             }
37477             if(t.canActivate){
37478                 this.setActiveItem(t, true);
37479             }
37480             return;
37481             
37482             
37483         }
37484         
37485         t.onClick(e);
37486         this.fireEvent("click", this, t, e);
37487     },
37488
37489     // private
37490     setActiveItem : function(item, autoExpand){
37491         if(item != this.activeItem){
37492             if(this.activeItem){
37493                 this.activeItem.deactivate();
37494             }
37495             this.activeItem = item;
37496             item.activate(autoExpand);
37497         }else if(autoExpand){
37498             item.expandMenu();
37499         }
37500     },
37501
37502     // private
37503     tryActivate : function(start, step){
37504         var items = this.items;
37505         for(var i = start, len = items.length; i >= 0 && i < len; i+= step){
37506             var item = items.get(i);
37507             if(!item.disabled && item.canActivate){
37508                 this.setActiveItem(item, false);
37509                 return item;
37510             }
37511         }
37512         return false;
37513     },
37514
37515     // private
37516     onMouseOver : function(e){
37517         var t;
37518         if(t = this.findTargetItem(e)){
37519             if(t.canActivate && !t.disabled){
37520                 this.setActiveItem(t, true);
37521             }
37522         }
37523         this.fireEvent("mouseover", this, e, t);
37524     },
37525
37526     // private
37527     onMouseOut : function(e){
37528         var t;
37529         if(t = this.findTargetItem(e)){
37530             if(t == this.activeItem && t.shouldDeactivate(e)){
37531                 this.activeItem.deactivate();
37532                 delete this.activeItem;
37533             }
37534         }
37535         this.fireEvent("mouseout", this, e, t);
37536     },
37537
37538     /**
37539      * Read-only.  Returns true if the menu is currently displayed, else false.
37540      * @type Boolean
37541      */
37542     isVisible : function(){
37543         return this.el && !this.hidden;
37544     },
37545
37546     /**
37547      * Displays this menu relative to another element
37548      * @param {String/HTMLElement/Roo.Element} element The element to align to
37549      * @param {String} position (optional) The {@link Roo.Element#alignTo} anchor position to use in aligning to
37550      * the element (defaults to this.defaultAlign)
37551      * @param {Roo.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
37552      */
37553     show : function(el, pos, parentMenu){
37554         this.parentMenu = parentMenu;
37555         if(!this.el){
37556             this.render();
37557         }
37558         this.fireEvent("beforeshow", this);
37559         this.showAt(this.el.getAlignToXY(el, pos || this.defaultAlign), parentMenu, false);
37560     },
37561
37562     /**
37563      * Displays this menu at a specific xy position
37564      * @param {Array} xyPosition Contains X & Y [x, y] values for the position at which to show the menu (coordinates are page-based)
37565      * @param {Roo.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
37566      */
37567     showAt : function(xy, parentMenu, /* private: */_e){
37568         this.parentMenu = parentMenu;
37569         if(!this.el){
37570             this.render();
37571         }
37572         if(_e !== false){
37573             this.fireEvent("beforeshow", this);
37574             xy = this.el.adjustForConstraints(xy);
37575         }
37576         this.el.setXY(xy);
37577         this.el.show();
37578         this.hidden = false;
37579         this.focus();
37580         this.fireEvent("show", this);
37581     },
37582
37583     focus : function(){
37584         if(!this.hidden){
37585             this.doFocus.defer(50, this);
37586         }
37587     },
37588
37589     doFocus : function(){
37590         if(!this.hidden){
37591             this.focusEl.focus();
37592         }
37593     },
37594
37595     /**
37596      * Hides this menu and optionally all parent menus
37597      * @param {Boolean} deep (optional) True to hide all parent menus recursively, if any (defaults to false)
37598      */
37599     hide : function(deep){
37600         if(this.el && this.isVisible()){
37601             this.fireEvent("beforehide", this);
37602             if(this.activeItem){
37603                 this.activeItem.deactivate();
37604                 this.activeItem = null;
37605             }
37606             this.el.hide();
37607             this.hidden = true;
37608             this.fireEvent("hide", this);
37609         }
37610         if(deep === true && this.parentMenu){
37611             this.parentMenu.hide(true);
37612         }
37613     },
37614
37615     /**
37616      * Addds one or more items of any type supported by the Menu class, or that can be converted into menu items.
37617      * Any of the following are valid:
37618      * <ul>
37619      * <li>Any menu item object based on {@link Roo.menu.Item}</li>
37620      * <li>An HTMLElement object which will be converted to a menu item</li>
37621      * <li>A menu item config object that will be created as a new menu item</li>
37622      * <li>A string, which can either be '-' or 'separator' to add a menu separator, otherwise
37623      * it will be converted into a {@link Roo.menu.TextItem} and added</li>
37624      * </ul>
37625      * Usage:
37626      * <pre><code>
37627 // Create the menu
37628 var menu = new Roo.menu.Menu();
37629
37630 // Create a menu item to add by reference
37631 var menuItem = new Roo.menu.Item({ text: 'New Item!' });
37632
37633 // Add a bunch of items at once using different methods.
37634 // Only the last item added will be returned.
37635 var item = menu.add(
37636     menuItem,                // add existing item by ref
37637     'Dynamic Item',          // new TextItem
37638     '-',                     // new separator
37639     { text: 'Config Item' }  // new item by config
37640 );
37641 </code></pre>
37642      * @param {Mixed} args One or more menu items, menu item configs or other objects that can be converted to menu items
37643      * @return {Roo.menu.Item} The menu item that was added, or the last one if multiple items were added
37644      */
37645     add : function(){
37646         var a = arguments, l = a.length, item;
37647         for(var i = 0; i < l; i++){
37648             var el = a[i];
37649             if ((typeof(el) == "object") && el.xtype && el.xns) {
37650                 el = Roo.factory(el, Roo.menu);
37651             }
37652             
37653             if(el.render){ // some kind of Item
37654                 item = this.addItem(el);
37655             }else if(typeof el == "string"){ // string
37656                 if(el == "separator" || el == "-"){
37657                     item = this.addSeparator();
37658                 }else{
37659                     item = this.addText(el);
37660                 }
37661             }else if(el.tagName || el.el){ // element
37662                 item = this.addElement(el);
37663             }else if(typeof el == "object"){ // must be menu item config?
37664                 item = this.addMenuItem(el);
37665             }
37666         }
37667         return item;
37668     },
37669
37670     /**
37671      * Returns this menu's underlying {@link Roo.Element} object
37672      * @return {Roo.Element} The element
37673      */
37674     getEl : function(){
37675         if(!this.el){
37676             this.render();
37677         }
37678         return this.el;
37679     },
37680
37681     /**
37682      * Adds a separator bar to the menu
37683      * @return {Roo.menu.Item} The menu item that was added
37684      */
37685     addSeparator : function(){
37686         return this.addItem(new Roo.menu.Separator());
37687     },
37688
37689     /**
37690      * Adds an {@link Roo.Element} object to the menu
37691      * @param {String/HTMLElement/Roo.Element} el The element or DOM node to add, or its id
37692      * @return {Roo.menu.Item} The menu item that was added
37693      */
37694     addElement : function(el){
37695         return this.addItem(new Roo.menu.BaseItem(el));
37696     },
37697
37698     /**
37699      * Adds an existing object based on {@link Roo.menu.Item} to the menu
37700      * @param {Roo.menu.Item} item The menu item to add
37701      * @return {Roo.menu.Item} The menu item that was added
37702      */
37703     addItem : function(item){
37704         this.items.add(item);
37705         if(this.ul){
37706             var li = document.createElement("li");
37707             li.className = "x-menu-list-item";
37708             this.ul.dom.appendChild(li);
37709             item.render(li, this);
37710             this.delayAutoWidth();
37711         }
37712         return item;
37713     },
37714
37715     /**
37716      * Creates a new {@link Roo.menu.Item} based an the supplied config object and adds it to the menu
37717      * @param {Object} config A MenuItem config object
37718      * @return {Roo.menu.Item} The menu item that was added
37719      */
37720     addMenuItem : function(config){
37721         if(!(config instanceof Roo.menu.Item)){
37722             if(typeof config.checked == "boolean"){ // must be check menu item config?
37723                 config = new Roo.menu.CheckItem(config);
37724             }else{
37725                 config = new Roo.menu.Item(config);
37726             }
37727         }
37728         return this.addItem(config);
37729     },
37730
37731     /**
37732      * Creates a new {@link Roo.menu.TextItem} with the supplied text and adds it to the menu
37733      * @param {String} text The text to display in the menu item
37734      * @return {Roo.menu.Item} The menu item that was added
37735      */
37736     addText : function(text){
37737         return this.addItem(new Roo.menu.TextItem({ text : text }));
37738     },
37739
37740     /**
37741      * Inserts an existing object based on {@link Roo.menu.Item} to the menu at a specified index
37742      * @param {Number} index The index in the menu's list of current items where the new item should be inserted
37743      * @param {Roo.menu.Item} item The menu item to add
37744      * @return {Roo.menu.Item} The menu item that was added
37745      */
37746     insert : function(index, item){
37747         this.items.insert(index, item);
37748         if(this.ul){
37749             var li = document.createElement("li");
37750             li.className = "x-menu-list-item";
37751             this.ul.dom.insertBefore(li, this.ul.dom.childNodes[index]);
37752             item.render(li, this);
37753             this.delayAutoWidth();
37754         }
37755         return item;
37756     },
37757
37758     /**
37759      * Removes an {@link Roo.menu.Item} from the menu and destroys the object
37760      * @param {Roo.menu.Item} item The menu item to remove
37761      */
37762     remove : function(item){
37763         this.items.removeKey(item.id);
37764         item.destroy();
37765     },
37766
37767     /**
37768      * Removes and destroys all items in the menu
37769      */
37770     removeAll : function(){
37771         var f;
37772         while(f = this.items.first()){
37773             this.remove(f);
37774         }
37775     }
37776 });
37777
37778 // MenuNav is a private utility class used internally by the Menu
37779 Roo.menu.MenuNav = function(menu){
37780     Roo.menu.MenuNav.superclass.constructor.call(this, menu.el);
37781     this.scope = this.menu = menu;
37782 };
37783
37784 Roo.extend(Roo.menu.MenuNav, Roo.KeyNav, {
37785     doRelay : function(e, h){
37786         var k = e.getKey();
37787         if(!this.menu.activeItem && e.isNavKeyPress() && k != e.SPACE && k != e.RETURN){
37788             this.menu.tryActivate(0, 1);
37789             return false;
37790         }
37791         return h.call(this.scope || this, e, this.menu);
37792     },
37793
37794     up : function(e, m){
37795         if(!m.tryActivate(m.items.indexOf(m.activeItem)-1, -1)){
37796             m.tryActivate(m.items.length-1, -1);
37797         }
37798     },
37799
37800     down : function(e, m){
37801         if(!m.tryActivate(m.items.indexOf(m.activeItem)+1, 1)){
37802             m.tryActivate(0, 1);
37803         }
37804     },
37805
37806     right : function(e, m){
37807         if(m.activeItem){
37808             m.activeItem.expandMenu(true);
37809         }
37810     },
37811
37812     left : function(e, m){
37813         m.hide();
37814         if(m.parentMenu && m.parentMenu.activeItem){
37815             m.parentMenu.activeItem.activate();
37816         }
37817     },
37818
37819     enter : function(e, m){
37820         if(m.activeItem){
37821             e.stopPropagation();
37822             m.activeItem.onClick(e);
37823             m.fireEvent("click", this, m.activeItem);
37824             return true;
37825         }
37826     }
37827 });/*
37828  * Based on:
37829  * Ext JS Library 1.1.1
37830  * Copyright(c) 2006-2007, Ext JS, LLC.
37831  *
37832  * Originally Released Under LGPL - original licence link has changed is not relivant.
37833  *
37834  * Fork - LGPL
37835  * <script type="text/javascript">
37836  */
37837  
37838 /**
37839  * @class Roo.menu.MenuMgr
37840  * Provides a common registry of all menu items on a page so that they can be easily accessed by id.
37841  * @singleton
37842  */
37843 Roo.menu.MenuMgr = function(){
37844    var menus, active, groups = {}, attached = false, lastShow = new Date();
37845
37846    // private - called when first menu is created
37847    function init(){
37848        menus = {};
37849        active = new Roo.util.MixedCollection();
37850        Roo.get(document).addKeyListener(27, function(){
37851            if(active.length > 0){
37852                hideAll();
37853            }
37854        });
37855    }
37856
37857    // private
37858    function hideAll(){
37859        if(active && active.length > 0){
37860            var c = active.clone();
37861            c.each(function(m){
37862                m.hide();
37863            });
37864        }
37865    }
37866
37867    // private
37868    function onHide(m){
37869        active.remove(m);
37870        if(active.length < 1){
37871            Roo.get(document).un("mousedown", onMouseDown);
37872            attached = false;
37873        }
37874    }
37875
37876    // private
37877    function onShow(m){
37878        var last = active.last();
37879        lastShow = new Date();
37880        active.add(m);
37881        if(!attached){
37882            Roo.get(document).on("mousedown", onMouseDown);
37883            attached = true;
37884        }
37885        if(m.parentMenu){
37886           m.getEl().setZIndex(parseInt(m.parentMenu.getEl().getStyle("z-index"), 10) + 3);
37887           m.parentMenu.activeChild = m;
37888        }else if(last && last.isVisible()){
37889           m.getEl().setZIndex(parseInt(last.getEl().getStyle("z-index"), 10) + 3);
37890        }
37891    }
37892
37893    // private
37894    function onBeforeHide(m){
37895        if(m.activeChild){
37896            m.activeChild.hide();
37897        }
37898        if(m.autoHideTimer){
37899            clearTimeout(m.autoHideTimer);
37900            delete m.autoHideTimer;
37901        }
37902    }
37903
37904    // private
37905    function onBeforeShow(m){
37906        var pm = m.parentMenu;
37907        if(!pm && !m.allowOtherMenus){
37908            hideAll();
37909        }else if(pm && pm.activeChild && active != m){
37910            pm.activeChild.hide();
37911        }
37912    }
37913
37914    // private
37915    function onMouseDown(e){
37916        if(lastShow.getElapsed() > 50 && active.length > 0 && !e.getTarget(".x-menu")){
37917            hideAll();
37918        }
37919    }
37920
37921    // private
37922    function onBeforeCheck(mi, state){
37923        if(state){
37924            var g = groups[mi.group];
37925            for(var i = 0, l = g.length; i < l; i++){
37926                if(g[i] != mi){
37927                    g[i].setChecked(false);
37928                }
37929            }
37930        }
37931    }
37932
37933    return {
37934
37935        /**
37936         * Hides all menus that are currently visible
37937         */
37938        hideAll : function(){
37939             hideAll();  
37940        },
37941
37942        // private
37943        register : function(menu){
37944            if(!menus){
37945                init();
37946            }
37947            menus[menu.id] = menu;
37948            menu.on("beforehide", onBeforeHide);
37949            menu.on("hide", onHide);
37950            menu.on("beforeshow", onBeforeShow);
37951            menu.on("show", onShow);
37952            var g = menu.group;
37953            if(g && menu.events["checkchange"]){
37954                if(!groups[g]){
37955                    groups[g] = [];
37956                }
37957                groups[g].push(menu);
37958                menu.on("checkchange", onCheck);
37959            }
37960        },
37961
37962         /**
37963          * Returns a {@link Roo.menu.Menu} object
37964          * @param {String/Object} menu The string menu id, an existing menu object reference, or a Menu config that will
37965          * be used to generate and return a new Menu instance.
37966          */
37967        get : function(menu){
37968            if(typeof menu == "string"){ // menu id
37969                return menus[menu];
37970            }else if(menu.events){  // menu instance
37971                return menu;
37972            }else if(typeof menu.length == 'number'){ // array of menu items?
37973                return new Roo.menu.Menu({items:menu});
37974            }else{ // otherwise, must be a config
37975                return new Roo.menu.Menu(menu);
37976            }
37977        },
37978
37979        // private
37980        unregister : function(menu){
37981            delete menus[menu.id];
37982            menu.un("beforehide", onBeforeHide);
37983            menu.un("hide", onHide);
37984            menu.un("beforeshow", onBeforeShow);
37985            menu.un("show", onShow);
37986            var g = menu.group;
37987            if(g && menu.events["checkchange"]){
37988                groups[g].remove(menu);
37989                menu.un("checkchange", onCheck);
37990            }
37991        },
37992
37993        // private
37994        registerCheckable : function(menuItem){
37995            var g = menuItem.group;
37996            if(g){
37997                if(!groups[g]){
37998                    groups[g] = [];
37999                }
38000                groups[g].push(menuItem);
38001                menuItem.on("beforecheckchange", onBeforeCheck);
38002            }
38003        },
38004
38005        // private
38006        unregisterCheckable : function(menuItem){
38007            var g = menuItem.group;
38008            if(g){
38009                groups[g].remove(menuItem);
38010                menuItem.un("beforecheckchange", onBeforeCheck);
38011            }
38012        }
38013    };
38014 }();/*
38015  * Based on:
38016  * Ext JS Library 1.1.1
38017  * Copyright(c) 2006-2007, Ext JS, LLC.
38018  *
38019  * Originally Released Under LGPL - original licence link has changed is not relivant.
38020  *
38021  * Fork - LGPL
38022  * <script type="text/javascript">
38023  */
38024  
38025
38026 /**
38027  * @class Roo.menu.BaseItem
38028  * @extends Roo.Component
38029  * The base class for all items that render into menus.  BaseItem provides default rendering, activated state
38030  * management and base configuration options shared by all menu components.
38031  * @constructor
38032  * Creates a new BaseItem
38033  * @param {Object} config Configuration options
38034  */
38035 Roo.menu.BaseItem = function(config){
38036     Roo.menu.BaseItem.superclass.constructor.call(this, config);
38037
38038     this.addEvents({
38039         /**
38040          * @event click
38041          * Fires when this item is clicked
38042          * @param {Roo.menu.BaseItem} this
38043          * @param {Roo.EventObject} e
38044          */
38045         click: true,
38046         /**
38047          * @event activate
38048          * Fires when this item is activated
38049          * @param {Roo.menu.BaseItem} this
38050          */
38051         activate : true,
38052         /**
38053          * @event deactivate
38054          * Fires when this item is deactivated
38055          * @param {Roo.menu.BaseItem} this
38056          */
38057         deactivate : true
38058     });
38059
38060     if(this.handler){
38061         this.on("click", this.handler, this.scope, true);
38062     }
38063 };
38064
38065 Roo.extend(Roo.menu.BaseItem, Roo.Component, {
38066     /**
38067      * @cfg {Function} handler
38068      * A function that will handle the click event of this menu item (defaults to undefined)
38069      */
38070     /**
38071      * @cfg {Boolean} canActivate True if this item can be visually activated (defaults to false)
38072      */
38073     canActivate : false,
38074     
38075      /**
38076      * @cfg {Boolean} hidden True to prevent creation of this menu item (defaults to false)
38077      */
38078     hidden: false,
38079     
38080     /**
38081      * @cfg {String} activeClass The CSS class to use when the item becomes activated (defaults to "x-menu-item-active")
38082      */
38083     activeClass : "x-menu-item-active",
38084     /**
38085      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to true)
38086      */
38087     hideOnClick : true,
38088     /**
38089      * @cfg {Number} hideDelay Length of time in milliseconds to wait before hiding after a click (defaults to 100)
38090      */
38091     hideDelay : 100,
38092
38093     // private
38094     ctype: "Roo.menu.BaseItem",
38095
38096     // private
38097     actionMode : "container",
38098
38099     // private
38100     render : function(container, parentMenu){
38101         this.parentMenu = parentMenu;
38102         Roo.menu.BaseItem.superclass.render.call(this, container);
38103         this.container.menuItemId = this.id;
38104     },
38105
38106     // private
38107     onRender : function(container, position){
38108         this.el = Roo.get(this.el);
38109         container.dom.appendChild(this.el.dom);
38110     },
38111
38112     // private
38113     onClick : function(e){
38114         if(!this.disabled && this.fireEvent("click", this, e) !== false
38115                 && this.parentMenu.fireEvent("itemclick", this, e) !== false){
38116             this.handleClick(e);
38117         }else{
38118             e.stopEvent();
38119         }
38120     },
38121
38122     // private
38123     activate : function(){
38124         if(this.disabled){
38125             return false;
38126         }
38127         var li = this.container;
38128         li.addClass(this.activeClass);
38129         this.region = li.getRegion().adjust(2, 2, -2, -2);
38130         this.fireEvent("activate", this);
38131         return true;
38132     },
38133
38134     // private
38135     deactivate : function(){
38136         this.container.removeClass(this.activeClass);
38137         this.fireEvent("deactivate", this);
38138     },
38139
38140     // private
38141     shouldDeactivate : function(e){
38142         return !this.region || !this.region.contains(e.getPoint());
38143     },
38144
38145     // private
38146     handleClick : function(e){
38147         if(this.hideOnClick){
38148             this.parentMenu.hide.defer(this.hideDelay, this.parentMenu, [true]);
38149         }
38150     },
38151
38152     // private
38153     expandMenu : function(autoActivate){
38154         // do nothing
38155     },
38156
38157     // private
38158     hideMenu : function(){
38159         // do nothing
38160     }
38161 });/*
38162  * Based on:
38163  * Ext JS Library 1.1.1
38164  * Copyright(c) 2006-2007, Ext JS, LLC.
38165  *
38166  * Originally Released Under LGPL - original licence link has changed is not relivant.
38167  *
38168  * Fork - LGPL
38169  * <script type="text/javascript">
38170  */
38171  
38172 /**
38173  * @class Roo.menu.Adapter
38174  * @extends Roo.menu.BaseItem
38175  * 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.
38176  * It provides basic rendering, activation management and enable/disable logic required to work in menus.
38177  * @constructor
38178  * Creates a new Adapter
38179  * @param {Object} config Configuration options
38180  */
38181 Roo.menu.Adapter = function(component, config){
38182     Roo.menu.Adapter.superclass.constructor.call(this, config);
38183     this.component = component;
38184 };
38185 Roo.extend(Roo.menu.Adapter, Roo.menu.BaseItem, {
38186     // private
38187     canActivate : true,
38188
38189     // private
38190     onRender : function(container, position){
38191         this.component.render(container);
38192         this.el = this.component.getEl();
38193     },
38194
38195     // private
38196     activate : function(){
38197         if(this.disabled){
38198             return false;
38199         }
38200         this.component.focus();
38201         this.fireEvent("activate", this);
38202         return true;
38203     },
38204
38205     // private
38206     deactivate : function(){
38207         this.fireEvent("deactivate", this);
38208     },
38209
38210     // private
38211     disable : function(){
38212         this.component.disable();
38213         Roo.menu.Adapter.superclass.disable.call(this);
38214     },
38215
38216     // private
38217     enable : function(){
38218         this.component.enable();
38219         Roo.menu.Adapter.superclass.enable.call(this);
38220     }
38221 });/*
38222  * Based on:
38223  * Ext JS Library 1.1.1
38224  * Copyright(c) 2006-2007, Ext JS, LLC.
38225  *
38226  * Originally Released Under LGPL - original licence link has changed is not relivant.
38227  *
38228  * Fork - LGPL
38229  * <script type="text/javascript">
38230  */
38231
38232 /**
38233  * @class Roo.menu.TextItem
38234  * @extends Roo.menu.BaseItem
38235  * Adds a static text string to a menu, usually used as either a heading or group separator.
38236  * Note: old style constructor with text is still supported.
38237  * 
38238  * @constructor
38239  * Creates a new TextItem
38240  * @param {Object} cfg Configuration
38241  */
38242 Roo.menu.TextItem = function(cfg){
38243     if (typeof(cfg) == 'string') {
38244         this.text = cfg;
38245     } else {
38246         Roo.apply(this,cfg);
38247     }
38248     
38249     Roo.menu.TextItem.superclass.constructor.call(this);
38250 };
38251
38252 Roo.extend(Roo.menu.TextItem, Roo.menu.BaseItem, {
38253     /**
38254      * @cfg {Boolean} text Text to show on item.
38255      */
38256     text : '',
38257     
38258     /**
38259      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to false)
38260      */
38261     hideOnClick : false,
38262     /**
38263      * @cfg {String} itemCls The default CSS class to use for text items (defaults to "x-menu-text")
38264      */
38265     itemCls : "x-menu-text",
38266
38267     // private
38268     onRender : function(){
38269         var s = document.createElement("span");
38270         s.className = this.itemCls;
38271         s.innerHTML = this.text;
38272         this.el = s;
38273         Roo.menu.TextItem.superclass.onRender.apply(this, arguments);
38274     }
38275 });/*
38276  * Based on:
38277  * Ext JS Library 1.1.1
38278  * Copyright(c) 2006-2007, Ext JS, LLC.
38279  *
38280  * Originally Released Under LGPL - original licence link has changed is not relivant.
38281  *
38282  * Fork - LGPL
38283  * <script type="text/javascript">
38284  */
38285
38286 /**
38287  * @class Roo.menu.Separator
38288  * @extends Roo.menu.BaseItem
38289  * Adds a separator bar to a menu, used to divide logical groups of menu items. Generally you will
38290  * add one of these by using "-" in you call to add() or in your items config rather than creating one directly.
38291  * @constructor
38292  * @param {Object} config Configuration options
38293  */
38294 Roo.menu.Separator = function(config){
38295     Roo.menu.Separator.superclass.constructor.call(this, config);
38296 };
38297
38298 Roo.extend(Roo.menu.Separator, Roo.menu.BaseItem, {
38299     /**
38300      * @cfg {String} itemCls The default CSS class to use for separators (defaults to "x-menu-sep")
38301      */
38302     itemCls : "x-menu-sep",
38303     /**
38304      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to false)
38305      */
38306     hideOnClick : false,
38307
38308     // private
38309     onRender : function(li){
38310         var s = document.createElement("span");
38311         s.className = this.itemCls;
38312         s.innerHTML = "&#160;";
38313         this.el = s;
38314         li.addClass("x-menu-sep-li");
38315         Roo.menu.Separator.superclass.onRender.apply(this, arguments);
38316     }
38317 });/*
38318  * Based on:
38319  * Ext JS Library 1.1.1
38320  * Copyright(c) 2006-2007, Ext JS, LLC.
38321  *
38322  * Originally Released Under LGPL - original licence link has changed is not relivant.
38323  *
38324  * Fork - LGPL
38325  * <script type="text/javascript">
38326  */
38327 /**
38328  * @class Roo.menu.Item
38329  * @extends Roo.menu.BaseItem
38330  * A base class for all menu items that require menu-related functionality (like sub-menus) and are not static
38331  * display items.  Item extends the base functionality of {@link Roo.menu.BaseItem} by adding menu-specific
38332  * activation and click handling.
38333  * @constructor
38334  * Creates a new Item
38335  * @param {Object} config Configuration options
38336  */
38337 Roo.menu.Item = function(config){
38338     Roo.menu.Item.superclass.constructor.call(this, config);
38339     if(this.menu){
38340         this.menu = Roo.menu.MenuMgr.get(this.menu);
38341     }
38342 };
38343 Roo.extend(Roo.menu.Item, Roo.menu.BaseItem, {
38344     
38345     /**
38346      * @cfg {String} text
38347      * The text to show on the menu item.
38348      */
38349     text: '',
38350      /**
38351      * @cfg {String} HTML to render in menu
38352      * The text to show on the menu item (HTML version).
38353      */
38354     html: '',
38355     /**
38356      * @cfg {String} icon
38357      * The path to an icon to display in this menu item (defaults to Roo.BLANK_IMAGE_URL)
38358      */
38359     icon: undefined,
38360     /**
38361      * @cfg {String} itemCls The default CSS class to use for menu items (defaults to "x-menu-item")
38362      */
38363     itemCls : "x-menu-item",
38364     /**
38365      * @cfg {Boolean} canActivate True if this item can be visually activated (defaults to true)
38366      */
38367     canActivate : true,
38368     /**
38369      * @cfg {Number} showDelay Length of time in milliseconds to wait before showing this item (defaults to 200)
38370      */
38371     showDelay: 200,
38372     // doc'd in BaseItem
38373     hideDelay: 200,
38374
38375     // private
38376     ctype: "Roo.menu.Item",
38377     
38378     // private
38379     onRender : function(container, position){
38380         var el = document.createElement("a");
38381         el.hideFocus = true;
38382         el.unselectable = "on";
38383         el.href = this.href || "#";
38384         if(this.hrefTarget){
38385             el.target = this.hrefTarget;
38386         }
38387         el.className = this.itemCls + (this.menu ?  " x-menu-item-arrow" : "") + (this.cls ?  " " + this.cls : "");
38388         
38389         var html = this.html.length ? this.html  : String.format('{0}',this.text);
38390         
38391         el.innerHTML = String.format(
38392                 '<img src="{0}" class="x-menu-item-icon {1}" />' + html,
38393                 this.icon || Roo.BLANK_IMAGE_URL, this.iconCls || '');
38394         this.el = el;
38395         Roo.menu.Item.superclass.onRender.call(this, container, position);
38396     },
38397
38398     /**
38399      * Sets the text to display in this menu item
38400      * @param {String} text The text to display
38401      * @param {Boolean} isHTML true to indicate text is pure html.
38402      */
38403     setText : function(text, isHTML){
38404         if (isHTML) {
38405             this.html = text;
38406         } else {
38407             this.text = text;
38408             this.html = '';
38409         }
38410         if(this.rendered){
38411             var html = this.html.length ? this.html  : String.format('{0}',this.text);
38412      
38413             this.el.update(String.format(
38414                 '<img src="{0}" class="x-menu-item-icon {2}">' + html,
38415                 this.icon || Roo.BLANK_IMAGE_URL, this.text, this.iconCls || ''));
38416             this.parentMenu.autoWidth();
38417         }
38418     },
38419
38420     // private
38421     handleClick : function(e){
38422         if(!this.href){ // if no link defined, stop the event automatically
38423             e.stopEvent();
38424         }
38425         Roo.menu.Item.superclass.handleClick.apply(this, arguments);
38426     },
38427
38428     // private
38429     activate : function(autoExpand){
38430         if(Roo.menu.Item.superclass.activate.apply(this, arguments)){
38431             this.focus();
38432             if(autoExpand){
38433                 this.expandMenu();
38434             }
38435         }
38436         return true;
38437     },
38438
38439     // private
38440     shouldDeactivate : function(e){
38441         if(Roo.menu.Item.superclass.shouldDeactivate.call(this, e)){
38442             if(this.menu && this.menu.isVisible()){
38443                 return !this.menu.getEl().getRegion().contains(e.getPoint());
38444             }
38445             return true;
38446         }
38447         return false;
38448     },
38449
38450     // private
38451     deactivate : function(){
38452         Roo.menu.Item.superclass.deactivate.apply(this, arguments);
38453         this.hideMenu();
38454     },
38455
38456     // private
38457     expandMenu : function(autoActivate){
38458         if(!this.disabled && this.menu){
38459             clearTimeout(this.hideTimer);
38460             delete this.hideTimer;
38461             if(!this.menu.isVisible() && !this.showTimer){
38462                 this.showTimer = this.deferExpand.defer(this.showDelay, this, [autoActivate]);
38463             }else if (this.menu.isVisible() && autoActivate){
38464                 this.menu.tryActivate(0, 1);
38465             }
38466         }
38467     },
38468
38469     // private
38470     deferExpand : function(autoActivate){
38471         delete this.showTimer;
38472         this.menu.show(this.container, this.parentMenu.subMenuAlign || "tl-tr?", this.parentMenu);
38473         if(autoActivate){
38474             this.menu.tryActivate(0, 1);
38475         }
38476     },
38477
38478     // private
38479     hideMenu : function(){
38480         clearTimeout(this.showTimer);
38481         delete this.showTimer;
38482         if(!this.hideTimer && this.menu && this.menu.isVisible()){
38483             this.hideTimer = this.deferHide.defer(this.hideDelay, this);
38484         }
38485     },
38486
38487     // private
38488     deferHide : function(){
38489         delete this.hideTimer;
38490         this.menu.hide();
38491     }
38492 });/*
38493  * Based on:
38494  * Ext JS Library 1.1.1
38495  * Copyright(c) 2006-2007, Ext JS, LLC.
38496  *
38497  * Originally Released Under LGPL - original licence link has changed is not relivant.
38498  *
38499  * Fork - LGPL
38500  * <script type="text/javascript">
38501  */
38502  
38503 /**
38504  * @class Roo.menu.CheckItem
38505  * @extends Roo.menu.Item
38506  * Adds a menu item that contains a checkbox by default, but can also be part of a radio group.
38507  * @constructor
38508  * Creates a new CheckItem
38509  * @param {Object} config Configuration options
38510  */
38511 Roo.menu.CheckItem = function(config){
38512     Roo.menu.CheckItem.superclass.constructor.call(this, config);
38513     this.addEvents({
38514         /**
38515          * @event beforecheckchange
38516          * Fires before the checked value is set, providing an opportunity to cancel if needed
38517          * @param {Roo.menu.CheckItem} this
38518          * @param {Boolean} checked The new checked value that will be set
38519          */
38520         "beforecheckchange" : true,
38521         /**
38522          * @event checkchange
38523          * Fires after the checked value has been set
38524          * @param {Roo.menu.CheckItem} this
38525          * @param {Boolean} checked The checked value that was set
38526          */
38527         "checkchange" : true
38528     });
38529     if(this.checkHandler){
38530         this.on('checkchange', this.checkHandler, this.scope);
38531     }
38532 };
38533 Roo.extend(Roo.menu.CheckItem, Roo.menu.Item, {
38534     /**
38535      * @cfg {String} group
38536      * All check items with the same group name will automatically be grouped into a single-select
38537      * radio button group (defaults to '')
38538      */
38539     /**
38540      * @cfg {String} itemCls The default CSS class to use for check items (defaults to "x-menu-item x-menu-check-item")
38541      */
38542     itemCls : "x-menu-item x-menu-check-item",
38543     /**
38544      * @cfg {String} groupClass The default CSS class to use for radio group check items (defaults to "x-menu-group-item")
38545      */
38546     groupClass : "x-menu-group-item",
38547
38548     /**
38549      * @cfg {Boolean} checked True to initialize this checkbox as checked (defaults to false).  Note that
38550      * if this checkbox is part of a radio group (group = true) only the last item in the group that is
38551      * initialized with checked = true will be rendered as checked.
38552      */
38553     checked: false,
38554
38555     // private
38556     ctype: "Roo.menu.CheckItem",
38557
38558     // private
38559     onRender : function(c){
38560         Roo.menu.CheckItem.superclass.onRender.apply(this, arguments);
38561         if(this.group){
38562             this.el.addClass(this.groupClass);
38563         }
38564         Roo.menu.MenuMgr.registerCheckable(this);
38565         if(this.checked){
38566             this.checked = false;
38567             this.setChecked(true, true);
38568         }
38569     },
38570
38571     // private
38572     destroy : function(){
38573         if(this.rendered){
38574             Roo.menu.MenuMgr.unregisterCheckable(this);
38575         }
38576         Roo.menu.CheckItem.superclass.destroy.apply(this, arguments);
38577     },
38578
38579     /**
38580      * Set the checked state of this item
38581      * @param {Boolean} checked The new checked value
38582      * @param {Boolean} suppressEvent (optional) True to prevent the checkchange event from firing (defaults to false)
38583      */
38584     setChecked : function(state, suppressEvent){
38585         if(this.checked != state && this.fireEvent("beforecheckchange", this, state) !== false){
38586             if(this.container){
38587                 this.container[state ? "addClass" : "removeClass"]("x-menu-item-checked");
38588             }
38589             this.checked = state;
38590             if(suppressEvent !== true){
38591                 this.fireEvent("checkchange", this, state);
38592             }
38593         }
38594     },
38595
38596     // private
38597     handleClick : function(e){
38598        if(!this.disabled && !(this.checked && this.group)){// disable unselect on radio item
38599            this.setChecked(!this.checked);
38600        }
38601        Roo.menu.CheckItem.superclass.handleClick.apply(this, arguments);
38602     }
38603 });/*
38604  * Based on:
38605  * Ext JS Library 1.1.1
38606  * Copyright(c) 2006-2007, Ext JS, LLC.
38607  *
38608  * Originally Released Under LGPL - original licence link has changed is not relivant.
38609  *
38610  * Fork - LGPL
38611  * <script type="text/javascript">
38612  */
38613  
38614 /**
38615  * @class Roo.menu.DateItem
38616  * @extends Roo.menu.Adapter
38617  * A menu item that wraps the {@link Roo.DatPicker} component.
38618  * @constructor
38619  * Creates a new DateItem
38620  * @param {Object} config Configuration options
38621  */
38622 Roo.menu.DateItem = function(config){
38623     Roo.menu.DateItem.superclass.constructor.call(this, new Roo.DatePicker(config), config);
38624     /** The Roo.DatePicker object @type Roo.DatePicker */
38625     this.picker = this.component;
38626     this.addEvents({select: true});
38627     
38628     this.picker.on("render", function(picker){
38629         picker.getEl().swallowEvent("click");
38630         picker.container.addClass("x-menu-date-item");
38631     });
38632
38633     this.picker.on("select", this.onSelect, this);
38634 };
38635
38636 Roo.extend(Roo.menu.DateItem, Roo.menu.Adapter, {
38637     // private
38638     onSelect : function(picker, date){
38639         this.fireEvent("select", this, date, picker);
38640         Roo.menu.DateItem.superclass.handleClick.call(this);
38641     }
38642 });/*
38643  * Based on:
38644  * Ext JS Library 1.1.1
38645  * Copyright(c) 2006-2007, Ext JS, LLC.
38646  *
38647  * Originally Released Under LGPL - original licence link has changed is not relivant.
38648  *
38649  * Fork - LGPL
38650  * <script type="text/javascript">
38651  */
38652  
38653 /**
38654  * @class Roo.menu.ColorItem
38655  * @extends Roo.menu.Adapter
38656  * A menu item that wraps the {@link Roo.ColorPalette} component.
38657  * @constructor
38658  * Creates a new ColorItem
38659  * @param {Object} config Configuration options
38660  */
38661 Roo.menu.ColorItem = function(config){
38662     Roo.menu.ColorItem.superclass.constructor.call(this, new Roo.ColorPalette(config), config);
38663     /** The Roo.ColorPalette object @type Roo.ColorPalette */
38664     this.palette = this.component;
38665     this.relayEvents(this.palette, ["select"]);
38666     if(this.selectHandler){
38667         this.on('select', this.selectHandler, this.scope);
38668     }
38669 };
38670 Roo.extend(Roo.menu.ColorItem, Roo.menu.Adapter);/*
38671  * Based on:
38672  * Ext JS Library 1.1.1
38673  * Copyright(c) 2006-2007, Ext JS, LLC.
38674  *
38675  * Originally Released Under LGPL - original licence link has changed is not relivant.
38676  *
38677  * Fork - LGPL
38678  * <script type="text/javascript">
38679  */
38680  
38681
38682 /**
38683  * @class Roo.menu.DateMenu
38684  * @extends Roo.menu.Menu
38685  * A menu containing a {@link Roo.menu.DateItem} component (which provides a date picker).
38686  * @constructor
38687  * Creates a new DateMenu
38688  * @param {Object} config Configuration options
38689  */
38690 Roo.menu.DateMenu = function(config){
38691     Roo.menu.DateMenu.superclass.constructor.call(this, config);
38692     this.plain = true;
38693     var di = new Roo.menu.DateItem(config);
38694     this.add(di);
38695     /**
38696      * The {@link Roo.DatePicker} instance for this DateMenu
38697      * @type DatePicker
38698      */
38699     this.picker = di.picker;
38700     /**
38701      * @event select
38702      * @param {DatePicker} picker
38703      * @param {Date} date
38704      */
38705     this.relayEvents(di, ["select"]);
38706     this.on('beforeshow', function(){
38707         if(this.picker){
38708             this.picker.hideMonthPicker(false);
38709         }
38710     }, this);
38711 };
38712 Roo.extend(Roo.menu.DateMenu, Roo.menu.Menu, {
38713     cls:'x-date-menu'
38714 });/*
38715  * Based on:
38716  * Ext JS Library 1.1.1
38717  * Copyright(c) 2006-2007, Ext JS, LLC.
38718  *
38719  * Originally Released Under LGPL - original licence link has changed is not relivant.
38720  *
38721  * Fork - LGPL
38722  * <script type="text/javascript">
38723  */
38724  
38725
38726 /**
38727  * @class Roo.menu.ColorMenu
38728  * @extends Roo.menu.Menu
38729  * A menu containing a {@link Roo.menu.ColorItem} component (which provides a basic color picker).
38730  * @constructor
38731  * Creates a new ColorMenu
38732  * @param {Object} config Configuration options
38733  */
38734 Roo.menu.ColorMenu = function(config){
38735     Roo.menu.ColorMenu.superclass.constructor.call(this, config);
38736     this.plain = true;
38737     var ci = new Roo.menu.ColorItem(config);
38738     this.add(ci);
38739     /**
38740      * The {@link Roo.ColorPalette} instance for this ColorMenu
38741      * @type ColorPalette
38742      */
38743     this.palette = ci.palette;
38744     /**
38745      * @event select
38746      * @param {ColorPalette} palette
38747      * @param {String} color
38748      */
38749     this.relayEvents(ci, ["select"]);
38750 };
38751 Roo.extend(Roo.menu.ColorMenu, Roo.menu.Menu);/*
38752  * Based on:
38753  * Ext JS Library 1.1.1
38754  * Copyright(c) 2006-2007, Ext JS, LLC.
38755  *
38756  * Originally Released Under LGPL - original licence link has changed is not relivant.
38757  *
38758  * Fork - LGPL
38759  * <script type="text/javascript">
38760  */
38761  
38762 /**
38763  * @class Roo.form.TextItem
38764  * @extends Roo.BoxComponent
38765  * Base class for form fields that provides default event handling, sizing, value handling and other functionality.
38766  * @constructor
38767  * Creates a new TextItem
38768  * @param {Object} config Configuration options
38769  */
38770 Roo.form.TextItem = function(config){
38771     Roo.form.TextItem.superclass.constructor.call(this, config);
38772 };
38773
38774 Roo.extend(Roo.form.TextItem, Roo.BoxComponent,  {
38775     
38776     /**
38777      * @cfg {String} tag the tag for this item (default div)
38778      */
38779     tag : 'div',
38780     /**
38781      * @cfg {String} html the content for this item
38782      */
38783     html : '',
38784     
38785     getAutoCreate : function()
38786     {
38787         var cfg = {
38788             id: this.id,
38789             tag: this.tag,
38790             html: this.html,
38791             cls: 'x-form-item'
38792         };
38793         
38794         return cfg;
38795         
38796     },
38797     
38798     onRender : function(ct, position)
38799     {
38800         Roo.form.TextItem.superclass.onRender.call(this, ct, position);
38801         
38802         if(!this.el){
38803             var cfg = this.getAutoCreate();
38804             if(!cfg.name){
38805                 cfg.name = typeof(this.name) == 'undefined' ? this.id : this.name;
38806             }
38807             if (!cfg.name.length) {
38808                 delete cfg.name;
38809             }
38810             this.el = ct.createChild(cfg, position);
38811         }
38812     }
38813     
38814 });/*
38815  * Based on:
38816  * Ext JS Library 1.1.1
38817  * Copyright(c) 2006-2007, Ext JS, LLC.
38818  *
38819  * Originally Released Under LGPL - original licence link has changed is not relivant.
38820  *
38821  * Fork - LGPL
38822  * <script type="text/javascript">
38823  */
38824  
38825 /**
38826  * @class Roo.form.Field
38827  * @extends Roo.BoxComponent
38828  * Base class for form fields that provides default event handling, sizing, value handling and other functionality.
38829  * @constructor
38830  * Creates a new Field
38831  * @param {Object} config Configuration options
38832  */
38833 Roo.form.Field = function(config){
38834     Roo.form.Field.superclass.constructor.call(this, config);
38835 };
38836
38837 Roo.extend(Roo.form.Field, Roo.BoxComponent,  {
38838     /**
38839      * @cfg {String} fieldLabel Label to use when rendering a form.
38840      */
38841        /**
38842      * @cfg {String} qtip Mouse over tip
38843      */
38844      
38845     /**
38846      * @cfg {String} invalidClass The CSS class to use when marking a field invalid (defaults to "x-form-invalid")
38847      */
38848     invalidClass : "x-form-invalid",
38849     /**
38850      * @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")
38851      */
38852     invalidText : "The value in this field is invalid",
38853     /**
38854      * @cfg {String} focusClass The CSS class to use when the field receives focus (defaults to "x-form-focus")
38855      */
38856     focusClass : "x-form-focus",
38857     /**
38858      * @cfg {String/Boolean} validationEvent The event that should initiate field validation. Set to false to disable
38859       automatic validation (defaults to "keyup").
38860      */
38861     validationEvent : "keyup",
38862     /**
38863      * @cfg {Boolean} validateOnBlur Whether the field should validate when it loses focus (defaults to true).
38864      */
38865     validateOnBlur : true,
38866     /**
38867      * @cfg {Number} validationDelay The length of time in milliseconds after user input begins until validation is initiated (defaults to 250)
38868      */
38869     validationDelay : 250,
38870     /**
38871      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
38872      * {tag: "input", type: "text", size: "20", autocomplete: "off"})
38873      */
38874     defaultAutoCreate : {tag: "input", type: "text", size: "20", autocomplete: "new-password"},
38875     /**
38876      * @cfg {String} fieldClass The default CSS class for the field (defaults to "x-form-field")
38877      */
38878     fieldClass : "x-form-field",
38879     /**
38880      * @cfg {String} msgTarget The location where error text should display.  Should be one of the following values (defaults to 'qtip'):
38881      *<pre>
38882 Value         Description
38883 -----------   ----------------------------------------------------------------------
38884 qtip          Display a quick tip when the user hovers over the field
38885 title         Display a default browser title attribute popup
38886 under         Add a block div beneath the field containing the error text
38887 side          Add an error icon to the right of the field with a popup on hover
38888 [element id]  Add the error text directly to the innerHTML of the specified element
38889 </pre>
38890      */
38891     msgTarget : 'qtip',
38892     /**
38893      * @cfg {String} msgFx <b>Experimental</b> The effect used when displaying a validation message under the field (defaults to 'normal').
38894      */
38895     msgFx : 'normal',
38896
38897     /**
38898      * @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.
38899      */
38900     readOnly : false,
38901
38902     /**
38903      * @cfg {Boolean} disabled True to disable the field (defaults to false).
38904      */
38905     disabled : false,
38906
38907     /**
38908      * @cfg {String} inputType The type attribute for input fields -- e.g. radio, text, password (defaults to "text").
38909      */
38910     inputType : undefined,
38911     
38912     /**
38913      * @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).
38914          */
38915         tabIndex : undefined,
38916         
38917     // private
38918     isFormField : true,
38919
38920     // private
38921     hasFocus : false,
38922     /**
38923      * @property {Roo.Element} fieldEl
38924      * Element Containing the rendered Field (with label etc.)
38925      */
38926     /**
38927      * @cfg {Mixed} value A value to initialize this field with.
38928      */
38929     value : undefined,
38930
38931     /**
38932      * @cfg {String} name The field's HTML name attribute.
38933      */
38934     /**
38935      * @cfg {String} cls A CSS class to apply to the field's underlying element.
38936      */
38937     // private
38938     loadedValue : false,
38939      
38940      
38941         // private ??
38942         initComponent : function(){
38943         Roo.form.Field.superclass.initComponent.call(this);
38944         this.addEvents({
38945             /**
38946              * @event focus
38947              * Fires when this field receives input focus.
38948              * @param {Roo.form.Field} this
38949              */
38950             focus : true,
38951             /**
38952              * @event blur
38953              * Fires when this field loses input focus.
38954              * @param {Roo.form.Field} this
38955              */
38956             blur : true,
38957             /**
38958              * @event specialkey
38959              * Fires when any key related to navigation (arrows, tab, enter, esc, etc.) is pressed.  You can check
38960              * {@link Roo.EventObject#getKey} to determine which key was pressed.
38961              * @param {Roo.form.Field} this
38962              * @param {Roo.EventObject} e The event object
38963              */
38964             specialkey : true,
38965             /**
38966              * @event change
38967              * Fires just before the field blurs if the field value has changed.
38968              * @param {Roo.form.Field} this
38969              * @param {Mixed} newValue The new value
38970              * @param {Mixed} oldValue The original value
38971              */
38972             change : true,
38973             /**
38974              * @event invalid
38975              * Fires after the field has been marked as invalid.
38976              * @param {Roo.form.Field} this
38977              * @param {String} msg The validation message
38978              */
38979             invalid : true,
38980             /**
38981              * @event valid
38982              * Fires after the field has been validated with no errors.
38983              * @param {Roo.form.Field} this
38984              */
38985             valid : true,
38986              /**
38987              * @event keyup
38988              * Fires after the key up
38989              * @param {Roo.form.Field} this
38990              * @param {Roo.EventObject}  e The event Object
38991              */
38992             keyup : true
38993         });
38994     },
38995
38996     /**
38997      * Returns the name attribute of the field if available
38998      * @return {String} name The field name
38999      */
39000     getName: function(){
39001          return this.rendered && this.el.dom.name ? this.el.dom.name : (this.hiddenName || '');
39002     },
39003
39004     // private
39005     onRender : function(ct, position){
39006         Roo.form.Field.superclass.onRender.call(this, ct, position);
39007         if(!this.el){
39008             var cfg = this.getAutoCreate();
39009             if(!cfg.name){
39010                 cfg.name = typeof(this.name) == 'undefined' ? this.id : this.name;
39011             }
39012             if (!cfg.name.length) {
39013                 delete cfg.name;
39014             }
39015             if(this.inputType){
39016                 cfg.type = this.inputType;
39017             }
39018             this.el = ct.createChild(cfg, position);
39019         }
39020         var type = this.el.dom.type;
39021         if(type){
39022             if(type == 'password'){
39023                 type = 'text';
39024             }
39025             this.el.addClass('x-form-'+type);
39026         }
39027         if(this.readOnly){
39028             this.el.dom.readOnly = true;
39029         }
39030         if(this.tabIndex !== undefined){
39031             this.el.dom.setAttribute('tabIndex', this.tabIndex);
39032         }
39033
39034         this.el.addClass([this.fieldClass, this.cls]);
39035         this.initValue();
39036     },
39037
39038     /**
39039      * Apply the behaviors of this component to an existing element. <b>This is used instead of render().</b>
39040      * @param {String/HTMLElement/Element} el The id of the node, a DOM node or an existing Element
39041      * @return {Roo.form.Field} this
39042      */
39043     applyTo : function(target){
39044         this.allowDomMove = false;
39045         this.el = Roo.get(target);
39046         this.render(this.el.dom.parentNode);
39047         return this;
39048     },
39049
39050     // private
39051     initValue : function(){
39052         if(this.value !== undefined){
39053             this.setValue(this.value);
39054         }else if(this.el.dom.value.length > 0){
39055             this.setValue(this.el.dom.value);
39056         }
39057     },
39058
39059     /**
39060      * Returns true if this field has been changed since it was originally loaded and is not disabled.
39061      * DEPRICATED  - it never worked well - use hasChanged/resetHasChanged.
39062      */
39063     isDirty : function() {
39064         if(this.disabled) {
39065             return false;
39066         }
39067         return String(this.getValue()) !== String(this.originalValue);
39068     },
39069
39070     /**
39071      * stores the current value in loadedValue
39072      */
39073     resetHasChanged : function()
39074     {
39075         this.loadedValue = String(this.getValue());
39076     },
39077     /**
39078      * checks the current value against the 'loaded' value.
39079      * Note - will return false if 'resetHasChanged' has not been called first.
39080      */
39081     hasChanged : function()
39082     {
39083         if(this.disabled || this.readOnly) {
39084             return false;
39085         }
39086         return this.loadedValue !== false && String(this.getValue()) !== this.loadedValue;
39087     },
39088     
39089     
39090     
39091     // private
39092     afterRender : function(){
39093         Roo.form.Field.superclass.afterRender.call(this);
39094         this.initEvents();
39095     },
39096
39097     // private
39098     fireKey : function(e){
39099         //Roo.log('field ' + e.getKey());
39100         if(e.isNavKeyPress()){
39101             this.fireEvent("specialkey", this, e);
39102         }
39103     },
39104
39105     /**
39106      * Resets the current field value to the originally loaded value and clears any validation messages
39107      */
39108     reset : function(){
39109         this.setValue(this.resetValue);
39110         this.originalValue = this.getValue();
39111         this.clearInvalid();
39112     },
39113
39114     // private
39115     initEvents : function(){
39116         // safari killled keypress - so keydown is now used..
39117         this.el.on("keydown" , this.fireKey,  this);
39118         this.el.on("focus", this.onFocus,  this);
39119         this.el.on("blur", this.onBlur,  this);
39120         this.el.relayEvent('keyup', this);
39121
39122         // reference to original value for reset
39123         this.originalValue = this.getValue();
39124         this.resetValue =  this.getValue();
39125     },
39126
39127     // private
39128     onFocus : function(){
39129         if(!Roo.isOpera && this.focusClass){ // don't touch in Opera
39130             this.el.addClass(this.focusClass);
39131         }
39132         if(!this.hasFocus){
39133             this.hasFocus = true;
39134             this.startValue = this.getValue();
39135             this.fireEvent("focus", this);
39136         }
39137     },
39138
39139     beforeBlur : Roo.emptyFn,
39140
39141     // private
39142     onBlur : function(){
39143         this.beforeBlur();
39144         if(!Roo.isOpera && this.focusClass){ // don't touch in Opera
39145             this.el.removeClass(this.focusClass);
39146         }
39147         this.hasFocus = false;
39148         if(this.validationEvent !== false && this.validateOnBlur && this.validationEvent != "blur"){
39149             this.validate();
39150         }
39151         var v = this.getValue();
39152         if(String(v) !== String(this.startValue)){
39153             this.fireEvent('change', this, v, this.startValue);
39154         }
39155         this.fireEvent("blur", this);
39156     },
39157
39158     /**
39159      * Returns whether or not the field value is currently valid
39160      * @param {Boolean} preventMark True to disable marking the field invalid
39161      * @return {Boolean} True if the value is valid, else false
39162      */
39163     isValid : function(preventMark){
39164         if(this.disabled){
39165             return true;
39166         }
39167         var restore = this.preventMark;
39168         this.preventMark = preventMark === true;
39169         var v = this.validateValue(this.processValue(this.getRawValue()));
39170         this.preventMark = restore;
39171         return v;
39172     },
39173
39174     /**
39175      * Validates the field value
39176      * @return {Boolean} True if the value is valid, else false
39177      */
39178     validate : function(){
39179         if(this.disabled || this.validateValue(this.processValue(this.getRawValue()))){
39180             this.clearInvalid();
39181             return true;
39182         }
39183         return false;
39184     },
39185
39186     processValue : function(value){
39187         return value;
39188     },
39189
39190     // private
39191     // Subclasses should provide the validation implementation by overriding this
39192     validateValue : function(value){
39193         return true;
39194     },
39195
39196     /**
39197      * Mark this field as invalid
39198      * @param {String} msg The validation message
39199      */
39200     markInvalid : function(msg){
39201         if(!this.rendered || this.preventMark){ // not rendered
39202             return;
39203         }
39204         
39205         var obj = (typeof(this.combo) != 'undefined') ? this.combo : this; // fix the combox array!!
39206         
39207         obj.el.addClass(this.invalidClass);
39208         msg = msg || this.invalidText;
39209         switch(this.msgTarget){
39210             case 'qtip':
39211                 obj.el.dom.qtip = msg;
39212                 obj.el.dom.qclass = 'x-form-invalid-tip';
39213                 if(Roo.QuickTips){ // fix for floating editors interacting with DND
39214                     Roo.QuickTips.enable();
39215                 }
39216                 break;
39217             case 'title':
39218                 this.el.dom.title = msg;
39219                 break;
39220             case 'under':
39221                 if(!this.errorEl){
39222                     var elp = this.el.findParent('.x-form-element', 5, true);
39223                     this.errorEl = elp.createChild({cls:'x-form-invalid-msg'});
39224                     this.errorEl.setWidth(elp.getWidth(true)-20);
39225                 }
39226                 this.errorEl.update(msg);
39227                 Roo.form.Field.msgFx[this.msgFx].show(this.errorEl, this);
39228                 break;
39229             case 'side':
39230                 if(!this.errorIcon){
39231                     var elp = this.el.findParent('.x-form-element', 5, true);
39232                     this.errorIcon = elp.createChild({cls:'x-form-invalid-icon'});
39233                 }
39234                 this.alignErrorIcon();
39235                 this.errorIcon.dom.qtip = msg;
39236                 this.errorIcon.dom.qclass = 'x-form-invalid-tip';
39237                 this.errorIcon.show();
39238                 this.on('resize', this.alignErrorIcon, this);
39239                 break;
39240             default:
39241                 var t = Roo.getDom(this.msgTarget);
39242                 t.innerHTML = msg;
39243                 t.style.display = this.msgDisplay;
39244                 break;
39245         }
39246         this.fireEvent('invalid', this, msg);
39247     },
39248
39249     // private
39250     alignErrorIcon : function(){
39251         this.errorIcon.alignTo(this.el, 'tl-tr', [2, 0]);
39252     },
39253
39254     /**
39255      * Clear any invalid styles/messages for this field
39256      */
39257     clearInvalid : function(){
39258         if(!this.rendered || this.preventMark){ // not rendered
39259             return;
39260         }
39261         var obj = (typeof(this.combo) != 'undefined') ? this.combo : this; // fix the combox array!!
39262         
39263         obj.el.removeClass(this.invalidClass);
39264         switch(this.msgTarget){
39265             case 'qtip':
39266                 obj.el.dom.qtip = '';
39267                 break;
39268             case 'title':
39269                 this.el.dom.title = '';
39270                 break;
39271             case 'under':
39272                 if(this.errorEl){
39273                     Roo.form.Field.msgFx[this.msgFx].hide(this.errorEl, this);
39274                 }
39275                 break;
39276             case 'side':
39277                 if(this.errorIcon){
39278                     this.errorIcon.dom.qtip = '';
39279                     this.errorIcon.hide();
39280                     this.un('resize', this.alignErrorIcon, this);
39281                 }
39282                 break;
39283             default:
39284                 var t = Roo.getDom(this.msgTarget);
39285                 t.innerHTML = '';
39286                 t.style.display = 'none';
39287                 break;
39288         }
39289         this.fireEvent('valid', this);
39290     },
39291
39292     /**
39293      * Returns the raw data value which may or may not be a valid, defined value.  To return a normalized value see {@link #getValue}.
39294      * @return {Mixed} value The field value
39295      */
39296     getRawValue : function(){
39297         var v = this.el.getValue();
39298         
39299         return v;
39300     },
39301
39302     /**
39303      * Returns the normalized data value (undefined or emptyText will be returned as '').  To return the raw value see {@link #getRawValue}.
39304      * @return {Mixed} value The field value
39305      */
39306     getValue : function(){
39307         var v = this.el.getValue();
39308          
39309         return v;
39310     },
39311
39312     /**
39313      * Sets the underlying DOM field's value directly, bypassing validation.  To set the value with validation see {@link #setValue}.
39314      * @param {Mixed} value The value to set
39315      */
39316     setRawValue : function(v){
39317         return this.el.dom.value = (v === null || v === undefined ? '' : v);
39318     },
39319
39320     /**
39321      * Sets a data value into the field and validates it.  To set the value directly without validation see {@link #setRawValue}.
39322      * @param {Mixed} value The value to set
39323      */
39324     setValue : function(v){
39325         this.value = v;
39326         if(this.rendered){
39327             this.el.dom.value = (v === null || v === undefined ? '' : v);
39328              this.validate();
39329         }
39330     },
39331
39332     adjustSize : function(w, h){
39333         var s = Roo.form.Field.superclass.adjustSize.call(this, w, h);
39334         s.width = this.adjustWidth(this.el.dom.tagName, s.width);
39335         return s;
39336     },
39337
39338     adjustWidth : function(tag, w){
39339         tag = tag.toLowerCase();
39340         if(typeof w == 'number' && Roo.isStrict && !Roo.isSafari){
39341             if(Roo.isIE && (tag == 'input' || tag == 'textarea')){
39342                 if(tag == 'input'){
39343                     return w + 2;
39344                 }
39345                 if(tag == 'textarea'){
39346                     return w-2;
39347                 }
39348             }else if(Roo.isOpera){
39349                 if(tag == 'input'){
39350                     return w + 2;
39351                 }
39352                 if(tag == 'textarea'){
39353                     return w-2;
39354                 }
39355             }
39356         }
39357         return w;
39358     }
39359 });
39360
39361
39362 // anything other than normal should be considered experimental
39363 Roo.form.Field.msgFx = {
39364     normal : {
39365         show: function(msgEl, f){
39366             msgEl.setDisplayed('block');
39367         },
39368
39369         hide : function(msgEl, f){
39370             msgEl.setDisplayed(false).update('');
39371         }
39372     },
39373
39374     slide : {
39375         show: function(msgEl, f){
39376             msgEl.slideIn('t', {stopFx:true});
39377         },
39378
39379         hide : function(msgEl, f){
39380             msgEl.slideOut('t', {stopFx:true,useDisplay:true});
39381         }
39382     },
39383
39384     slideRight : {
39385         show: function(msgEl, f){
39386             msgEl.fixDisplay();
39387             msgEl.alignTo(f.el, 'tl-tr');
39388             msgEl.slideIn('l', {stopFx:true});
39389         },
39390
39391         hide : function(msgEl, f){
39392             msgEl.slideOut('l', {stopFx:true,useDisplay:true});
39393         }
39394     }
39395 };/*
39396  * Based on:
39397  * Ext JS Library 1.1.1
39398  * Copyright(c) 2006-2007, Ext JS, LLC.
39399  *
39400  * Originally Released Under LGPL - original licence link has changed is not relivant.
39401  *
39402  * Fork - LGPL
39403  * <script type="text/javascript">
39404  */
39405  
39406
39407 /**
39408  * @class Roo.form.TextField
39409  * @extends Roo.form.Field
39410  * Basic text field.  Can be used as a direct replacement for traditional text inputs, or as the base
39411  * class for more sophisticated input controls (like {@link Roo.form.TextArea} and {@link Roo.form.ComboBox}).
39412  * @constructor
39413  * Creates a new TextField
39414  * @param {Object} config Configuration options
39415  */
39416 Roo.form.TextField = function(config){
39417     Roo.form.TextField.superclass.constructor.call(this, config);
39418     this.addEvents({
39419         /**
39420          * @event autosize
39421          * Fires when the autosize function is triggered.  The field may or may not have actually changed size
39422          * according to the default logic, but this event provides a hook for the developer to apply additional
39423          * logic at runtime to resize the field if needed.
39424              * @param {Roo.form.Field} this This text field
39425              * @param {Number} width The new field width
39426              */
39427         autosize : true
39428     });
39429 };
39430
39431 Roo.extend(Roo.form.TextField, Roo.form.Field,  {
39432     /**
39433      * @cfg {Boolean} grow True if this field should automatically grow and shrink to its content
39434      */
39435     grow : false,
39436     /**
39437      * @cfg {Number} growMin The minimum width to allow when grow = true (defaults to 30)
39438      */
39439     growMin : 30,
39440     /**
39441      * @cfg {Number} growMax The maximum width to allow when grow = true (defaults to 800)
39442      */
39443     growMax : 800,
39444     /**
39445      * @cfg {String} vtype A validation type name as defined in {@link Roo.form.VTypes} (defaults to null)
39446      */
39447     vtype : null,
39448     /**
39449      * @cfg {String} maskRe An input mask regular expression that will be used to filter keystrokes that don't match (defaults to null)
39450      */
39451     maskRe : null,
39452     /**
39453      * @cfg {Boolean} disableKeyFilter True to disable input keystroke filtering (defaults to false)
39454      */
39455     disableKeyFilter : false,
39456     /**
39457      * @cfg {Boolean} allowBlank False to validate that the value length > 0 (defaults to true)
39458      */
39459     allowBlank : true,
39460     /**
39461      * @cfg {Number} minLength Minimum input field length required (defaults to 0)
39462      */
39463     minLength : 0,
39464     /**
39465      * @cfg {Number} maxLength Maximum input field length allowed (defaults to Number.MAX_VALUE)
39466      */
39467     maxLength : Number.MAX_VALUE,
39468     /**
39469      * @cfg {String} minLengthText Error text to display if the minimum length validation fails (defaults to "The minimum length for this field is {minLength}")
39470      */
39471     minLengthText : "The minimum length for this field is {0}",
39472     /**
39473      * @cfg {String} maxLengthText Error text to display if the maximum length validation fails (defaults to "The maximum length for this field is {maxLength}")
39474      */
39475     maxLengthText : "The maximum length for this field is {0}",
39476     /**
39477      * @cfg {Boolean} selectOnFocus True to automatically select any existing field text when the field receives input focus (defaults to false)
39478      */
39479     selectOnFocus : false,
39480     /**
39481      * @cfg {Boolean} allowLeadingSpace True to prevent the stripping of leading white space 
39482      */    
39483     allowLeadingSpace : false,
39484     /**
39485      * @cfg {String} blankText Error text to display if the allow blank validation fails (defaults to "This field is required")
39486      */
39487     blankText : "This field is required",
39488     /**
39489      * @cfg {Function} validator A custom validation function to be called during field validation (defaults to null).
39490      * If available, this function will be called only after the basic validators all return true, and will be passed the
39491      * current field value and expected to return boolean true if the value is valid or a string error message if invalid.
39492      */
39493     validator : null,
39494     /**
39495      * @cfg {RegExp} regex A JavaScript RegExp object to be tested against the field value during validation (defaults to null).
39496      * If available, this regex will be evaluated only after the basic validators all return true, and will be passed the
39497      * current field value.  If the test fails, the field will be marked invalid using {@link #regexText}.
39498      */
39499     regex : null,
39500     /**
39501      * @cfg {String} regexText The error text to display if {@link #regex} is used and the test fails during validation (defaults to "")
39502      */
39503     regexText : "",
39504     /**
39505      * @cfg {String} emptyText The default text to display in an empty field - placeholder... (defaults to null).
39506      */
39507     emptyText : null,
39508    
39509
39510     // private
39511     initEvents : function()
39512     {
39513         if (this.emptyText) {
39514             this.el.attr('placeholder', this.emptyText);
39515         }
39516         
39517         Roo.form.TextField.superclass.initEvents.call(this);
39518         if(this.validationEvent == 'keyup'){
39519             this.validationTask = new Roo.util.DelayedTask(this.validate, this);
39520             this.el.on('keyup', this.filterValidation, this);
39521         }
39522         else if(this.validationEvent !== false){
39523             this.el.on(this.validationEvent, this.validate, this, {buffer: this.validationDelay});
39524         }
39525         
39526         if(this.selectOnFocus){
39527             this.on("focus", this.preFocus, this);
39528         }
39529         if (!this.allowLeadingSpace) {
39530             this.on('blur', this.cleanLeadingSpace, this);
39531         }
39532         
39533         if(this.maskRe || (this.vtype && this.disableKeyFilter !== true && (this.maskRe = Roo.form.VTypes[this.vtype+'Mask']))){
39534             this.el.on("keypress", this.filterKeys, this);
39535         }
39536         if(this.grow){
39537             this.el.on("keyup", this.onKeyUp,  this, {buffer:50});
39538             this.el.on("click", this.autoSize,  this);
39539         }
39540         if(this.el.is('input[type=password]') && Roo.isSafari){
39541             this.el.on('keydown', this.SafariOnKeyDown, this);
39542         }
39543     },
39544
39545     processValue : function(value){
39546         if(this.stripCharsRe){
39547             var newValue = value.replace(this.stripCharsRe, '');
39548             if(newValue !== value){
39549                 this.setRawValue(newValue);
39550                 return newValue;
39551             }
39552         }
39553         return value;
39554     },
39555
39556     filterValidation : function(e){
39557         if(!e.isNavKeyPress()){
39558             this.validationTask.delay(this.validationDelay);
39559         }
39560     },
39561
39562     // private
39563     onKeyUp : function(e){
39564         if(!e.isNavKeyPress()){
39565             this.autoSize();
39566         }
39567     },
39568     // private - clean the leading white space
39569     cleanLeadingSpace : function(e)
39570     {
39571         if ( this.inputType == 'file') {
39572             return;
39573         }
39574         
39575         this.setValue((this.getValue() + '').replace(/^\s+/,''));
39576     },
39577     /**
39578      * Resets the current field value to the originally-loaded value and clears any validation messages.
39579      *  
39580      */
39581     reset : function(){
39582         Roo.form.TextField.superclass.reset.call(this);
39583        
39584     }, 
39585     // private
39586     preFocus : function(){
39587         
39588         if(this.selectOnFocus){
39589             this.el.dom.select();
39590         }
39591     },
39592
39593     
39594     // private
39595     filterKeys : function(e){
39596         var k = e.getKey();
39597         if(!Roo.isIE && (e.isNavKeyPress() || k == e.BACKSPACE || (k == e.DELETE && e.button == -1))){
39598             return;
39599         }
39600         var c = e.getCharCode(), cc = String.fromCharCode(c);
39601         if(Roo.isIE && (e.isSpecialKey() || !cc)){
39602             return;
39603         }
39604         if(!this.maskRe.test(cc)){
39605             e.stopEvent();
39606         }
39607     },
39608
39609     setValue : function(v){
39610         
39611         Roo.form.TextField.superclass.setValue.apply(this, arguments);
39612         
39613         this.autoSize();
39614     },
39615
39616     /**
39617      * Validates a value according to the field's validation rules and marks the field as invalid
39618      * if the validation fails
39619      * @param {Mixed} value The value to validate
39620      * @return {Boolean} True if the value is valid, else false
39621      */
39622     validateValue : function(value){
39623         if(value.length < 1)  { // if it's blank
39624              if(this.allowBlank){
39625                 this.clearInvalid();
39626                 return true;
39627              }else{
39628                 this.markInvalid(this.blankText);
39629                 return false;
39630              }
39631         }
39632         if(value.length < this.minLength){
39633             this.markInvalid(String.format(this.minLengthText, this.minLength));
39634             return false;
39635         }
39636         if(value.length > this.maxLength){
39637             this.markInvalid(String.format(this.maxLengthText, this.maxLength));
39638             return false;
39639         }
39640         if(this.vtype){
39641             var vt = Roo.form.VTypes;
39642             if(!vt[this.vtype](value, this)){
39643                 this.markInvalid(this.vtypeText || vt[this.vtype +'Text']);
39644                 return false;
39645             }
39646         }
39647         if(typeof this.validator == "function"){
39648             var msg = this.validator(value);
39649             if(msg !== true){
39650                 this.markInvalid(msg);
39651                 return false;
39652             }
39653         }
39654         if(this.regex && !this.regex.test(value)){
39655             this.markInvalid(this.regexText);
39656             return false;
39657         }
39658         return true;
39659     },
39660
39661     /**
39662      * Selects text in this field
39663      * @param {Number} start (optional) The index where the selection should start (defaults to 0)
39664      * @param {Number} end (optional) The index where the selection should end (defaults to the text length)
39665      */
39666     selectText : function(start, end){
39667         var v = this.getRawValue();
39668         if(v.length > 0){
39669             start = start === undefined ? 0 : start;
39670             end = end === undefined ? v.length : end;
39671             var d = this.el.dom;
39672             if(d.setSelectionRange){
39673                 d.setSelectionRange(start, end);
39674             }else if(d.createTextRange){
39675                 var range = d.createTextRange();
39676                 range.moveStart("character", start);
39677                 range.moveEnd("character", v.length-end);
39678                 range.select();
39679             }
39680         }
39681     },
39682
39683     /**
39684      * Automatically grows the field to accomodate the width of the text up to the maximum field width allowed.
39685      * This only takes effect if grow = true, and fires the autosize event.
39686      */
39687     autoSize : function(){
39688         if(!this.grow || !this.rendered){
39689             return;
39690         }
39691         if(!this.metrics){
39692             this.metrics = Roo.util.TextMetrics.createInstance(this.el);
39693         }
39694         var el = this.el;
39695         var v = el.dom.value;
39696         var d = document.createElement('div');
39697         d.appendChild(document.createTextNode(v));
39698         v = d.innerHTML;
39699         d = null;
39700         v += "&#160;";
39701         var w = Math.min(this.growMax, Math.max(this.metrics.getWidth(v) + /* add extra padding */ 10, this.growMin));
39702         this.el.setWidth(w);
39703         this.fireEvent("autosize", this, w);
39704     },
39705     
39706     // private
39707     SafariOnKeyDown : function(event)
39708     {
39709         // this is a workaround for a password hang bug on chrome/ webkit.
39710         
39711         var isSelectAll = false;
39712         
39713         if(this.el.dom.selectionEnd > 0){
39714             isSelectAll = (this.el.dom.selectionEnd - this.el.dom.selectionStart - this.getValue().length == 0) ? true : false;
39715         }
39716         if(((event.getKey() == 8 || event.getKey() == 46) && this.getValue().length ==1)){ // backspace and delete key
39717             event.preventDefault();
39718             this.setValue('');
39719             return;
39720         }
39721         
39722         if(isSelectAll && event.getCharCode() > 31){ // backspace and delete key
39723             
39724             event.preventDefault();
39725             // this is very hacky as keydown always get's upper case.
39726             
39727             var cc = String.fromCharCode(event.getCharCode());
39728             
39729             
39730             this.setValue( event.shiftKey ?  cc : cc.toLowerCase());
39731             
39732         }
39733         
39734         
39735     }
39736 });/*
39737  * Based on:
39738  * Ext JS Library 1.1.1
39739  * Copyright(c) 2006-2007, Ext JS, LLC.
39740  *
39741  * Originally Released Under LGPL - original licence link has changed is not relivant.
39742  *
39743  * Fork - LGPL
39744  * <script type="text/javascript">
39745  */
39746  
39747 /**
39748  * @class Roo.form.Hidden
39749  * @extends Roo.form.TextField
39750  * Simple Hidden element used on forms 
39751  * 
39752  * usage: form.add(new Roo.form.HiddenField({ 'name' : 'test1' }));
39753  * 
39754  * @constructor
39755  * Creates a new Hidden form element.
39756  * @param {Object} config Configuration options
39757  */
39758
39759
39760
39761 // easy hidden field...
39762 Roo.form.Hidden = function(config){
39763     Roo.form.Hidden.superclass.constructor.call(this, config);
39764 };
39765   
39766 Roo.extend(Roo.form.Hidden, Roo.form.TextField, {
39767     fieldLabel:      '',
39768     inputType:      'hidden',
39769     width:          50,
39770     allowBlank:     true,
39771     labelSeparator: '',
39772     hidden:         true,
39773     itemCls :       'x-form-item-display-none'
39774
39775
39776 });
39777
39778
39779 /*
39780  * Based on:
39781  * Ext JS Library 1.1.1
39782  * Copyright(c) 2006-2007, Ext JS, LLC.
39783  *
39784  * Originally Released Under LGPL - original licence link has changed is not relivant.
39785  *
39786  * Fork - LGPL
39787  * <script type="text/javascript">
39788  */
39789  
39790 /**
39791  * @class Roo.form.TriggerField
39792  * @extends Roo.form.TextField
39793  * Provides a convenient wrapper for TextFields that adds a clickable trigger button (looks like a combobox by default).
39794  * The trigger has no default action, so you must assign a function to implement the trigger click handler by
39795  * overriding {@link #onTriggerClick}. You can create a TriggerField directly, as it renders exactly like a combobox
39796  * for which you can provide a custom implementation.  For example:
39797  * <pre><code>
39798 var trigger = new Roo.form.TriggerField();
39799 trigger.onTriggerClick = myTriggerFn;
39800 trigger.applyTo('my-field');
39801 </code></pre>
39802  *
39803  * However, in general you will most likely want to use TriggerField as the base class for a reusable component.
39804  * {@link Roo.form.DateField} and {@link Roo.form.ComboBox} are perfect examples of this.
39805  * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
39806  * class 'x-form-trigger' by default and triggerClass will be <b>appended</b> if specified.
39807  * @constructor
39808  * Create a new TriggerField.
39809  * @param {Object} config Configuration options (valid {@Roo.form.TextField} config options will also be applied
39810  * to the base TextField)
39811  */
39812 Roo.form.TriggerField = function(config){
39813     this.mimicing = false;
39814     Roo.form.TriggerField.superclass.constructor.call(this, config);
39815 };
39816
39817 Roo.extend(Roo.form.TriggerField, Roo.form.TextField,  {
39818     /**
39819      * @cfg {String} triggerClass A CSS class to apply to the trigger
39820      */
39821     /**
39822      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
39823      * {tag: "input", type: "text", size: "16", autocomplete: "off"})
39824      */
39825     defaultAutoCreate : {tag: "input", type: "text", size: "16", autocomplete: "new-password"},
39826     /**
39827      * @cfg {Boolean} hideTrigger True to hide the trigger element and display only the base text field (defaults to false)
39828      */
39829     hideTrigger:false,
39830
39831     /** @cfg {Boolean} grow @hide */
39832     /** @cfg {Number} growMin @hide */
39833     /** @cfg {Number} growMax @hide */
39834
39835     /**
39836      * @hide 
39837      * @method
39838      */
39839     autoSize: Roo.emptyFn,
39840     // private
39841     monitorTab : true,
39842     // private
39843     deferHeight : true,
39844
39845     
39846     actionMode : 'wrap',
39847     // private
39848     onResize : function(w, h){
39849         Roo.form.TriggerField.superclass.onResize.apply(this, arguments);
39850         if(typeof w == 'number'){
39851             var x = w - this.trigger.getWidth();
39852             this.el.setWidth(this.adjustWidth('input', x));
39853             this.trigger.setStyle('left', x+'px');
39854         }
39855     },
39856
39857     // private
39858     adjustSize : Roo.BoxComponent.prototype.adjustSize,
39859
39860     // private
39861     getResizeEl : function(){
39862         return this.wrap;
39863     },
39864
39865     // private
39866     getPositionEl : function(){
39867         return this.wrap;
39868     },
39869
39870     // private
39871     alignErrorIcon : function(){
39872         this.errorIcon.alignTo(this.wrap, 'tl-tr', [2, 0]);
39873     },
39874
39875     // private
39876     onRender : function(ct, position){
39877         Roo.form.TriggerField.superclass.onRender.call(this, ct, position);
39878         this.wrap = this.el.wrap({cls: "x-form-field-wrap"});
39879         this.trigger = this.wrap.createChild(this.triggerConfig ||
39880                 {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.triggerClass});
39881         if(this.hideTrigger){
39882             this.trigger.setDisplayed(false);
39883         }
39884         this.initTrigger();
39885         if(!this.width){
39886             this.wrap.setWidth(this.el.getWidth()+this.trigger.getWidth());
39887         }
39888     },
39889
39890     // private
39891     initTrigger : function(){
39892         this.trigger.on("click", this.onTriggerClick, this, {preventDefault:true});
39893         this.trigger.addClassOnOver('x-form-trigger-over');
39894         this.trigger.addClassOnClick('x-form-trigger-click');
39895     },
39896
39897     // private
39898     onDestroy : function(){
39899         if(this.trigger){
39900             this.trigger.removeAllListeners();
39901             this.trigger.remove();
39902         }
39903         if(this.wrap){
39904             this.wrap.remove();
39905         }
39906         Roo.form.TriggerField.superclass.onDestroy.call(this);
39907     },
39908
39909     // private
39910     onFocus : function(){
39911         Roo.form.TriggerField.superclass.onFocus.call(this);
39912         if(!this.mimicing){
39913             this.wrap.addClass('x-trigger-wrap-focus');
39914             this.mimicing = true;
39915             Roo.get(Roo.isIE ? document.body : document).on("mousedown", this.mimicBlur, this);
39916             if(this.monitorTab){
39917                 this.el.on("keydown", this.checkTab, this);
39918             }
39919         }
39920     },
39921
39922     // private
39923     checkTab : function(e){
39924         if(e.getKey() == e.TAB){
39925             this.triggerBlur();
39926         }
39927     },
39928
39929     // private
39930     onBlur : function(){
39931         // do nothing
39932     },
39933
39934     // private
39935     mimicBlur : function(e, t){
39936         if(!this.wrap.contains(t) && this.validateBlur()){
39937             this.triggerBlur();
39938         }
39939     },
39940
39941     // private
39942     triggerBlur : function(){
39943         this.mimicing = false;
39944         Roo.get(Roo.isIE ? document.body : document).un("mousedown", this.mimicBlur);
39945         if(this.monitorTab){
39946             this.el.un("keydown", this.checkTab, this);
39947         }
39948         this.wrap.removeClass('x-trigger-wrap-focus');
39949         Roo.form.TriggerField.superclass.onBlur.call(this);
39950     },
39951
39952     // private
39953     // This should be overriden by any subclass that needs to check whether or not the field can be blurred.
39954     validateBlur : function(e, t){
39955         return true;
39956     },
39957
39958     // private
39959     onDisable : function(){
39960         Roo.form.TriggerField.superclass.onDisable.call(this);
39961         if(this.wrap){
39962             this.wrap.addClass('x-item-disabled');
39963         }
39964     },
39965
39966     // private
39967     onEnable : function(){
39968         Roo.form.TriggerField.superclass.onEnable.call(this);
39969         if(this.wrap){
39970             this.wrap.removeClass('x-item-disabled');
39971         }
39972     },
39973
39974     // private
39975     onShow : function(){
39976         var ae = this.getActionEl();
39977         
39978         if(ae){
39979             ae.dom.style.display = '';
39980             ae.dom.style.visibility = 'visible';
39981         }
39982     },
39983
39984     // private
39985     
39986     onHide : function(){
39987         var ae = this.getActionEl();
39988         ae.dom.style.display = 'none';
39989     },
39990
39991     /**
39992      * The function that should handle the trigger's click event.  This method does nothing by default until overridden
39993      * by an implementing function.
39994      * @method
39995      * @param {EventObject} e
39996      */
39997     onTriggerClick : Roo.emptyFn
39998 });
39999
40000 // TwinTriggerField is not a public class to be used directly.  It is meant as an abstract base class
40001 // to be extended by an implementing class.  For an example of implementing this class, see the custom
40002 // SearchField implementation here: http://extjs.com/deploy/ext/examples/form/custom.html
40003 Roo.form.TwinTriggerField = Roo.extend(Roo.form.TriggerField, {
40004     initComponent : function(){
40005         Roo.form.TwinTriggerField.superclass.initComponent.call(this);
40006
40007         this.triggerConfig = {
40008             tag:'span', cls:'x-form-twin-triggers', cn:[
40009             {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.trigger1Class},
40010             {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.trigger2Class}
40011         ]};
40012     },
40013
40014     getTrigger : function(index){
40015         return this.triggers[index];
40016     },
40017
40018     initTrigger : function(){
40019         var ts = this.trigger.select('.x-form-trigger', true);
40020         this.wrap.setStyle('overflow', 'hidden');
40021         var triggerField = this;
40022         ts.each(function(t, all, index){
40023             t.hide = function(){
40024                 var w = triggerField.wrap.getWidth();
40025                 this.dom.style.display = 'none';
40026                 triggerField.el.setWidth(w-triggerField.trigger.getWidth());
40027             };
40028             t.show = function(){
40029                 var w = triggerField.wrap.getWidth();
40030                 this.dom.style.display = '';
40031                 triggerField.el.setWidth(w-triggerField.trigger.getWidth());
40032             };
40033             var triggerIndex = 'Trigger'+(index+1);
40034
40035             if(this['hide'+triggerIndex]){
40036                 t.dom.style.display = 'none';
40037             }
40038             t.on("click", this['on'+triggerIndex+'Click'], this, {preventDefault:true});
40039             t.addClassOnOver('x-form-trigger-over');
40040             t.addClassOnClick('x-form-trigger-click');
40041         }, this);
40042         this.triggers = ts.elements;
40043     },
40044
40045     onTrigger1Click : Roo.emptyFn,
40046     onTrigger2Click : Roo.emptyFn
40047 });/*
40048  * Based on:
40049  * Ext JS Library 1.1.1
40050  * Copyright(c) 2006-2007, Ext JS, LLC.
40051  *
40052  * Originally Released Under LGPL - original licence link has changed is not relivant.
40053  *
40054  * Fork - LGPL
40055  * <script type="text/javascript">
40056  */
40057  
40058 /**
40059  * @class Roo.form.TextArea
40060  * @extends Roo.form.TextField
40061  * Multiline text field.  Can be used as a direct replacement for traditional textarea fields, plus adds
40062  * support for auto-sizing.
40063  * @constructor
40064  * Creates a new TextArea
40065  * @param {Object} config Configuration options
40066  */
40067 Roo.form.TextArea = function(config){
40068     Roo.form.TextArea.superclass.constructor.call(this, config);
40069     // these are provided exchanges for backwards compat
40070     // minHeight/maxHeight were replaced by growMin/growMax to be
40071     // compatible with TextField growing config values
40072     if(this.minHeight !== undefined){
40073         this.growMin = this.minHeight;
40074     }
40075     if(this.maxHeight !== undefined){
40076         this.growMax = this.maxHeight;
40077     }
40078 };
40079
40080 Roo.extend(Roo.form.TextArea, Roo.form.TextField,  {
40081     /**
40082      * @cfg {Number} growMin The minimum height to allow when grow = true (defaults to 60)
40083      */
40084     growMin : 60,
40085     /**
40086      * @cfg {Number} growMax The maximum height to allow when grow = true (defaults to 1000)
40087      */
40088     growMax: 1000,
40089     /**
40090      * @cfg {Boolean} preventScrollbars True to prevent scrollbars from appearing regardless of how much text is
40091      * in the field (equivalent to setting overflow: hidden, defaults to false)
40092      */
40093     preventScrollbars: false,
40094     /**
40095      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
40096      * {tag: "textarea", style: "width:300px;height:60px;", autocomplete: "off"})
40097      */
40098
40099     // private
40100     onRender : function(ct, position){
40101         if(!this.el){
40102             this.defaultAutoCreate = {
40103                 tag: "textarea",
40104                 style:"width:300px;height:60px;",
40105                 autocomplete: "new-password"
40106             };
40107         }
40108         Roo.form.TextArea.superclass.onRender.call(this, ct, position);
40109         if(this.grow){
40110             this.textSizeEl = Roo.DomHelper.append(document.body, {
40111                 tag: "pre", cls: "x-form-grow-sizer"
40112             });
40113             if(this.preventScrollbars){
40114                 this.el.setStyle("overflow", "hidden");
40115             }
40116             this.el.setHeight(this.growMin);
40117         }
40118     },
40119
40120     onDestroy : function(){
40121         if(this.textSizeEl){
40122             this.textSizeEl.parentNode.removeChild(this.textSizeEl);
40123         }
40124         Roo.form.TextArea.superclass.onDestroy.call(this);
40125     },
40126
40127     // private
40128     onKeyUp : function(e){
40129         if(!e.isNavKeyPress() || e.getKey() == e.ENTER){
40130             this.autoSize();
40131         }
40132     },
40133
40134     /**
40135      * Automatically grows the field to accomodate the height of the text up to the maximum field height allowed.
40136      * This only takes effect if grow = true, and fires the autosize event if the height changes.
40137      */
40138     autoSize : function(){
40139         if(!this.grow || !this.textSizeEl){
40140             return;
40141         }
40142         var el = this.el;
40143         var v = el.dom.value;
40144         var ts = this.textSizeEl;
40145
40146         ts.innerHTML = '';
40147         ts.appendChild(document.createTextNode(v));
40148         v = ts.innerHTML;
40149
40150         Roo.fly(ts).setWidth(this.el.getWidth());
40151         if(v.length < 1){
40152             v = "&#160;&#160;";
40153         }else{
40154             if(Roo.isIE){
40155                 v = v.replace(/\n/g, '<p>&#160;</p>');
40156             }
40157             v += "&#160;\n&#160;";
40158         }
40159         ts.innerHTML = v;
40160         var h = Math.min(this.growMax, Math.max(ts.offsetHeight, this.growMin));
40161         if(h != this.lastHeight){
40162             this.lastHeight = h;
40163             this.el.setHeight(h);
40164             this.fireEvent("autosize", this, h);
40165         }
40166     }
40167 });/*
40168  * Based on:
40169  * Ext JS Library 1.1.1
40170  * Copyright(c) 2006-2007, Ext JS, LLC.
40171  *
40172  * Originally Released Under LGPL - original licence link has changed is not relivant.
40173  *
40174  * Fork - LGPL
40175  * <script type="text/javascript">
40176  */
40177  
40178
40179 /**
40180  * @class Roo.form.NumberField
40181  * @extends Roo.form.TextField
40182  * Numeric text field that provides automatic keystroke filtering and numeric validation.
40183  * @constructor
40184  * Creates a new NumberField
40185  * @param {Object} config Configuration options
40186  */
40187 Roo.form.NumberField = function(config){
40188     Roo.form.NumberField.superclass.constructor.call(this, config);
40189 };
40190
40191 Roo.extend(Roo.form.NumberField, Roo.form.TextField,  {
40192     /**
40193      * @cfg {String} fieldClass The default CSS class for the field (defaults to "x-form-field x-form-num-field")
40194      */
40195     fieldClass: "x-form-field x-form-num-field",
40196     /**
40197      * @cfg {Boolean} allowDecimals False to disallow decimal values (defaults to true)
40198      */
40199     allowDecimals : true,
40200     /**
40201      * @cfg {String} decimalSeparator Character(s) to allow as the decimal separator (defaults to '.')
40202      */
40203     decimalSeparator : ".",
40204     /**
40205      * @cfg {Number} decimalPrecision The maximum precision to display after the decimal separator (defaults to 2)
40206      */
40207     decimalPrecision : 2,
40208     /**
40209      * @cfg {Boolean} allowNegative False to prevent entering a negative sign (defaults to true)
40210      */
40211     allowNegative : true,
40212     /**
40213      * @cfg {Number} minValue The minimum allowed value (defaults to Number.NEGATIVE_INFINITY)
40214      */
40215     minValue : Number.NEGATIVE_INFINITY,
40216     /**
40217      * @cfg {Number} maxValue The maximum allowed value (defaults to Number.MAX_VALUE)
40218      */
40219     maxValue : Number.MAX_VALUE,
40220     /**
40221      * @cfg {String} minText Error text to display if the minimum value validation fails (defaults to "The minimum value for this field is {minValue}")
40222      */
40223     minText : "The minimum value for this field is {0}",
40224     /**
40225      * @cfg {String} maxText Error text to display if the maximum value validation fails (defaults to "The maximum value for this field is {maxValue}")
40226      */
40227     maxText : "The maximum value for this field is {0}",
40228     /**
40229      * @cfg {String} nanText Error text to display if the value is not a valid number.  For example, this can happen
40230      * if a valid character like '.' or '-' is left in the field with no number (defaults to "{value} is not a valid number")
40231      */
40232     nanText : "{0} is not a valid number",
40233
40234     // private
40235     initEvents : function(){
40236         Roo.form.NumberField.superclass.initEvents.call(this);
40237         var allowed = "0123456789";
40238         if(this.allowDecimals){
40239             allowed += this.decimalSeparator;
40240         }
40241         if(this.allowNegative){
40242             allowed += "-";
40243         }
40244         this.stripCharsRe = new RegExp('[^'+allowed+']', 'gi');
40245         var keyPress = function(e){
40246             var k = e.getKey();
40247             if(!Roo.isIE && (e.isSpecialKey() || k == e.BACKSPACE || k == e.DELETE)){
40248                 return;
40249             }
40250             var c = e.getCharCode();
40251             if(allowed.indexOf(String.fromCharCode(c)) === -1){
40252                 e.stopEvent();
40253             }
40254         };
40255         this.el.on("keypress", keyPress, this);
40256     },
40257
40258     // private
40259     validateValue : function(value){
40260         if(!Roo.form.NumberField.superclass.validateValue.call(this, value)){
40261             return false;
40262         }
40263         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40264              return true;
40265         }
40266         var num = this.parseValue(value);
40267         if(isNaN(num)){
40268             this.markInvalid(String.format(this.nanText, value));
40269             return false;
40270         }
40271         if(num < this.minValue){
40272             this.markInvalid(String.format(this.minText, this.minValue));
40273             return false;
40274         }
40275         if(num > this.maxValue){
40276             this.markInvalid(String.format(this.maxText, this.maxValue));
40277             return false;
40278         }
40279         return true;
40280     },
40281
40282     getValue : function(){
40283         return this.fixPrecision(this.parseValue(Roo.form.NumberField.superclass.getValue.call(this)));
40284     },
40285
40286     // private
40287     parseValue : function(value){
40288         value = parseFloat(String(value).replace(this.decimalSeparator, "."));
40289         return isNaN(value) ? '' : value;
40290     },
40291
40292     // private
40293     fixPrecision : function(value){
40294         var nan = isNaN(value);
40295         if(!this.allowDecimals || this.decimalPrecision == -1 || nan || !value){
40296             return nan ? '' : value;
40297         }
40298         return parseFloat(value).toFixed(this.decimalPrecision);
40299     },
40300
40301     setValue : function(v){
40302         v = this.fixPrecision(v);
40303         Roo.form.NumberField.superclass.setValue.call(this, String(v).replace(".", this.decimalSeparator));
40304     },
40305
40306     // private
40307     decimalPrecisionFcn : function(v){
40308         return Math.floor(v);
40309     },
40310
40311     beforeBlur : function(){
40312         var v = this.parseValue(this.getRawValue());
40313         if(v){
40314             this.setValue(v);
40315         }
40316     }
40317 });/*
40318  * Based on:
40319  * Ext JS Library 1.1.1
40320  * Copyright(c) 2006-2007, Ext JS, LLC.
40321  *
40322  * Originally Released Under LGPL - original licence link has changed is not relivant.
40323  *
40324  * Fork - LGPL
40325  * <script type="text/javascript">
40326  */
40327  
40328 /**
40329  * @class Roo.form.DateField
40330  * @extends Roo.form.TriggerField
40331  * Provides a date input field with a {@link Roo.DatePicker} dropdown and automatic date validation.
40332 * @constructor
40333 * Create a new DateField
40334 * @param {Object} config
40335  */
40336 Roo.form.DateField = function(config)
40337 {
40338     Roo.form.DateField.superclass.constructor.call(this, config);
40339     
40340       this.addEvents({
40341          
40342         /**
40343          * @event select
40344          * Fires when a date is selected
40345              * @param {Roo.form.DateField} combo This combo box
40346              * @param {Date} date The date selected
40347              */
40348         'select' : true
40349          
40350     });
40351     
40352     
40353     if(typeof this.minValue == "string") {
40354         this.minValue = this.parseDate(this.minValue);
40355     }
40356     if(typeof this.maxValue == "string") {
40357         this.maxValue = this.parseDate(this.maxValue);
40358     }
40359     this.ddMatch = null;
40360     if(this.disabledDates){
40361         var dd = this.disabledDates;
40362         var re = "(?:";
40363         for(var i = 0; i < dd.length; i++){
40364             re += dd[i];
40365             if(i != dd.length-1) {
40366                 re += "|";
40367             }
40368         }
40369         this.ddMatch = new RegExp(re + ")");
40370     }
40371 };
40372
40373 Roo.extend(Roo.form.DateField, Roo.form.TriggerField,  {
40374     /**
40375      * @cfg {String} format
40376      * The default date format string which can be overriden for localization support.  The format must be
40377      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
40378      */
40379     format : "m/d/y",
40380     /**
40381      * @cfg {String} altFormats
40382      * Multiple date formats separated by "|" to try when parsing a user input value and it doesn't match the defined
40383      * format (defaults to 'm/d/Y|m-d-y|m-d-Y|m/d|m-d|d').
40384      */
40385     altFormats : "m/d/Y|m-d-y|m-d-Y|m/d|m-d|md|mdy|mdY|d",
40386     /**
40387      * @cfg {Array} disabledDays
40388      * An array of days to disable, 0 based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
40389      */
40390     disabledDays : null,
40391     /**
40392      * @cfg {String} disabledDaysText
40393      * The tooltip to display when the date falls on a disabled day (defaults to 'Disabled')
40394      */
40395     disabledDaysText : "Disabled",
40396     /**
40397      * @cfg {Array} disabledDates
40398      * An array of "dates" to disable, as strings. These strings will be used to build a dynamic regular
40399      * expression so they are very powerful. Some examples:
40400      * <ul>
40401      * <li>["03/08/2003", "09/16/2003"] would disable those exact dates</li>
40402      * <li>["03/08", "09/16"] would disable those days for every year</li>
40403      * <li>["^03/08"] would only match the beginning (useful if you are using short years)</li>
40404      * <li>["03/../2006"] would disable every day in March 2006</li>
40405      * <li>["^03"] would disable every day in every March</li>
40406      * </ul>
40407      * In order to support regular expressions, if you are using a date format that has "." in it, you will have to
40408      * escape the dot when restricting dates. For example: ["03\\.08\\.03"].
40409      */
40410     disabledDates : null,
40411     /**
40412      * @cfg {String} disabledDatesText
40413      * The tooltip text to display when the date falls on a disabled date (defaults to 'Disabled')
40414      */
40415     disabledDatesText : "Disabled",
40416     /**
40417      * @cfg {Date/String} minValue
40418      * The minimum allowed date. Can be either a Javascript date object or a string date in a
40419      * valid format (defaults to null).
40420      */
40421     minValue : null,
40422     /**
40423      * @cfg {Date/String} maxValue
40424      * The maximum allowed date. Can be either a Javascript date object or a string date in a
40425      * valid format (defaults to null).
40426      */
40427     maxValue : null,
40428     /**
40429      * @cfg {String} minText
40430      * The error text to display when the date in the cell is before minValue (defaults to
40431      * 'The date in this field must be after {minValue}').
40432      */
40433     minText : "The date in this field must be equal to or after {0}",
40434     /**
40435      * @cfg {String} maxText
40436      * The error text to display when the date in the cell is after maxValue (defaults to
40437      * 'The date in this field must be before {maxValue}').
40438      */
40439     maxText : "The date in this field must be equal to or before {0}",
40440     /**
40441      * @cfg {String} invalidText
40442      * The error text to display when the date in the field is invalid (defaults to
40443      * '{value} is not a valid date - it must be in the format {format}').
40444      */
40445     invalidText : "{0} is not a valid date - it must be in the format {1}",
40446     /**
40447      * @cfg {String} triggerClass
40448      * An additional CSS class used to style the trigger button.  The trigger will always get the
40449      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-date-trigger'
40450      * which displays a calendar icon).
40451      */
40452     triggerClass : 'x-form-date-trigger',
40453     
40454
40455     /**
40456      * @cfg {Boolean} useIso
40457      * if enabled, then the date field will use a hidden field to store the 
40458      * real value as iso formated date. default (false)
40459      */ 
40460     useIso : false,
40461     /**
40462      * @cfg {String/Object} autoCreate
40463      * A DomHelper element spec, or true for a default element spec (defaults to
40464      * {tag: "input", type: "text", size: "10", autocomplete: "off"})
40465      */ 
40466     // private
40467     defaultAutoCreate : {tag: "input", type: "text", size: "10", autocomplete: "off"},
40468     
40469     // private
40470     hiddenField: false,
40471     
40472     onRender : function(ct, position)
40473     {
40474         Roo.form.DateField.superclass.onRender.call(this, ct, position);
40475         if (this.useIso) {
40476             //this.el.dom.removeAttribute('name'); 
40477             Roo.log("Changing name?");
40478             this.el.dom.setAttribute('name', this.name + '____hidden___' ); 
40479             this.hiddenField = this.el.insertSibling({ tag:'input', type:'hidden', name: this.name },
40480                     'before', true);
40481             this.hiddenField.value = this.value ? this.formatDate(this.value, 'Y-m-d') : '';
40482             // prevent input submission
40483             this.hiddenName = this.name;
40484         }
40485             
40486             
40487     },
40488     
40489     // private
40490     validateValue : function(value)
40491     {
40492         value = this.formatDate(value);
40493         if(!Roo.form.DateField.superclass.validateValue.call(this, value)){
40494             Roo.log('super failed');
40495             return false;
40496         }
40497         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40498              return true;
40499         }
40500         var svalue = value;
40501         value = this.parseDate(value);
40502         if(!value){
40503             Roo.log('parse date failed' + svalue);
40504             this.markInvalid(String.format(this.invalidText, svalue, this.format));
40505             return false;
40506         }
40507         var time = value.getTime();
40508         if(this.minValue && time < this.minValue.getTime()){
40509             this.markInvalid(String.format(this.minText, this.formatDate(this.minValue)));
40510             return false;
40511         }
40512         if(this.maxValue && time > this.maxValue.getTime()){
40513             this.markInvalid(String.format(this.maxText, this.formatDate(this.maxValue)));
40514             return false;
40515         }
40516         if(this.disabledDays){
40517             var day = value.getDay();
40518             for(var i = 0; i < this.disabledDays.length; i++) {
40519                 if(day === this.disabledDays[i]){
40520                     this.markInvalid(this.disabledDaysText);
40521                     return false;
40522                 }
40523             }
40524         }
40525         var fvalue = this.formatDate(value);
40526         if(this.ddMatch && this.ddMatch.test(fvalue)){
40527             this.markInvalid(String.format(this.disabledDatesText, fvalue));
40528             return false;
40529         }
40530         return true;
40531     },
40532
40533     // private
40534     // Provides logic to override the default TriggerField.validateBlur which just returns true
40535     validateBlur : function(){
40536         return !this.menu || !this.menu.isVisible();
40537     },
40538     
40539     getName: function()
40540     {
40541         // returns hidden if it's set..
40542         if (!this.rendered) {return ''};
40543         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
40544         
40545     },
40546
40547     /**
40548      * Returns the current date value of the date field.
40549      * @return {Date} The date value
40550      */
40551     getValue : function(){
40552         
40553         return  this.hiddenField ?
40554                 this.hiddenField.value :
40555                 this.parseDate(Roo.form.DateField.superclass.getValue.call(this)) || "";
40556     },
40557
40558     /**
40559      * Sets the value of the date field.  You can pass a date object or any string that can be parsed into a valid
40560      * date, using DateField.format as the date format, according to the same rules as {@link Date#parseDate}
40561      * (the default format used is "m/d/y").
40562      * <br />Usage:
40563      * <pre><code>
40564 //All of these calls set the same date value (May 4, 2006)
40565
40566 //Pass a date object:
40567 var dt = new Date('5/4/06');
40568 dateField.setValue(dt);
40569
40570 //Pass a date string (default format):
40571 dateField.setValue('5/4/06');
40572
40573 //Pass a date string (custom format):
40574 dateField.format = 'Y-m-d';
40575 dateField.setValue('2006-5-4');
40576 </code></pre>
40577      * @param {String/Date} date The date or valid date string
40578      */
40579     setValue : function(date){
40580         if (this.hiddenField) {
40581             this.hiddenField.value = this.formatDate(this.parseDate(date), 'Y-m-d');
40582         }
40583         Roo.form.DateField.superclass.setValue.call(this, this.formatDate(this.parseDate(date)));
40584         // make sure the value field is always stored as a date..
40585         this.value = this.parseDate(date);
40586         
40587         
40588     },
40589
40590     // private
40591     parseDate : function(value){
40592         if(!value || value instanceof Date){
40593             return value;
40594         }
40595         var v = Date.parseDate(value, this.format);
40596          if (!v && this.useIso) {
40597             v = Date.parseDate(value, 'Y-m-d');
40598         }
40599         if(!v && this.altFormats){
40600             if(!this.altFormatsArray){
40601                 this.altFormatsArray = this.altFormats.split("|");
40602             }
40603             for(var i = 0, len = this.altFormatsArray.length; i < len && !v; i++){
40604                 v = Date.parseDate(value, this.altFormatsArray[i]);
40605             }
40606         }
40607         return v;
40608     },
40609
40610     // private
40611     formatDate : function(date, fmt){
40612         return (!date || !(date instanceof Date)) ?
40613                date : date.dateFormat(fmt || this.format);
40614     },
40615
40616     // private
40617     menuListeners : {
40618         select: function(m, d){
40619             
40620             this.setValue(d);
40621             this.fireEvent('select', this, d);
40622         },
40623         show : function(){ // retain focus styling
40624             this.onFocus();
40625         },
40626         hide : function(){
40627             this.focus.defer(10, this);
40628             var ml = this.menuListeners;
40629             this.menu.un("select", ml.select,  this);
40630             this.menu.un("show", ml.show,  this);
40631             this.menu.un("hide", ml.hide,  this);
40632         }
40633     },
40634
40635     // private
40636     // Implements the default empty TriggerField.onTriggerClick function to display the DatePicker
40637     onTriggerClick : function(){
40638         if(this.disabled){
40639             return;
40640         }
40641         if(this.menu == null){
40642             this.menu = new Roo.menu.DateMenu();
40643         }
40644         Roo.apply(this.menu.picker,  {
40645             showClear: this.allowBlank,
40646             minDate : this.minValue,
40647             maxDate : this.maxValue,
40648             disabledDatesRE : this.ddMatch,
40649             disabledDatesText : this.disabledDatesText,
40650             disabledDays : this.disabledDays,
40651             disabledDaysText : this.disabledDaysText,
40652             format : this.useIso ? 'Y-m-d' : this.format,
40653             minText : String.format(this.minText, this.formatDate(this.minValue)),
40654             maxText : String.format(this.maxText, this.formatDate(this.maxValue))
40655         });
40656         this.menu.on(Roo.apply({}, this.menuListeners, {
40657             scope:this
40658         }));
40659         this.menu.picker.setValue(this.getValue() || new Date());
40660         this.menu.show(this.el, "tl-bl?");
40661     },
40662
40663     beforeBlur : function(){
40664         var v = this.parseDate(this.getRawValue());
40665         if(v){
40666             this.setValue(v);
40667         }
40668     },
40669
40670     /*@
40671      * overide
40672      * 
40673      */
40674     isDirty : function() {
40675         if(this.disabled) {
40676             return false;
40677         }
40678         
40679         if(typeof(this.startValue) === 'undefined'){
40680             return false;
40681         }
40682         
40683         return String(this.getValue()) !== String(this.startValue);
40684         
40685     },
40686     // @overide
40687     cleanLeadingSpace : function(e)
40688     {
40689        return;
40690     }
40691     
40692 });/*
40693  * Based on:
40694  * Ext JS Library 1.1.1
40695  * Copyright(c) 2006-2007, Ext JS, LLC.
40696  *
40697  * Originally Released Under LGPL - original licence link has changed is not relivant.
40698  *
40699  * Fork - LGPL
40700  * <script type="text/javascript">
40701  */
40702  
40703 /**
40704  * @class Roo.form.MonthField
40705  * @extends Roo.form.TriggerField
40706  * Provides a date input field with a {@link Roo.DatePicker} dropdown and automatic date validation.
40707 * @constructor
40708 * Create a new MonthField
40709 * @param {Object} config
40710  */
40711 Roo.form.MonthField = function(config){
40712     
40713     Roo.form.MonthField.superclass.constructor.call(this, config);
40714     
40715       this.addEvents({
40716          
40717         /**
40718          * @event select
40719          * Fires when a date is selected
40720              * @param {Roo.form.MonthFieeld} combo This combo box
40721              * @param {Date} date The date selected
40722              */
40723         'select' : true
40724          
40725     });
40726     
40727     
40728     if(typeof this.minValue == "string") {
40729         this.minValue = this.parseDate(this.minValue);
40730     }
40731     if(typeof this.maxValue == "string") {
40732         this.maxValue = this.parseDate(this.maxValue);
40733     }
40734     this.ddMatch = null;
40735     if(this.disabledDates){
40736         var dd = this.disabledDates;
40737         var re = "(?:";
40738         for(var i = 0; i < dd.length; i++){
40739             re += dd[i];
40740             if(i != dd.length-1) {
40741                 re += "|";
40742             }
40743         }
40744         this.ddMatch = new RegExp(re + ")");
40745     }
40746 };
40747
40748 Roo.extend(Roo.form.MonthField, Roo.form.TriggerField,  {
40749     /**
40750      * @cfg {String} format
40751      * The default date format string which can be overriden for localization support.  The format must be
40752      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
40753      */
40754     format : "M Y",
40755     /**
40756      * @cfg {String} altFormats
40757      * Multiple date formats separated by "|" to try when parsing a user input value and it doesn't match the defined
40758      * format (defaults to 'm/d/Y|m-d-y|m-d-Y|m/d|m-d|d').
40759      */
40760     altFormats : "M Y|m/Y|m-y|m-Y|my|mY",
40761     /**
40762      * @cfg {Array} disabledDays
40763      * An array of days to disable, 0 based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
40764      */
40765     disabledDays : [0,1,2,3,4,5,6],
40766     /**
40767      * @cfg {String} disabledDaysText
40768      * The tooltip to display when the date falls on a disabled day (defaults to 'Disabled')
40769      */
40770     disabledDaysText : "Disabled",
40771     /**
40772      * @cfg {Array} disabledDates
40773      * An array of "dates" to disable, as strings. These strings will be used to build a dynamic regular
40774      * expression so they are very powerful. Some examples:
40775      * <ul>
40776      * <li>["03/08/2003", "09/16/2003"] would disable those exact dates</li>
40777      * <li>["03/08", "09/16"] would disable those days for every year</li>
40778      * <li>["^03/08"] would only match the beginning (useful if you are using short years)</li>
40779      * <li>["03/../2006"] would disable every day in March 2006</li>
40780      * <li>["^03"] would disable every day in every March</li>
40781      * </ul>
40782      * In order to support regular expressions, if you are using a date format that has "." in it, you will have to
40783      * escape the dot when restricting dates. For example: ["03\\.08\\.03"].
40784      */
40785     disabledDates : null,
40786     /**
40787      * @cfg {String} disabledDatesText
40788      * The tooltip text to display when the date falls on a disabled date (defaults to 'Disabled')
40789      */
40790     disabledDatesText : "Disabled",
40791     /**
40792      * @cfg {Date/String} minValue
40793      * The minimum allowed date. Can be either a Javascript date object or a string date in a
40794      * valid format (defaults to null).
40795      */
40796     minValue : null,
40797     /**
40798      * @cfg {Date/String} maxValue
40799      * The maximum allowed date. Can be either a Javascript date object or a string date in a
40800      * valid format (defaults to null).
40801      */
40802     maxValue : null,
40803     /**
40804      * @cfg {String} minText
40805      * The error text to display when the date in the cell is before minValue (defaults to
40806      * 'The date in this field must be after {minValue}').
40807      */
40808     minText : "The date in this field must be equal to or after {0}",
40809     /**
40810      * @cfg {String} maxTextf
40811      * The error text to display when the date in the cell is after maxValue (defaults to
40812      * 'The date in this field must be before {maxValue}').
40813      */
40814     maxText : "The date in this field must be equal to or before {0}",
40815     /**
40816      * @cfg {String} invalidText
40817      * The error text to display when the date in the field is invalid (defaults to
40818      * '{value} is not a valid date - it must be in the format {format}').
40819      */
40820     invalidText : "{0} is not a valid date - it must be in the format {1}",
40821     /**
40822      * @cfg {String} triggerClass
40823      * An additional CSS class used to style the trigger button.  The trigger will always get the
40824      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-date-trigger'
40825      * which displays a calendar icon).
40826      */
40827     triggerClass : 'x-form-date-trigger',
40828     
40829
40830     /**
40831      * @cfg {Boolean} useIso
40832      * if enabled, then the date field will use a hidden field to store the 
40833      * real value as iso formated date. default (true)
40834      */ 
40835     useIso : true,
40836     /**
40837      * @cfg {String/Object} autoCreate
40838      * A DomHelper element spec, or true for a default element spec (defaults to
40839      * {tag: "input", type: "text", size: "10", autocomplete: "off"})
40840      */ 
40841     // private
40842     defaultAutoCreate : {tag: "input", type: "text", size: "10", autocomplete: "new-password"},
40843     
40844     // private
40845     hiddenField: false,
40846     
40847     hideMonthPicker : false,
40848     
40849     onRender : function(ct, position)
40850     {
40851         Roo.form.MonthField.superclass.onRender.call(this, ct, position);
40852         if (this.useIso) {
40853             this.el.dom.removeAttribute('name'); 
40854             this.hiddenField = this.el.insertSibling({ tag:'input', type:'hidden', name: this.name },
40855                     'before', true);
40856             this.hiddenField.value = this.value ? this.formatDate(this.value, 'Y-m-d') : '';
40857             // prevent input submission
40858             this.hiddenName = this.name;
40859         }
40860             
40861             
40862     },
40863     
40864     // private
40865     validateValue : function(value)
40866     {
40867         value = this.formatDate(value);
40868         if(!Roo.form.MonthField.superclass.validateValue.call(this, value)){
40869             return false;
40870         }
40871         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40872              return true;
40873         }
40874         var svalue = value;
40875         value = this.parseDate(value);
40876         if(!value){
40877             this.markInvalid(String.format(this.invalidText, svalue, this.format));
40878             return false;
40879         }
40880         var time = value.getTime();
40881         if(this.minValue && time < this.minValue.getTime()){
40882             this.markInvalid(String.format(this.minText, this.formatDate(this.minValue)));
40883             return false;
40884         }
40885         if(this.maxValue && time > this.maxValue.getTime()){
40886             this.markInvalid(String.format(this.maxText, this.formatDate(this.maxValue)));
40887             return false;
40888         }
40889         /*if(this.disabledDays){
40890             var day = value.getDay();
40891             for(var i = 0; i < this.disabledDays.length; i++) {
40892                 if(day === this.disabledDays[i]){
40893                     this.markInvalid(this.disabledDaysText);
40894                     return false;
40895                 }
40896             }
40897         }
40898         */
40899         var fvalue = this.formatDate(value);
40900         /*if(this.ddMatch && this.ddMatch.test(fvalue)){
40901             this.markInvalid(String.format(this.disabledDatesText, fvalue));
40902             return false;
40903         }
40904         */
40905         return true;
40906     },
40907
40908     // private
40909     // Provides logic to override the default TriggerField.validateBlur which just returns true
40910     validateBlur : function(){
40911         return !this.menu || !this.menu.isVisible();
40912     },
40913
40914     /**
40915      * Returns the current date value of the date field.
40916      * @return {Date} The date value
40917      */
40918     getValue : function(){
40919         
40920         
40921         
40922         return  this.hiddenField ?
40923                 this.hiddenField.value :
40924                 this.parseDate(Roo.form.MonthField.superclass.getValue.call(this)) || "";
40925     },
40926
40927     /**
40928      * Sets the value of the date field.  You can pass a date object or any string that can be parsed into a valid
40929      * date, using MonthField.format as the date format, according to the same rules as {@link Date#parseDate}
40930      * (the default format used is "m/d/y").
40931      * <br />Usage:
40932      * <pre><code>
40933 //All of these calls set the same date value (May 4, 2006)
40934
40935 //Pass a date object:
40936 var dt = new Date('5/4/06');
40937 monthField.setValue(dt);
40938
40939 //Pass a date string (default format):
40940 monthField.setValue('5/4/06');
40941
40942 //Pass a date string (custom format):
40943 monthField.format = 'Y-m-d';
40944 monthField.setValue('2006-5-4');
40945 </code></pre>
40946      * @param {String/Date} date The date or valid date string
40947      */
40948     setValue : function(date){
40949         Roo.log('month setValue' + date);
40950         // can only be first of month..
40951         
40952         var val = this.parseDate(date);
40953         
40954         if (this.hiddenField) {
40955             this.hiddenField.value = this.formatDate(this.parseDate(date), 'Y-m-d');
40956         }
40957         Roo.form.MonthField.superclass.setValue.call(this, this.formatDate(this.parseDate(date)));
40958         this.value = this.parseDate(date);
40959     },
40960
40961     // private
40962     parseDate : function(value){
40963         if(!value || value instanceof Date){
40964             value = value ? Date.parseDate(value.format('Y-m') + '-01', 'Y-m-d') : null;
40965             return value;
40966         }
40967         var v = Date.parseDate(value, this.format);
40968         if (!v && this.useIso) {
40969             v = Date.parseDate(value, 'Y-m-d');
40970         }
40971         if (v) {
40972             // 
40973             v = Date.parseDate(v.format('Y-m') +'-01', 'Y-m-d');
40974         }
40975         
40976         
40977         if(!v && this.altFormats){
40978             if(!this.altFormatsArray){
40979                 this.altFormatsArray = this.altFormats.split("|");
40980             }
40981             for(var i = 0, len = this.altFormatsArray.length; i < len && !v; i++){
40982                 v = Date.parseDate(value, this.altFormatsArray[i]);
40983             }
40984         }
40985         return v;
40986     },
40987
40988     // private
40989     formatDate : function(date, fmt){
40990         return (!date || !(date instanceof Date)) ?
40991                date : date.dateFormat(fmt || this.format);
40992     },
40993
40994     // private
40995     menuListeners : {
40996         select: function(m, d){
40997             this.setValue(d);
40998             this.fireEvent('select', this, d);
40999         },
41000         show : function(){ // retain focus styling
41001             this.onFocus();
41002         },
41003         hide : function(){
41004             this.focus.defer(10, this);
41005             var ml = this.menuListeners;
41006             this.menu.un("select", ml.select,  this);
41007             this.menu.un("show", ml.show,  this);
41008             this.menu.un("hide", ml.hide,  this);
41009         }
41010     },
41011     // private
41012     // Implements the default empty TriggerField.onTriggerClick function to display the DatePicker
41013     onTriggerClick : function(){
41014         if(this.disabled){
41015             return;
41016         }
41017         if(this.menu == null){
41018             this.menu = new Roo.menu.DateMenu();
41019            
41020         }
41021         
41022         Roo.apply(this.menu.picker,  {
41023             
41024             showClear: this.allowBlank,
41025             minDate : this.minValue,
41026             maxDate : this.maxValue,
41027             disabledDatesRE : this.ddMatch,
41028             disabledDatesText : this.disabledDatesText,
41029             
41030             format : this.useIso ? 'Y-m-d' : this.format,
41031             minText : String.format(this.minText, this.formatDate(this.minValue)),
41032             maxText : String.format(this.maxText, this.formatDate(this.maxValue))
41033             
41034         });
41035          this.menu.on(Roo.apply({}, this.menuListeners, {
41036             scope:this
41037         }));
41038        
41039         
41040         var m = this.menu;
41041         var p = m.picker;
41042         
41043         // hide month picker get's called when we called by 'before hide';
41044         
41045         var ignorehide = true;
41046         p.hideMonthPicker  = function(disableAnim){
41047             if (ignorehide) {
41048                 return;
41049             }
41050              if(this.monthPicker){
41051                 Roo.log("hideMonthPicker called");
41052                 if(disableAnim === true){
41053                     this.monthPicker.hide();
41054                 }else{
41055                     this.monthPicker.slideOut('t', {duration:.2});
41056                     p.setValue(new Date(m.picker.mpSelYear, m.picker.mpSelMonth, 1));
41057                     p.fireEvent("select", this, this.value);
41058                     m.hide();
41059                 }
41060             }
41061         }
41062         
41063         Roo.log('picker set value');
41064         Roo.log(this.getValue());
41065         p.setValue(this.getValue() ? this.parseDate(this.getValue()) : new Date());
41066         m.show(this.el, 'tl-bl?');
41067         ignorehide  = false;
41068         // this will trigger hideMonthPicker..
41069         
41070         
41071         // hidden the day picker
41072         Roo.select('.x-date-picker table', true).first().dom.style.visibility = "hidden";
41073         
41074         
41075         
41076       
41077         
41078         p.showMonthPicker.defer(100, p);
41079     
41080         
41081        
41082     },
41083
41084     beforeBlur : function(){
41085         var v = this.parseDate(this.getRawValue());
41086         if(v){
41087             this.setValue(v);
41088         }
41089     }
41090
41091     /** @cfg {Boolean} grow @hide */
41092     /** @cfg {Number} growMin @hide */
41093     /** @cfg {Number} growMax @hide */
41094     /**
41095      * @hide
41096      * @method autoSize
41097      */
41098 });/*
41099  * Based on:
41100  * Ext JS Library 1.1.1
41101  * Copyright(c) 2006-2007, Ext JS, LLC.
41102  *
41103  * Originally Released Under LGPL - original licence link has changed is not relivant.
41104  *
41105  * Fork - LGPL
41106  * <script type="text/javascript">
41107  */
41108  
41109
41110 /**
41111  * @class Roo.form.ComboBox
41112  * @extends Roo.form.TriggerField
41113  * A combobox control with support for autocomplete, remote-loading, paging and many other features.
41114  * @constructor
41115  * Create a new ComboBox.
41116  * @param {Object} config Configuration options
41117  */
41118 Roo.form.ComboBox = function(config){
41119     Roo.form.ComboBox.superclass.constructor.call(this, config);
41120     this.addEvents({
41121         /**
41122          * @event expand
41123          * Fires when the dropdown list is expanded
41124              * @param {Roo.form.ComboBox} combo This combo box
41125              */
41126         'expand' : true,
41127         /**
41128          * @event collapse
41129          * Fires when the dropdown list is collapsed
41130              * @param {Roo.form.ComboBox} combo This combo box
41131              */
41132         'collapse' : true,
41133         /**
41134          * @event beforeselect
41135          * Fires before a list item is selected. Return false to cancel the selection.
41136              * @param {Roo.form.ComboBox} combo This combo box
41137              * @param {Roo.data.Record} record The data record returned from the underlying store
41138              * @param {Number} index The index of the selected item in the dropdown list
41139              */
41140         'beforeselect' : true,
41141         /**
41142          * @event select
41143          * Fires when a list item is selected
41144              * @param {Roo.form.ComboBox} combo This combo box
41145              * @param {Roo.data.Record} record The data record returned from the underlying store (or false on clear)
41146              * @param {Number} index The index of the selected item in the dropdown list
41147              */
41148         'select' : true,
41149         /**
41150          * @event beforequery
41151          * Fires before all queries are processed. Return false to cancel the query or set cancel to true.
41152          * The event object passed has these properties:
41153              * @param {Roo.form.ComboBox} combo This combo box
41154              * @param {String} query The query
41155              * @param {Boolean} forceAll true to force "all" query
41156              * @param {Boolean} cancel true to cancel the query
41157              * @param {Object} e The query event object
41158              */
41159         'beforequery': true,
41160          /**
41161          * @event add
41162          * Fires when the 'add' icon is pressed (add a listener to enable add button)
41163              * @param {Roo.form.ComboBox} combo This combo box
41164              */
41165         'add' : true,
41166         /**
41167          * @event edit
41168          * Fires when the 'edit' icon is pressed (add a listener to enable add button)
41169              * @param {Roo.form.ComboBox} combo This combo box
41170              * @param {Roo.data.Record|false} record The data record returned from the underlying store (or false on nothing selected)
41171              */
41172         'edit' : true
41173         
41174         
41175     });
41176     if(this.transform){
41177         this.allowDomMove = false;
41178         var s = Roo.getDom(this.transform);
41179         if(!this.hiddenName){
41180             this.hiddenName = s.name;
41181         }
41182         if(!this.store){
41183             this.mode = 'local';
41184             var d = [], opts = s.options;
41185             for(var i = 0, len = opts.length;i < len; i++){
41186                 var o = opts[i];
41187                 var value = (Roo.isIE ? o.getAttributeNode('value').specified : o.hasAttribute('value')) ? o.value : o.text;
41188                 if(o.selected) {
41189                     this.value = value;
41190                 }
41191                 d.push([value, o.text]);
41192             }
41193             this.store = new Roo.data.SimpleStore({
41194                 'id': 0,
41195                 fields: ['value', 'text'],
41196                 data : d
41197             });
41198             this.valueField = 'value';
41199             this.displayField = 'text';
41200         }
41201         s.name = Roo.id(); // wipe out the name in case somewhere else they have a reference
41202         if(!this.lazyRender){
41203             this.target = true;
41204             this.el = Roo.DomHelper.insertBefore(s, this.autoCreate || this.defaultAutoCreate);
41205             s.parentNode.removeChild(s); // remove it
41206             this.render(this.el.parentNode);
41207         }else{
41208             s.parentNode.removeChild(s); // remove it
41209         }
41210
41211     }
41212     if (this.store) {
41213         this.store = Roo.factory(this.store, Roo.data);
41214     }
41215     
41216     this.selectedIndex = -1;
41217     if(this.mode == 'local'){
41218         if(config.queryDelay === undefined){
41219             this.queryDelay = 10;
41220         }
41221         if(config.minChars === undefined){
41222             this.minChars = 0;
41223         }
41224     }
41225 };
41226
41227 Roo.extend(Roo.form.ComboBox, Roo.form.TriggerField, {
41228     /**
41229      * @cfg {String/HTMLElement/Element} transform The id, DOM node or element of an existing select to convert to a ComboBox
41230      */
41231     /**
41232      * @cfg {Boolean} lazyRender True to prevent the ComboBox from rendering until requested (should always be used when
41233      * rendering into an Roo.Editor, defaults to false)
41234      */
41235     /**
41236      * @cfg {Boolean/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to:
41237      * {tag: "input", type: "text", size: "24", autocomplete: "off"})
41238      */
41239     /**
41240      * @cfg {Roo.data.Store} store The data store to which this combo is bound (defaults to undefined)
41241      */
41242     /**
41243      * @cfg {String} title If supplied, a header element is created containing this text and added into the top of
41244      * the dropdown list (defaults to undefined, with no header element)
41245      */
41246
41247      /**
41248      * @cfg {String/Roo.Template} tpl The template to use to render the output
41249      */
41250      
41251     // private
41252     defaultAutoCreate : {tag: "input", type: "text", size: "24", autocomplete: "off"},
41253     /**
41254      * @cfg {Number} listWidth The width in pixels of the dropdown list (defaults to the width of the ComboBox field)
41255      */
41256     listWidth: undefined,
41257     /**
41258      * @cfg {String} displayField The underlying data field name to bind to this CombBox (defaults to undefined if
41259      * mode = 'remote' or 'text' if mode = 'local')
41260      */
41261     displayField: undefined,
41262     /**
41263      * @cfg {String} valueField The underlying data value name to bind to this CombBox (defaults to undefined if
41264      * mode = 'remote' or 'value' if mode = 'local'). 
41265      * Note: use of a valueField requires the user make a selection
41266      * in order for a value to be mapped.
41267      */
41268     valueField: undefined,
41269     
41270     
41271     /**
41272      * @cfg {String} hiddenName If specified, a hidden form field with this name is dynamically generated to store the
41273      * field's data value (defaults to the underlying DOM element's name)
41274      */
41275     hiddenName: undefined,
41276     /**
41277      * @cfg {String} listClass CSS class to apply to the dropdown list element (defaults to '')
41278      */
41279     listClass: '',
41280     /**
41281      * @cfg {String} selectedClass CSS class to apply to the selected item in the dropdown list (defaults to 'x-combo-selected')
41282      */
41283     selectedClass: 'x-combo-selected',
41284     /**
41285      * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
41286      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-arrow-trigger'
41287      * which displays a downward arrow icon).
41288      */
41289     triggerClass : 'x-form-arrow-trigger',
41290     /**
41291      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
41292      */
41293     shadow:'sides',
41294     /**
41295      * @cfg {String} listAlign A valid anchor position value. See {@link Roo.Element#alignTo} for details on supported
41296      * anchor positions (defaults to 'tl-bl')
41297      */
41298     listAlign: 'tl-bl?',
41299     /**
41300      * @cfg {Number} maxHeight The maximum height in pixels of the dropdown list before scrollbars are shown (defaults to 300)
41301      */
41302     maxHeight: 300,
41303     /**
41304      * @cfg {String} triggerAction The action to execute when the trigger field is activated.  Use 'all' to run the
41305      * query specified by the allQuery config option (defaults to 'query')
41306      */
41307     triggerAction: 'query',
41308     /**
41309      * @cfg {Number} minChars The minimum number of characters the user must type before autocomplete and typeahead activate
41310      * (defaults to 4, does not apply if editable = false)
41311      */
41312     minChars : 4,
41313     /**
41314      * @cfg {Boolean} typeAhead True to populate and autoselect the remainder of the text being typed after a configurable
41315      * delay (typeAheadDelay) if it matches a known value (defaults to false)
41316      */
41317     typeAhead: false,
41318     /**
41319      * @cfg {Number} queryDelay The length of time in milliseconds to delay between the start of typing and sending the
41320      * query to filter the dropdown list (defaults to 500 if mode = 'remote' or 10 if mode = 'local')
41321      */
41322     queryDelay: 500,
41323     /**
41324      * @cfg {Number} pageSize If greater than 0, a paging toolbar is displayed in the footer of the dropdown list and the
41325      * filter queries will execute with page start and limit parameters.  Only applies when mode = 'remote' (defaults to 0)
41326      */
41327     pageSize: 0,
41328     /**
41329      * @cfg {Boolean} selectOnFocus True to select any existing text in the field immediately on focus.  Only applies
41330      * when editable = true (defaults to false)
41331      */
41332     selectOnFocus:false,
41333     /**
41334      * @cfg {String} queryParam Name of the query as it will be passed on the querystring (defaults to 'query')
41335      */
41336     queryParam: 'query',
41337     /**
41338      * @cfg {String} loadingText The text to display in the dropdown list while data is loading.  Only applies
41339      * when mode = 'remote' (defaults to 'Loading...')
41340      */
41341     loadingText: 'Loading...',
41342     /**
41343      * @cfg {Boolean} resizable True to add a resize handle to the bottom of the dropdown list (defaults to false)
41344      */
41345     resizable: false,
41346     /**
41347      * @cfg {Number} handleHeight The height in pixels of the dropdown list resize handle if resizable = true (defaults to 8)
41348      */
41349     handleHeight : 8,
41350     /**
41351      * @cfg {Boolean} editable False to prevent the user from typing text directly into the field, just like a
41352      * traditional select (defaults to true)
41353      */
41354     editable: true,
41355     /**
41356      * @cfg {String} allQuery The text query to send to the server to return all records for the list with no filtering (defaults to '')
41357      */
41358     allQuery: '',
41359     /**
41360      * @cfg {String} mode Set to 'local' if the ComboBox loads local data (defaults to 'remote' which loads from the server)
41361      */
41362     mode: 'remote',
41363     /**
41364      * @cfg {Number} minListWidth The minimum width of the dropdown list in pixels (defaults to 70, will be ignored if
41365      * listWidth has a higher value)
41366      */
41367     minListWidth : 70,
41368     /**
41369      * @cfg {Boolean} forceSelection True to restrict the selected value to one of the values in the list, false to
41370      * allow the user to set arbitrary text into the field (defaults to false)
41371      */
41372     forceSelection:false,
41373     /**
41374      * @cfg {Number} typeAheadDelay The length of time in milliseconds to wait until the typeahead text is displayed
41375      * if typeAhead = true (defaults to 250)
41376      */
41377     typeAheadDelay : 250,
41378     /**
41379      * @cfg {String} valueNotFoundText When using a name/value combo, if the value passed to setValue is not found in
41380      * the store, valueNotFoundText will be displayed as the field text if defined (defaults to undefined)
41381      */
41382     valueNotFoundText : undefined,
41383     /**
41384      * @cfg {Boolean} blockFocus Prevents all focus calls, so it can work with things like HTML edtor bar
41385      */
41386     blockFocus : false,
41387     
41388     /**
41389      * @cfg {Boolean} disableClear Disable showing of clear button.
41390      */
41391     disableClear : false,
41392     /**
41393      * @cfg {Boolean} alwaysQuery  Disable caching of results, and always send query
41394      */
41395     alwaysQuery : false,
41396     
41397     //private
41398     addicon : false,
41399     editicon: false,
41400     
41401     // element that contains real text value.. (when hidden is used..)
41402      
41403     // private
41404     onRender : function(ct, position)
41405     {
41406         Roo.form.ComboBox.superclass.onRender.call(this, ct, position);
41407         
41408         if(this.hiddenName){
41409             this.hiddenField = this.el.insertSibling({tag:'input', type:'hidden', name: this.hiddenName, id:  (this.hiddenId||this.hiddenName)},
41410                     'before', true);
41411             this.hiddenField.value =
41412                 this.hiddenValue !== undefined ? this.hiddenValue :
41413                 this.value !== undefined ? this.value : '';
41414
41415             // prevent input submission
41416             this.el.dom.removeAttribute('name');
41417              
41418              
41419         }
41420         
41421         if(Roo.isGecko){
41422             this.el.dom.setAttribute('autocomplete', 'off');
41423         }
41424
41425         var cls = 'x-combo-list';
41426
41427         this.list = new Roo.Layer({
41428             shadow: this.shadow, cls: [cls, this.listClass].join(' '), constrain:false
41429         });
41430
41431         var lw = this.listWidth || Math.max(this.wrap.getWidth(), this.minListWidth);
41432         this.list.setWidth(lw);
41433         this.list.swallowEvent('mousewheel');
41434         this.assetHeight = 0;
41435
41436         if(this.title){
41437             this.header = this.list.createChild({cls:cls+'-hd', html: this.title});
41438             this.assetHeight += this.header.getHeight();
41439         }
41440
41441         this.innerList = this.list.createChild({cls:cls+'-inner'});
41442         this.innerList.on('mouseover', this.onViewOver, this);
41443         this.innerList.on('mousemove', this.onViewMove, this);
41444         this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
41445         
41446         if(this.allowBlank && !this.pageSize && !this.disableClear){
41447             this.footer = this.list.createChild({cls:cls+'-ft'});
41448             this.pageTb = new Roo.Toolbar(this.footer);
41449            
41450         }
41451         if(this.pageSize){
41452             this.footer = this.list.createChild({cls:cls+'-ft'});
41453             this.pageTb = new Roo.PagingToolbar(this.footer, this.store,
41454                     {pageSize: this.pageSize});
41455             
41456         }
41457         
41458         if (this.pageTb && this.allowBlank && !this.disableClear) {
41459             var _this = this;
41460             this.pageTb.add(new Roo.Toolbar.Fill(), {
41461                 cls: 'x-btn-icon x-btn-clear',
41462                 text: '&#160;',
41463                 handler: function()
41464                 {
41465                     _this.collapse();
41466                     _this.clearValue();
41467                     _this.onSelect(false, -1);
41468                 }
41469             });
41470         }
41471         if (this.footer) {
41472             this.assetHeight += this.footer.getHeight();
41473         }
41474         
41475
41476         if(!this.tpl){
41477             this.tpl = '<div class="'+cls+'-item">{' + this.displayField + '}</div>';
41478         }
41479
41480         this.view = new Roo.View(this.innerList, this.tpl, {
41481             singleSelect:true,
41482             store: this.store,
41483             selectedClass: this.selectedClass
41484         });
41485
41486         this.view.on('click', this.onViewClick, this);
41487
41488         this.store.on('beforeload', this.onBeforeLoad, this);
41489         this.store.on('load', this.onLoad, this);
41490         this.store.on('loadexception', this.onLoadException, this);
41491
41492         if(this.resizable){
41493             this.resizer = new Roo.Resizable(this.list,  {
41494                pinned:true, handles:'se'
41495             });
41496             this.resizer.on('resize', function(r, w, h){
41497                 this.maxHeight = h-this.handleHeight-this.list.getFrameWidth('tb')-this.assetHeight;
41498                 this.listWidth = w;
41499                 this.innerList.setWidth(w - this.list.getFrameWidth('lr'));
41500                 this.restrictHeight();
41501             }, this);
41502             this[this.pageSize?'footer':'innerList'].setStyle('margin-bottom', this.handleHeight+'px');
41503         }
41504         if(!this.editable){
41505             this.editable = true;
41506             this.setEditable(false);
41507         }  
41508         
41509         
41510         if (typeof(this.events.add.listeners) != 'undefined') {
41511             
41512             this.addicon = this.wrap.createChild(
41513                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-combo-add' });  
41514        
41515             this.addicon.on('click', function(e) {
41516                 this.fireEvent('add', this);
41517             }, this);
41518         }
41519         if (typeof(this.events.edit.listeners) != 'undefined') {
41520             
41521             this.editicon = this.wrap.createChild(
41522                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-combo-edit' });  
41523             if (this.addicon) {
41524                 this.editicon.setStyle('margin-left', '40px');
41525             }
41526             this.editicon.on('click', function(e) {
41527                 
41528                 // we fire even  if inothing is selected..
41529                 this.fireEvent('edit', this, this.lastData );
41530                 
41531             }, this);
41532         }
41533         
41534         
41535         
41536     },
41537
41538     // private
41539     initEvents : function(){
41540         Roo.form.ComboBox.superclass.initEvents.call(this);
41541
41542         this.keyNav = new Roo.KeyNav(this.el, {
41543             "up" : function(e){
41544                 this.inKeyMode = true;
41545                 this.selectPrev();
41546             },
41547
41548             "down" : function(e){
41549                 if(!this.isExpanded()){
41550                     this.onTriggerClick();
41551                 }else{
41552                     this.inKeyMode = true;
41553                     this.selectNext();
41554                 }
41555             },
41556
41557             "enter" : function(e){
41558                 this.onViewClick();
41559                 //return true;
41560             },
41561
41562             "esc" : function(e){
41563                 this.collapse();
41564             },
41565
41566             "tab" : function(e){
41567                 this.onViewClick(false);
41568                 this.fireEvent("specialkey", this, e);
41569                 return true;
41570             },
41571
41572             scope : this,
41573
41574             doRelay : function(foo, bar, hname){
41575                 if(hname == 'down' || this.scope.isExpanded()){
41576                    return Roo.KeyNav.prototype.doRelay.apply(this, arguments);
41577                 }
41578                 return true;
41579             },
41580
41581             forceKeyDown: true
41582         });
41583         this.queryDelay = Math.max(this.queryDelay || 10,
41584                 this.mode == 'local' ? 10 : 250);
41585         this.dqTask = new Roo.util.DelayedTask(this.initQuery, this);
41586         if(this.typeAhead){
41587             this.taTask = new Roo.util.DelayedTask(this.onTypeAhead, this);
41588         }
41589         if(this.editable !== false){
41590             this.el.on("keyup", this.onKeyUp, this);
41591         }
41592         if(this.forceSelection){
41593             this.on('blur', this.doForce, this);
41594         }
41595     },
41596
41597     onDestroy : function(){
41598         if(this.view){
41599             this.view.setStore(null);
41600             this.view.el.removeAllListeners();
41601             this.view.el.remove();
41602             this.view.purgeListeners();
41603         }
41604         if(this.list){
41605             this.list.destroy();
41606         }
41607         if(this.store){
41608             this.store.un('beforeload', this.onBeforeLoad, this);
41609             this.store.un('load', this.onLoad, this);
41610             this.store.un('loadexception', this.onLoadException, this);
41611         }
41612         Roo.form.ComboBox.superclass.onDestroy.call(this);
41613     },
41614
41615     // private
41616     fireKey : function(e){
41617         if(e.isNavKeyPress() && !this.list.isVisible()){
41618             this.fireEvent("specialkey", this, e);
41619         }
41620     },
41621
41622     // private
41623     onResize: function(w, h){
41624         Roo.form.ComboBox.superclass.onResize.apply(this, arguments);
41625         
41626         if(typeof w != 'number'){
41627             // we do not handle it!?!?
41628             return;
41629         }
41630         var tw = this.trigger.getWidth();
41631         tw += this.addicon ? this.addicon.getWidth() : 0;
41632         tw += this.editicon ? this.editicon.getWidth() : 0;
41633         var x = w - tw;
41634         this.el.setWidth( this.adjustWidth('input', x));
41635             
41636         this.trigger.setStyle('left', x+'px');
41637         
41638         if(this.list && this.listWidth === undefined){
41639             var lw = Math.max(x + this.trigger.getWidth(), this.minListWidth);
41640             this.list.setWidth(lw);
41641             this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
41642         }
41643         
41644     
41645         
41646     },
41647
41648     /**
41649      * Allow or prevent the user from directly editing the field text.  If false is passed,
41650      * the user will only be able to select from the items defined in the dropdown list.  This method
41651      * is the runtime equivalent of setting the 'editable' config option at config time.
41652      * @param {Boolean} value True to allow the user to directly edit the field text
41653      */
41654     setEditable : function(value){
41655         if(value == this.editable){
41656             return;
41657         }
41658         this.editable = value;
41659         if(!value){
41660             this.el.dom.setAttribute('readOnly', true);
41661             this.el.on('mousedown', this.onTriggerClick,  this);
41662             this.el.addClass('x-combo-noedit');
41663         }else{
41664             this.el.dom.setAttribute('readOnly', false);
41665             this.el.un('mousedown', this.onTriggerClick,  this);
41666             this.el.removeClass('x-combo-noedit');
41667         }
41668     },
41669
41670     // private
41671     onBeforeLoad : function(){
41672         if(!this.hasFocus){
41673             return;
41674         }
41675         this.innerList.update(this.loadingText ?
41676                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
41677         this.restrictHeight();
41678         this.selectedIndex = -1;
41679     },
41680
41681     // private
41682     onLoad : function(){
41683         if(!this.hasFocus){
41684             return;
41685         }
41686         if(this.store.getCount() > 0){
41687             this.expand();
41688             this.restrictHeight();
41689             if(this.lastQuery == this.allQuery){
41690                 if(this.editable){
41691                     this.el.dom.select();
41692                 }
41693                 if(!this.selectByValue(this.value, true)){
41694                     this.select(0, true);
41695                 }
41696             }else{
41697                 this.selectNext();
41698                 if(this.typeAhead && this.lastKey != Roo.EventObject.BACKSPACE && this.lastKey != Roo.EventObject.DELETE){
41699                     this.taTask.delay(this.typeAheadDelay);
41700                 }
41701             }
41702         }else{
41703             this.onEmptyResults();
41704         }
41705         //this.el.focus();
41706     },
41707     // private
41708     onLoadException : function()
41709     {
41710         this.collapse();
41711         Roo.log(this.store.reader.jsonData);
41712         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
41713             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
41714         }
41715         
41716         
41717     },
41718     // private
41719     onTypeAhead : function(){
41720         if(this.store.getCount() > 0){
41721             var r = this.store.getAt(0);
41722             var newValue = r.data[this.displayField];
41723             var len = newValue.length;
41724             var selStart = this.getRawValue().length;
41725             if(selStart != len){
41726                 this.setRawValue(newValue);
41727                 this.selectText(selStart, newValue.length);
41728             }
41729         }
41730     },
41731
41732     // private
41733     onSelect : function(record, index){
41734         if(this.fireEvent('beforeselect', this, record, index) !== false){
41735             this.setFromData(index > -1 ? record.data : false);
41736             this.collapse();
41737             this.fireEvent('select', this, record, index);
41738         }
41739     },
41740
41741     /**
41742      * Returns the currently selected field value or empty string if no value is set.
41743      * @return {String} value The selected value
41744      */
41745     getValue : function(){
41746         if(this.valueField){
41747             return typeof this.value != 'undefined' ? this.value : '';
41748         }
41749         return Roo.form.ComboBox.superclass.getValue.call(this);
41750     },
41751
41752     /**
41753      * Clears any text/value currently set in the field
41754      */
41755     clearValue : function(){
41756         if(this.hiddenField){
41757             this.hiddenField.value = '';
41758         }
41759         this.value = '';
41760         this.setRawValue('');
41761         this.lastSelectionText = '';
41762         
41763     },
41764
41765     /**
41766      * Sets the specified value into the field.  If the value finds a match, the corresponding record text
41767      * will be displayed in the field.  If the value does not match the data value of an existing item,
41768      * and the valueNotFoundText config option is defined, it will be displayed as the default field text.
41769      * Otherwise the field will be blank (although the value will still be set).
41770      * @param {String} value The value to match
41771      */
41772     setValue : function(v){
41773         var text = v;
41774         if(this.valueField){
41775             var r = this.findRecord(this.valueField, v);
41776             if(r){
41777                 text = r.data[this.displayField];
41778             }else if(this.valueNotFoundText !== undefined){
41779                 text = this.valueNotFoundText;
41780             }
41781         }
41782         this.lastSelectionText = text;
41783         if(this.hiddenField){
41784             this.hiddenField.value = v;
41785         }
41786         Roo.form.ComboBox.superclass.setValue.call(this, text);
41787         this.value = v;
41788     },
41789     /**
41790      * @property {Object} the last set data for the element
41791      */
41792     
41793     lastData : false,
41794     /**
41795      * Sets the value of the field based on a object which is related to the record format for the store.
41796      * @param {Object} value the value to set as. or false on reset?
41797      */
41798     setFromData : function(o){
41799         var dv = ''; // display value
41800         var vv = ''; // value value..
41801         this.lastData = o;
41802         if (this.displayField) {
41803             dv = !o || typeof(o[this.displayField]) == 'undefined' ? '' : o[this.displayField];
41804         } else {
41805             // this is an error condition!!!
41806             Roo.log('no  displayField value set for '+ (this.name ? this.name : this.id));
41807         }
41808         
41809         if(this.valueField){
41810             vv = !o || typeof(o[this.valueField]) == 'undefined' ? dv : o[this.valueField];
41811         }
41812         if(this.hiddenField){
41813             this.hiddenField.value = vv;
41814             
41815             this.lastSelectionText = dv;
41816             Roo.form.ComboBox.superclass.setValue.call(this, dv);
41817             this.value = vv;
41818             return;
41819         }
41820         // no hidden field.. - we store the value in 'value', but still display
41821         // display field!!!!
41822         this.lastSelectionText = dv;
41823         Roo.form.ComboBox.superclass.setValue.call(this, dv);
41824         this.value = vv;
41825         
41826         
41827     },
41828     // private
41829     reset : function(){
41830         // overridden so that last data is reset..
41831         this.setValue(this.resetValue);
41832         this.originalValue = this.getValue();
41833         this.clearInvalid();
41834         this.lastData = false;
41835         if (this.view) {
41836             this.view.clearSelections();
41837         }
41838     },
41839     // private
41840     findRecord : function(prop, value){
41841         var record;
41842         if(this.store.getCount() > 0){
41843             this.store.each(function(r){
41844                 if(r.data[prop] == value){
41845                     record = r;
41846                     return false;
41847                 }
41848                 return true;
41849             });
41850         }
41851         return record;
41852     },
41853     
41854     getName: function()
41855     {
41856         // returns hidden if it's set..
41857         if (!this.rendered) {return ''};
41858         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
41859         
41860     },
41861     // private
41862     onViewMove : function(e, t){
41863         this.inKeyMode = false;
41864     },
41865
41866     // private
41867     onViewOver : function(e, t){
41868         if(this.inKeyMode){ // prevent key nav and mouse over conflicts
41869             return;
41870         }
41871         var item = this.view.findItemFromChild(t);
41872         if(item){
41873             var index = this.view.indexOf(item);
41874             this.select(index, false);
41875         }
41876     },
41877
41878     // private
41879     onViewClick : function(doFocus)
41880     {
41881         var index = this.view.getSelectedIndexes()[0];
41882         var r = this.store.getAt(index);
41883         if(r){
41884             this.onSelect(r, index);
41885         }
41886         if(doFocus !== false && !this.blockFocus){
41887             this.el.focus();
41888         }
41889     },
41890
41891     // private
41892     restrictHeight : function(){
41893         this.innerList.dom.style.height = '';
41894         var inner = this.innerList.dom;
41895         var h = Math.max(inner.clientHeight, inner.offsetHeight, inner.scrollHeight);
41896         this.innerList.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
41897         this.list.beginUpdate();
41898         this.list.setHeight(this.innerList.getHeight()+this.list.getFrameWidth('tb')+(this.resizable?this.handleHeight:0)+this.assetHeight);
41899         this.list.alignTo(this.el, this.listAlign);
41900         this.list.endUpdate();
41901     },
41902
41903     // private
41904     onEmptyResults : function(){
41905         this.collapse();
41906     },
41907
41908     /**
41909      * Returns true if the dropdown list is expanded, else false.
41910      */
41911     isExpanded : function(){
41912         return this.list.isVisible();
41913     },
41914
41915     /**
41916      * Select an item in the dropdown list by its data value. This function does NOT cause the select event to fire.
41917      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
41918      * @param {String} value The data value of the item to select
41919      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
41920      * selected item if it is not currently in view (defaults to true)
41921      * @return {Boolean} True if the value matched an item in the list, else false
41922      */
41923     selectByValue : function(v, scrollIntoView){
41924         if(v !== undefined && v !== null){
41925             var r = this.findRecord(this.valueField || this.displayField, v);
41926             if(r){
41927                 this.select(this.store.indexOf(r), scrollIntoView);
41928                 return true;
41929             }
41930         }
41931         return false;
41932     },
41933
41934     /**
41935      * Select an item in the dropdown list by its numeric index in the list. This function does NOT cause the select event to fire.
41936      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
41937      * @param {Number} index The zero-based index of the list item to select
41938      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
41939      * selected item if it is not currently in view (defaults to true)
41940      */
41941     select : function(index, scrollIntoView){
41942         this.selectedIndex = index;
41943         this.view.select(index);
41944         if(scrollIntoView !== false){
41945             var el = this.view.getNode(index);
41946             if(el){
41947                 this.innerList.scrollChildIntoView(el, false);
41948             }
41949         }
41950     },
41951
41952     // private
41953     selectNext : function(){
41954         var ct = this.store.getCount();
41955         if(ct > 0){
41956             if(this.selectedIndex == -1){
41957                 this.select(0);
41958             }else if(this.selectedIndex < ct-1){
41959                 this.select(this.selectedIndex+1);
41960             }
41961         }
41962     },
41963
41964     // private
41965     selectPrev : function(){
41966         var ct = this.store.getCount();
41967         if(ct > 0){
41968             if(this.selectedIndex == -1){
41969                 this.select(0);
41970             }else if(this.selectedIndex != 0){
41971                 this.select(this.selectedIndex-1);
41972             }
41973         }
41974     },
41975
41976     // private
41977     onKeyUp : function(e){
41978         if(this.editable !== false && !e.isSpecialKey()){
41979             this.lastKey = e.getKey();
41980             this.dqTask.delay(this.queryDelay);
41981         }
41982     },
41983
41984     // private
41985     validateBlur : function(){
41986         return !this.list || !this.list.isVisible();   
41987     },
41988
41989     // private
41990     initQuery : function(){
41991         this.doQuery(this.getRawValue());
41992     },
41993
41994     // private
41995     doForce : function(){
41996         if(this.el.dom.value.length > 0){
41997             this.el.dom.value =
41998                 this.lastSelectionText === undefined ? '' : this.lastSelectionText;
41999              
42000         }
42001     },
42002
42003     /**
42004      * Execute a query to filter the dropdown list.  Fires the beforequery event prior to performing the
42005      * query allowing the query action to be canceled if needed.
42006      * @param {String} query The SQL query to execute
42007      * @param {Boolean} forceAll True to force the query to execute even if there are currently fewer characters
42008      * in the field than the minimum specified by the minChars config option.  It also clears any filter previously
42009      * saved in the current store (defaults to false)
42010      */
42011     doQuery : function(q, forceAll){
42012         if(q === undefined || q === null){
42013             q = '';
42014         }
42015         var qe = {
42016             query: q,
42017             forceAll: forceAll,
42018             combo: this,
42019             cancel:false
42020         };
42021         if(this.fireEvent('beforequery', qe)===false || qe.cancel){
42022             return false;
42023         }
42024         q = qe.query;
42025         forceAll = qe.forceAll;
42026         if(forceAll === true || (q.length >= this.minChars)){
42027             if(this.lastQuery != q || this.alwaysQuery){
42028                 this.lastQuery = q;
42029                 if(this.mode == 'local'){
42030                     this.selectedIndex = -1;
42031                     if(forceAll){
42032                         this.store.clearFilter();
42033                     }else{
42034                         this.store.filter(this.displayField, q);
42035                     }
42036                     this.onLoad();
42037                 }else{
42038                     this.store.baseParams[this.queryParam] = q;
42039                     this.store.load({
42040                         params: this.getParams(q)
42041                     });
42042                     this.expand();
42043                 }
42044             }else{
42045                 this.selectedIndex = -1;
42046                 this.onLoad();   
42047             }
42048         }
42049     },
42050
42051     // private
42052     getParams : function(q){
42053         var p = {};
42054         //p[this.queryParam] = q;
42055         if(this.pageSize){
42056             p.start = 0;
42057             p.limit = this.pageSize;
42058         }
42059         return p;
42060     },
42061
42062     /**
42063      * Hides the dropdown list if it is currently expanded. Fires the 'collapse' event on completion.
42064      */
42065     collapse : function(){
42066         if(!this.isExpanded()){
42067             return;
42068         }
42069         this.list.hide();
42070         Roo.get(document).un('mousedown', this.collapseIf, this);
42071         Roo.get(document).un('mousewheel', this.collapseIf, this);
42072         if (!this.editable) {
42073             Roo.get(document).un('keydown', this.listKeyPress, this);
42074         }
42075         this.fireEvent('collapse', this);
42076     },
42077
42078     // private
42079     collapseIf : function(e){
42080         if(!e.within(this.wrap) && !e.within(this.list)){
42081             this.collapse();
42082         }
42083     },
42084
42085     /**
42086      * Expands the dropdown list if it is currently hidden. Fires the 'expand' event on completion.
42087      */
42088     expand : function(){
42089         if(this.isExpanded() || !this.hasFocus){
42090             return;
42091         }
42092         this.list.alignTo(this.el, this.listAlign);
42093         this.list.show();
42094         Roo.get(document).on('mousedown', this.collapseIf, this);
42095         Roo.get(document).on('mousewheel', this.collapseIf, this);
42096         if (!this.editable) {
42097             Roo.get(document).on('keydown', this.listKeyPress, this);
42098         }
42099         
42100         this.fireEvent('expand', this);
42101     },
42102
42103     // private
42104     // Implements the default empty TriggerField.onTriggerClick function
42105     onTriggerClick : function(){
42106         if(this.disabled){
42107             return;
42108         }
42109         if(this.isExpanded()){
42110             this.collapse();
42111             if (!this.blockFocus) {
42112                 this.el.focus();
42113             }
42114             
42115         }else {
42116             this.hasFocus = true;
42117             if(this.triggerAction == 'all') {
42118                 this.doQuery(this.allQuery, true);
42119             } else {
42120                 this.doQuery(this.getRawValue());
42121             }
42122             if (!this.blockFocus) {
42123                 this.el.focus();
42124             }
42125         }
42126     },
42127     listKeyPress : function(e)
42128     {
42129         //Roo.log('listkeypress');
42130         // scroll to first matching element based on key pres..
42131         if (e.isSpecialKey()) {
42132             return false;
42133         }
42134         var k = String.fromCharCode(e.getKey()).toUpperCase();
42135         //Roo.log(k);
42136         var match  = false;
42137         var csel = this.view.getSelectedNodes();
42138         var cselitem = false;
42139         if (csel.length) {
42140             var ix = this.view.indexOf(csel[0]);
42141             cselitem  = this.store.getAt(ix);
42142             if (!cselitem.get(this.displayField) || cselitem.get(this.displayField).substring(0,1).toUpperCase() != k) {
42143                 cselitem = false;
42144             }
42145             
42146         }
42147         
42148         this.store.each(function(v) { 
42149             if (cselitem) {
42150                 // start at existing selection.
42151                 if (cselitem.id == v.id) {
42152                     cselitem = false;
42153                 }
42154                 return;
42155             }
42156                 
42157             if (v.get(this.displayField) && v.get(this.displayField).substring(0,1).toUpperCase() == k) {
42158                 match = this.store.indexOf(v);
42159                 return false;
42160             }
42161         }, this);
42162         
42163         if (match === false) {
42164             return true; // no more action?
42165         }
42166         // scroll to?
42167         this.view.select(match);
42168         var sn = Roo.get(this.view.getSelectedNodes()[0]);
42169         sn.scrollIntoView(sn.dom.parentNode, false);
42170     } 
42171
42172     /** 
42173     * @cfg {Boolean} grow 
42174     * @hide 
42175     */
42176     /** 
42177     * @cfg {Number} growMin 
42178     * @hide 
42179     */
42180     /** 
42181     * @cfg {Number} growMax 
42182     * @hide 
42183     */
42184     /**
42185      * @hide
42186      * @method autoSize
42187      */
42188 });/*
42189  * Copyright(c) 2010-2012, Roo J Solutions Limited
42190  *
42191  * Licence LGPL
42192  *
42193  */
42194
42195 /**
42196  * @class Roo.form.ComboBoxArray
42197  * @extends Roo.form.TextField
42198  * A facebook style adder... for lists of email / people / countries  etc...
42199  * pick multiple items from a combo box, and shows each one.
42200  *
42201  *  Fred [x]  Brian [x]  [Pick another |v]
42202  *
42203  *
42204  *  For this to work: it needs various extra information
42205  *    - normal combo problay has
42206  *      name, hiddenName
42207  *    + displayField, valueField
42208  *
42209  *    For our purpose...
42210  *
42211  *
42212  *   If we change from 'extends' to wrapping...
42213  *   
42214  *  
42215  *
42216  
42217  
42218  * @constructor
42219  * Create a new ComboBoxArray.
42220  * @param {Object} config Configuration options
42221  */
42222  
42223
42224 Roo.form.ComboBoxArray = function(config)
42225 {
42226     this.addEvents({
42227         /**
42228          * @event beforeremove
42229          * Fires before remove the value from the list
42230              * @param {Roo.form.ComboBoxArray} _self This combo box array
42231              * @param {Roo.form.ComboBoxArray.Item} item removed item
42232              */
42233         'beforeremove' : true,
42234         /**
42235          * @event remove
42236          * Fires when remove the value from the list
42237              * @param {Roo.form.ComboBoxArray} _self This combo box array
42238              * @param {Roo.form.ComboBoxArray.Item} item removed item
42239              */
42240         'remove' : true
42241         
42242         
42243     });
42244     
42245     Roo.form.ComboBoxArray.superclass.constructor.call(this, config);
42246     
42247     this.items = new Roo.util.MixedCollection(false);
42248     
42249     // construct the child combo...
42250     
42251     
42252     
42253     
42254    
42255     
42256 }
42257
42258  
42259 Roo.extend(Roo.form.ComboBoxArray, Roo.form.TextField,
42260
42261     /**
42262      * @cfg {Roo.form.Combo} combo The combo box that is wrapped
42263      */
42264     
42265     lastData : false,
42266     
42267     // behavies liek a hiddne field
42268     inputType:      'hidden',
42269     /**
42270      * @cfg {Number} width The width of the box that displays the selected element
42271      */ 
42272     width:          300,
42273
42274     
42275     
42276     /**
42277      * @cfg {String} name    The name of the visable items on this form (eg. titles not ids)
42278      */
42279     name : false,
42280     /**
42281      * @cfg {String} hiddenName    The hidden name of the field, often contains an comma seperated list of names
42282      */
42283     hiddenName : false,
42284     
42285     
42286     // private the array of items that are displayed..
42287     items  : false,
42288     // private - the hidden field el.
42289     hiddenEl : false,
42290     // private - the filed el..
42291     el : false,
42292     
42293     //validateValue : function() { return true; }, // all values are ok!
42294     //onAddClick: function() { },
42295     
42296     onRender : function(ct, position) 
42297     {
42298         
42299         // create the standard hidden element
42300         //Roo.form.ComboBoxArray.superclass.onRender.call(this, ct, position);
42301         
42302         
42303         // give fake names to child combo;
42304         this.combo.hiddenName = this.hiddenName ? (this.hiddenName+'-subcombo') : this.hiddenName;
42305         this.combo.name = this.name ? (this.name+'-subcombo') : this.name;
42306         
42307         this.combo = Roo.factory(this.combo, Roo.form);
42308         this.combo.onRender(ct, position);
42309         if (typeof(this.combo.width) != 'undefined') {
42310             this.combo.onResize(this.combo.width,0);
42311         }
42312         
42313         this.combo.initEvents();
42314         
42315         // assigned so form know we need to do this..
42316         this.store          = this.combo.store;
42317         this.valueField     = this.combo.valueField;
42318         this.displayField   = this.combo.displayField ;
42319         
42320         
42321         this.combo.wrap.addClass('x-cbarray-grp');
42322         
42323         var cbwrap = this.combo.wrap.createChild(
42324             {tag: 'div', cls: 'x-cbarray-cb'},
42325             this.combo.el.dom
42326         );
42327         
42328              
42329         this.hiddenEl = this.combo.wrap.createChild({
42330             tag: 'input',  type:'hidden' , name: this.hiddenName, value : ''
42331         });
42332         this.el = this.combo.wrap.createChild({
42333             tag: 'input',  type:'hidden' , name: this.name, value : ''
42334         });
42335          //   this.el.dom.removeAttribute("name");
42336         
42337         
42338         this.outerWrap = this.combo.wrap;
42339         this.wrap = cbwrap;
42340         
42341         this.outerWrap.setWidth(this.width);
42342         this.outerWrap.dom.removeChild(this.el.dom);
42343         
42344         this.wrap.dom.appendChild(this.el.dom);
42345         this.outerWrap.dom.removeChild(this.combo.trigger.dom);
42346         this.combo.wrap.dom.appendChild(this.combo.trigger.dom);
42347         
42348         this.combo.trigger.setStyle('position','relative');
42349         this.combo.trigger.setStyle('left', '0px');
42350         this.combo.trigger.setStyle('top', '2px');
42351         
42352         this.combo.el.setStyle('vertical-align', 'text-bottom');
42353         
42354         //this.trigger.setStyle('vertical-align', 'top');
42355         
42356         // this should use the code from combo really... on('add' ....)
42357         if (this.adder) {
42358             
42359         
42360             this.adder = this.outerWrap.createChild(
42361                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-adder', style: 'margin-left:2px'});  
42362             var _t = this;
42363             this.adder.on('click', function(e) {
42364                 _t.fireEvent('adderclick', this, e);
42365             }, _t);
42366         }
42367         //var _t = this;
42368         //this.adder.on('click', this.onAddClick, _t);
42369         
42370         
42371         this.combo.on('select', function(cb, rec, ix) {
42372             this.addItem(rec.data);
42373             
42374             cb.setValue('');
42375             cb.el.dom.value = '';
42376             //cb.lastData = rec.data;
42377             // add to list
42378             
42379         }, this);
42380         
42381         
42382     },
42383     
42384     
42385     getName: function()
42386     {
42387         // returns hidden if it's set..
42388         if (!this.rendered) {return ''};
42389         return  this.hiddenName ? this.hiddenName : this.name;
42390         
42391     },
42392     
42393     
42394     onResize: function(w, h){
42395         
42396         return;
42397         // not sure if this is needed..
42398         //this.combo.onResize(w,h);
42399         
42400         if(typeof w != 'number'){
42401             // we do not handle it!?!?
42402             return;
42403         }
42404         var tw = this.combo.trigger.getWidth();
42405         tw += this.addicon ? this.addicon.getWidth() : 0;
42406         tw += this.editicon ? this.editicon.getWidth() : 0;
42407         var x = w - tw;
42408         this.combo.el.setWidth( this.combo.adjustWidth('input', x));
42409             
42410         this.combo.trigger.setStyle('left', '0px');
42411         
42412         if(this.list && this.listWidth === undefined){
42413             var lw = Math.max(x + this.combo.trigger.getWidth(), this.combo.minListWidth);
42414             this.list.setWidth(lw);
42415             this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
42416         }
42417         
42418     
42419         
42420     },
42421     
42422     addItem: function(rec)
42423     {
42424         var valueField = this.combo.valueField;
42425         var displayField = this.combo.displayField;
42426         
42427         if (this.items.indexOfKey(rec[valueField]) > -1) {
42428             //console.log("GOT " + rec.data.id);
42429             return;
42430         }
42431         
42432         var x = new Roo.form.ComboBoxArray.Item({
42433             //id : rec[this.idField],
42434             data : rec,
42435             displayField : displayField ,
42436             tipField : displayField ,
42437             cb : this
42438         });
42439         // use the 
42440         this.items.add(rec[valueField],x);
42441         // add it before the element..
42442         this.updateHiddenEl();
42443         x.render(this.outerWrap, this.wrap.dom);
42444         // add the image handler..
42445     },
42446     
42447     updateHiddenEl : function()
42448     {
42449         this.validate();
42450         if (!this.hiddenEl) {
42451             return;
42452         }
42453         var ar = [];
42454         var idField = this.combo.valueField;
42455         
42456         this.items.each(function(f) {
42457             ar.push(f.data[idField]);
42458         });
42459         this.hiddenEl.dom.value = ar.join(',');
42460         this.validate();
42461     },
42462     
42463     reset : function()
42464     {
42465         this.items.clear();
42466         
42467         Roo.each(this.outerWrap.select('.x-cbarray-item', true).elements, function(el){
42468            el.remove();
42469         });
42470         
42471         this.el.dom.value = '';
42472         if (this.hiddenEl) {
42473             this.hiddenEl.dom.value = '';
42474         }
42475         
42476     },
42477     getValue: function()
42478     {
42479         return this.hiddenEl ? this.hiddenEl.dom.value : '';
42480     },
42481     setValue: function(v) // not a valid action - must use addItems..
42482     {
42483         
42484         this.reset();
42485          
42486         if (this.store.isLocal && (typeof(v) == 'string')) {
42487             // then we can use the store to find the values..
42488             // comma seperated at present.. this needs to allow JSON based encoding..
42489             this.hiddenEl.value  = v;
42490             var v_ar = [];
42491             Roo.each(v.split(','), function(k) {
42492                 Roo.log("CHECK " + this.valueField + ',' + k);
42493                 var li = this.store.query(this.valueField, k);
42494                 if (!li.length) {
42495                     return;
42496                 }
42497                 var add = {};
42498                 add[this.valueField] = k;
42499                 add[this.displayField] = li.item(0).data[this.displayField];
42500                 
42501                 this.addItem(add);
42502             }, this) 
42503              
42504         }
42505         if (typeof(v) == 'object' ) {
42506             // then let's assume it's an array of objects..
42507             Roo.each(v, function(l) {
42508                 this.addItem(l);
42509             }, this);
42510              
42511         }
42512         
42513         
42514     },
42515     setFromData: function(v)
42516     {
42517         // this recieves an object, if setValues is called.
42518         this.reset();
42519         this.el.dom.value = v[this.displayField];
42520         this.hiddenEl.dom.value = v[this.valueField];
42521         if (typeof(v[this.valueField]) != 'string' || !v[this.valueField].length) {
42522             return;
42523         }
42524         var kv = v[this.valueField];
42525         var dv = v[this.displayField];
42526         kv = typeof(kv) != 'string' ? '' : kv;
42527         dv = typeof(dv) != 'string' ? '' : dv;
42528         
42529         
42530         var keys = kv.split(',');
42531         var display = dv.split(',');
42532         for (var i = 0 ; i < keys.length; i++) {
42533             
42534             add = {};
42535             add[this.valueField] = keys[i];
42536             add[this.displayField] = display[i];
42537             this.addItem(add);
42538         }
42539       
42540         
42541     },
42542     
42543     /**
42544      * Validates the combox array value
42545      * @return {Boolean} True if the value is valid, else false
42546      */
42547     validate : function(){
42548         if(this.disabled || this.validateValue(this.processValue(this.getValue()))){
42549             this.clearInvalid();
42550             return true;
42551         }
42552         return false;
42553     },
42554     
42555     validateValue : function(value){
42556         return Roo.form.ComboBoxArray.superclass.validateValue.call(this, this.getValue());
42557         
42558     },
42559     
42560     /*@
42561      * overide
42562      * 
42563      */
42564     isDirty : function() {
42565         if(this.disabled) {
42566             return false;
42567         }
42568         
42569         try {
42570             var d = Roo.decode(String(this.originalValue));
42571         } catch (e) {
42572             return String(this.getValue()) !== String(this.originalValue);
42573         }
42574         
42575         var originalValue = [];
42576         
42577         for (var i = 0; i < d.length; i++){
42578             originalValue.push(d[i][this.valueField]);
42579         }
42580         
42581         return String(this.getValue()) !== String(originalValue.join(','));
42582         
42583     }
42584     
42585 });
42586
42587
42588
42589 /**
42590  * @class Roo.form.ComboBoxArray.Item
42591  * @extends Roo.BoxComponent
42592  * A selected item in the list
42593  *  Fred [x]  Brian [x]  [Pick another |v]
42594  * 
42595  * @constructor
42596  * Create a new item.
42597  * @param {Object} config Configuration options
42598  */
42599  
42600 Roo.form.ComboBoxArray.Item = function(config) {
42601     config.id = Roo.id();
42602     Roo.form.ComboBoxArray.Item.superclass.constructor.call(this, config);
42603 }
42604
42605 Roo.extend(Roo.form.ComboBoxArray.Item, Roo.BoxComponent, {
42606     data : {},
42607     cb: false,
42608     displayField : false,
42609     tipField : false,
42610     
42611     
42612     defaultAutoCreate : {
42613         tag: 'div',
42614         cls: 'x-cbarray-item',
42615         cn : [ 
42616             { tag: 'div' },
42617             {
42618                 tag: 'img',
42619                 width:16,
42620                 height : 16,
42621                 src : Roo.BLANK_IMAGE_URL ,
42622                 align: 'center'
42623             }
42624         ]
42625         
42626     },
42627     
42628  
42629     onRender : function(ct, position)
42630     {
42631         Roo.form.Field.superclass.onRender.call(this, ct, position);
42632         
42633         if(!this.el){
42634             var cfg = this.getAutoCreate();
42635             this.el = ct.createChild(cfg, position);
42636         }
42637         
42638         this.el.child('img').dom.setAttribute('src', Roo.BLANK_IMAGE_URL);
42639         
42640         this.el.child('div').dom.innerHTML = this.cb.renderer ? 
42641             this.cb.renderer(this.data) :
42642             String.format('{0}',this.data[this.displayField]);
42643         
42644             
42645         this.el.child('div').dom.setAttribute('qtip',
42646                         String.format('{0}',this.data[this.tipField])
42647         );
42648         
42649         this.el.child('img').on('click', this.remove, this);
42650         
42651     },
42652    
42653     remove : function()
42654     {
42655         if(this.cb.disabled){
42656             return;
42657         }
42658         
42659         if(false !== this.cb.fireEvent('beforeremove', this.cb, this)){
42660             this.cb.items.remove(this);
42661             this.el.child('img').un('click', this.remove, this);
42662             this.el.remove();
42663             this.cb.updateHiddenEl();
42664
42665             this.cb.fireEvent('remove', this.cb, this);
42666         }
42667         
42668     }
42669 });/*
42670  * RooJS Library 1.1.1
42671  * Copyright(c) 2008-2011  Alan Knowles
42672  *
42673  * License - LGPL
42674  */
42675  
42676
42677 /**
42678  * @class Roo.form.ComboNested
42679  * @extends Roo.form.ComboBox
42680  * A combobox for that allows selection of nested items in a list,
42681  * eg.
42682  *
42683  *  Book
42684  *    -> red
42685  *    -> green
42686  *  Table
42687  *    -> square
42688  *      ->red
42689  *      ->green
42690  *    -> rectangle
42691  *      ->green
42692  *      
42693  * 
42694  * @constructor
42695  * Create a new ComboNested
42696  * @param {Object} config Configuration options
42697  */
42698 Roo.form.ComboNested = function(config){
42699     Roo.form.ComboCheck.superclass.constructor.call(this, config);
42700     // should verify some data...
42701     // like
42702     // hiddenName = required..
42703     // displayField = required
42704     // valudField == required
42705     var req= [ 'hiddenName', 'displayField', 'valueField' ];
42706     var _t = this;
42707     Roo.each(req, function(e) {
42708         if ((typeof(_t[e]) == 'undefined' ) || !_t[e].length) {
42709             throw "Roo.form.ComboNested : missing value for: " + e;
42710         }
42711     });
42712      
42713     
42714 };
42715
42716 Roo.extend(Roo.form.ComboNested, Roo.form.ComboBox, {
42717    
42718     /*
42719      * @config {Number} max Number of columns to show
42720      */
42721     
42722     maxColumns : 3,
42723    
42724     list : null, // the outermost div..
42725     innerLists : null, // the
42726     views : null,
42727     stores : null,
42728     // private
42729     onRender : function(ct, position)
42730     {
42731         Roo.form.ComboBox.superclass.onRender.call(this, ct, position); // skip parent call - got to above..
42732         
42733         if(this.hiddenName){
42734             this.hiddenField = this.el.insertSibling({tag:'input', type:'hidden', name: this.hiddenName, id:  (this.hiddenId||this.hiddenName)},
42735                     'before', true);
42736             this.hiddenField.value =
42737                 this.hiddenValue !== undefined ? this.hiddenValue :
42738                 this.value !== undefined ? this.value : '';
42739
42740             // prevent input submission
42741             this.el.dom.removeAttribute('name');
42742              
42743              
42744         }
42745         
42746         if(Roo.isGecko){
42747             this.el.dom.setAttribute('autocomplete', 'off');
42748         }
42749
42750         var cls = 'x-combo-list';
42751
42752         this.list = new Roo.Layer({
42753             shadow: this.shadow, cls: [cls, this.listClass].join(' '), constrain:false
42754         });
42755
42756         var lw = this.listWidth || Math.max(this.wrap.getWidth(), this.minListWidth);
42757         this.list.setWidth(lw);
42758         this.list.swallowEvent('mousewheel');
42759         this.assetHeight = 0;
42760
42761         if(this.title){
42762             this.header = this.list.createChild({cls:cls+'-hd', html: this.title});
42763             this.assetHeight += this.header.getHeight();
42764         }
42765         this.innerLists = [];
42766         this.views = [];
42767         this.stores = [];
42768         for (var i =0 ; i < this.maxColumns; i++) {
42769             this.onRenderList( cls, i);
42770         }
42771         
42772         // always needs footer, as we are going to have an 'OK' button.
42773         this.footer = this.list.createChild({cls:cls+'-ft'});
42774         this.pageTb = new Roo.Toolbar(this.footer);  
42775         var _this = this;
42776         this.pageTb.add(  {
42777             
42778             text: 'Done',
42779             handler: function()
42780             {
42781                 _this.collapse();
42782             }
42783         });
42784         
42785         if ( this.allowBlank && !this.disableClear) {
42786             
42787             this.pageTb.add(new Roo.Toolbar.Fill(), {
42788                 cls: 'x-btn-icon x-btn-clear',
42789                 text: '&#160;',
42790                 handler: function()
42791                 {
42792                     _this.collapse();
42793                     _this.clearValue();
42794                     _this.onSelect(false, -1);
42795                 }
42796             });
42797         }
42798         if (this.footer) {
42799             this.assetHeight += this.footer.getHeight();
42800         }
42801         
42802     },
42803     onRenderList : function (  cls, i)
42804     {
42805         
42806         var lw = Math.floor(
42807                 ((this.listWidth * 3 || Math.max(this.wrap.getWidth(), this.minListWidth)) - this.list.getFrameWidth('lr')) / 3
42808         );
42809         
42810         this.list.setWidth(lw); // default to '1'
42811
42812         var il = this.innerLists[i] = this.list.createChild({cls:cls+'-inner'});
42813         //il.on('mouseover', this.onViewOver, this, { list:  i });
42814         //il.on('mousemove', this.onViewMove, this, { list:  i });
42815         il.setWidth(lw);
42816         il.setStyle({ 'overflow-x' : 'hidden'});
42817
42818         if(!this.tpl){
42819             this.tpl = new Roo.Template({
42820                 html :  '<div class="'+cls+'-item '+cls+'-item-{cn:this.isEmpty}">{' + this.displayField + '}</div>',
42821                 isEmpty: function (value, allValues) {
42822                     //Roo.log(value);
42823                     var dl = typeof(value.data) != 'undefined' ? value.data.length : value.length; ///json is a nested response..
42824                     return dl ? 'has-children' : 'no-children'
42825                 }
42826             });
42827         }
42828         
42829         var store  = this.store;
42830         if (i > 0) {
42831             store  = new Roo.data.SimpleStore({
42832                 //fields : this.store.reader.meta.fields,
42833                 reader : this.store.reader,
42834                 data : [ ]
42835             });
42836         }
42837         this.stores[i]  = store;
42838                 
42839         
42840         
42841         var view = this.views[i] = new Roo.View(
42842             il,
42843             this.tpl,
42844             {
42845                 singleSelect:true,
42846                 store: store,
42847                 selectedClass: this.selectedClass
42848             }
42849         );
42850         view.getEl().setWidth(lw);
42851         view.getEl().setStyle({
42852             position: i < 1 ? 'relative' : 'absolute',
42853             top: 0,
42854             left: (i * lw ) + 'px',
42855             display : i > 0 ? 'none' : 'block'
42856         });
42857         view.on('selectionchange', this.onSelectChange, this, {list : i });
42858         view.on('dblclick', this.onDoubleClick, this, {list : i });
42859         //view.on('click', this.onViewClick, this, { list : i });
42860
42861         store.on('beforeload', this.onBeforeLoad, this);
42862         store.on('load',  this.onLoad, this, { list  : i});
42863         store.on('loadexception', this.onLoadException, this);
42864
42865         // hide the other vies..
42866         
42867         
42868         
42869     },
42870     onResize : function()  {},
42871     
42872     restrictHeight : function()
42873     {
42874         var mh = 0;
42875         Roo.each(this.innerLists, function(il,i) {
42876             var el = this.views[i].getEl();
42877             el.dom.style.height = '';
42878             var inner = el.dom;
42879             var h = Math.max(inner.clientHeight, inner.offsetHeight, inner.scrollHeight);
42880             // only adjust heights on other ones..
42881             if (i < 1) {
42882                 
42883                 el.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
42884                 il.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
42885                 mh = Math.max(el.getHeight(), mh);
42886             }
42887             
42888             
42889         }, this);
42890         
42891         this.list.beginUpdate();
42892         this.list.setHeight(mh+this.list.getFrameWidth('tb')+this.assetHeight);
42893         this.list.alignTo(this.el, this.listAlign);
42894         this.list.endUpdate();
42895         
42896     },
42897      
42898     
42899     // -- store handlers..
42900     // private
42901     onBeforeLoad : function()
42902     {
42903         if(!this.hasFocus){
42904             return;
42905         }
42906         this.innerLists[0].update(this.loadingText ?
42907                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
42908         this.restrictHeight();
42909         this.selectedIndex = -1;
42910     },
42911     // private
42912     onLoad : function(a,b,c,d)
42913     {
42914         
42915         if(!this.hasFocus){
42916             return;
42917         }
42918         
42919         if(this.store.getCount() > 0) {
42920             this.expand();
42921             this.restrictHeight();   
42922         } else {
42923             this.onEmptyResults();
42924         }
42925         /*
42926         this.stores[1].loadData([]);
42927         this.stores[2].loadData([]);
42928         this.views
42929         */    
42930     
42931         //this.el.focus();
42932     },
42933     
42934     
42935     // private
42936     onLoadException : function()
42937     {
42938         this.collapse();
42939         Roo.log(this.store.reader.jsonData);
42940         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
42941             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
42942         }
42943         
42944         
42945     } ,
42946      
42947      
42948
42949     onSelectChange : function (view, sels, opts )
42950     {
42951         var ix = view.getSelectedIndexes();
42952         
42953         
42954         if (opts.list > 1) {
42955              
42956             this.setFromData(ix.length ? view.store.getAt(ix[0]).data : {});
42957             return;
42958         }
42959         
42960         if (!ix.length) {
42961             this.setFromData({});
42962             this.stores[opts.list+1].loadData( [] );
42963             return;
42964         }
42965         
42966         var rec = view.store.getAt(ix[0]);
42967         this.setFromData(rec.data);
42968         
42969         var lw = Math.floor(
42970                 ((this.listWidth * 3 || Math.max(this.wrap.getWidth(), this.minListWidth)) - this.list.getFrameWidth('lr')) / 3
42971         );
42972         var data =  typeof(rec.data.cn) == 'undefined' ? [] : rec.data.cn;
42973         var dl = typeof(data.data) != 'undefined' ? data.total : data.length; ///json is a nested response..
42974         this.stores[opts.list+1].loadData( data );
42975         this.views[opts.list+1].getEl().setHeight( this.innerLists[0].getHeight());
42976         this.views[opts.list+1].getEl().setStyle({ display : dl ? 'block' : 'none' });
42977         this.innerLists[opts.list+1].setHeight( this.innerLists[0].getHeight());
42978         this.list.setWidth(lw * (opts.list + (dl ? 2 : 1))); 
42979     },
42980     onDoubleClick : function()
42981     {
42982         this.collapse(); //??
42983     },
42984     
42985      
42986     
42987     findRecord : function (prop,value)
42988     {
42989         return this.findRecordInStore(this.store, prop,value);
42990     },
42991     
42992      // private
42993     findRecordInStore : function(store, prop, value)
42994     {
42995         var cstore = new Roo.data.SimpleStore({
42996             //fields : this.store.reader.meta.fields, // we need array reader.. for
42997             reader : this.store.reader,
42998             data : [ ]
42999         });
43000         var _this = this;
43001         var record  = false;
43002         if(store.getCount() > 0){
43003            store.each(function(r){
43004                 if(r.data[prop] == value){
43005                     record = r;
43006                     return false;
43007                 }
43008                 if (r.data.cn && r.data.cn.length) {
43009                     cstore.loadData( r.data.cn);
43010                     var cret = _this.findRecordInStore(cstore, prop, value);
43011                     if (cret !== false) {
43012                         record = cret;
43013                         return false;
43014                     }
43015                 }
43016                 
43017                 return true;
43018             });
43019         }
43020         return record;
43021     }
43022     
43023     
43024     
43025     
43026 });/*
43027  * Based on:
43028  * Ext JS Library 1.1.1
43029  * Copyright(c) 2006-2007, Ext JS, LLC.
43030  *
43031  * Originally Released Under LGPL - original licence link has changed is not relivant.
43032  *
43033  * Fork - LGPL
43034  * <script type="text/javascript">
43035  */
43036 /**
43037  * @class Roo.form.Checkbox
43038  * @extends Roo.form.Field
43039  * Single checkbox field.  Can be used as a direct replacement for traditional checkbox fields.
43040  * @constructor
43041  * Creates a new Checkbox
43042  * @param {Object} config Configuration options
43043  */
43044 Roo.form.Checkbox = function(config){
43045     Roo.form.Checkbox.superclass.constructor.call(this, config);
43046     this.addEvents({
43047         /**
43048          * @event check
43049          * Fires when the checkbox is checked or unchecked.
43050              * @param {Roo.form.Checkbox} this This checkbox
43051              * @param {Boolean} checked The new checked value
43052              */
43053         check : true
43054     });
43055 };
43056
43057 Roo.extend(Roo.form.Checkbox, Roo.form.Field,  {
43058     /**
43059      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
43060      */
43061     focusClass : undefined,
43062     /**
43063      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
43064      */
43065     fieldClass: "x-form-field",
43066     /**
43067      * @cfg {Boolean} checked True if the the checkbox should render already checked (defaults to false)
43068      */
43069     checked: false,
43070     /**
43071      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
43072      * {tag: "input", type: "checkbox", autocomplete: "off"})
43073      */
43074     defaultAutoCreate : { tag: "input", type: 'hidden', autocomplete: "off"},
43075     /**
43076      * @cfg {String} boxLabel The text that appears beside the checkbox
43077      */
43078     boxLabel : "",
43079     /**
43080      * @cfg {String} inputValue The value that should go into the generated input element's value attribute
43081      */  
43082     inputValue : '1',
43083     /**
43084      * @cfg {String} valueOff The value that should go into the generated input element's value when unchecked.
43085      */
43086      valueOff: '0', // value when not checked..
43087
43088     actionMode : 'viewEl', 
43089     //
43090     // private
43091     itemCls : 'x-menu-check-item x-form-item',
43092     groupClass : 'x-menu-group-item',
43093     inputType : 'hidden',
43094     
43095     
43096     inSetChecked: false, // check that we are not calling self...
43097     
43098     inputElement: false, // real input element?
43099     basedOn: false, // ????
43100     
43101     isFormField: true, // not sure where this is needed!!!!
43102
43103     onResize : function(){
43104         Roo.form.Checkbox.superclass.onResize.apply(this, arguments);
43105         if(!this.boxLabel){
43106             this.el.alignTo(this.wrap, 'c-c');
43107         }
43108     },
43109
43110     initEvents : function(){
43111         Roo.form.Checkbox.superclass.initEvents.call(this);
43112         this.el.on("click", this.onClick,  this);
43113         this.el.on("change", this.onClick,  this);
43114     },
43115
43116
43117     getResizeEl : function(){
43118         return this.wrap;
43119     },
43120
43121     getPositionEl : function(){
43122         return this.wrap;
43123     },
43124
43125     // private
43126     onRender : function(ct, position){
43127         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
43128         /*
43129         if(this.inputValue !== undefined){
43130             this.el.dom.value = this.inputValue;
43131         }
43132         */
43133         //this.wrap = this.el.wrap({cls: "x-form-check-wrap"});
43134         this.wrap = this.el.wrap({cls: 'x-menu-check-item '});
43135         var viewEl = this.wrap.createChild({ 
43136             tag: 'img', cls: 'x-menu-item-icon', style: 'margin: 0px;' ,src : Roo.BLANK_IMAGE_URL });
43137         this.viewEl = viewEl;   
43138         this.wrap.on('click', this.onClick,  this); 
43139         
43140         this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
43141         this.el.on('propertychange', this.setFromHidden,  this);  //ie
43142         
43143         
43144         
43145         if(this.boxLabel){
43146             this.wrap.createChild({tag: 'label', htmlFor: this.el.id, cls: 'x-form-cb-label', html: this.boxLabel});
43147         //    viewEl.on('click', this.onClick,  this); 
43148         }
43149         //if(this.checked){
43150             this.setChecked(this.checked);
43151         //}else{
43152             //this.checked = this.el.dom;
43153         //}
43154
43155     },
43156
43157     // private
43158     initValue : Roo.emptyFn,
43159
43160     /**
43161      * Returns the checked state of the checkbox.
43162      * @return {Boolean} True if checked, else false
43163      */
43164     getValue : function(){
43165         if(this.el){
43166             return String(this.el.dom.value) == String(this.inputValue ) ? this.inputValue : this.valueOff;
43167         }
43168         return this.valueOff;
43169         
43170     },
43171
43172         // private
43173     onClick : function(){ 
43174         if (this.disabled) {
43175             return;
43176         }
43177         this.setChecked(!this.checked);
43178
43179         //if(this.el.dom.checked != this.checked){
43180         //    this.setValue(this.el.dom.checked);
43181        // }
43182     },
43183
43184     /**
43185      * Sets the checked state of the checkbox.
43186      * On is always based on a string comparison between inputValue and the param.
43187      * @param {Boolean/String} value - the value to set 
43188      * @param {Boolean/String} suppressEvent - whether to suppress the checkchange event.
43189      */
43190     setValue : function(v,suppressEvent){
43191         
43192         
43193         //this.checked = (v === true || v === 'true' || v == '1' || String(v).toLowerCase() == 'on');
43194         //if(this.el && this.el.dom){
43195         //    this.el.dom.checked = this.checked;
43196         //    this.el.dom.defaultChecked = this.checked;
43197         //}
43198         this.setChecked(String(v) === String(this.inputValue), suppressEvent);
43199         //this.fireEvent("check", this, this.checked);
43200     },
43201     // private..
43202     setChecked : function(state,suppressEvent)
43203     {
43204         if (this.inSetChecked) {
43205             this.checked = state;
43206             return;
43207         }
43208         
43209     
43210         if(this.wrap){
43211             this.wrap[state ? 'addClass' : 'removeClass']('x-menu-item-checked');
43212         }
43213         this.checked = state;
43214         if(suppressEvent !== true){
43215             this.fireEvent('check', this, state);
43216         }
43217         this.inSetChecked = true;
43218         this.el.dom.value = state ? this.inputValue : this.valueOff;
43219         this.inSetChecked = false;
43220         
43221     },
43222     // handle setting of hidden value by some other method!!?!?
43223     setFromHidden: function()
43224     {
43225         if(!this.el){
43226             return;
43227         }
43228         //console.log("SET FROM HIDDEN");
43229         //alert('setFrom hidden');
43230         this.setValue(this.el.dom.value);
43231     },
43232     
43233     onDestroy : function()
43234     {
43235         if(this.viewEl){
43236             Roo.get(this.viewEl).remove();
43237         }
43238          
43239         Roo.form.Checkbox.superclass.onDestroy.call(this);
43240     },
43241     
43242     setBoxLabel : function(str)
43243     {
43244         this.wrap.select('.x-form-cb-label', true).first().dom.innerHTML = str;
43245     }
43246
43247 });/*
43248  * Based on:
43249  * Ext JS Library 1.1.1
43250  * Copyright(c) 2006-2007, Ext JS, LLC.
43251  *
43252  * Originally Released Under LGPL - original licence link has changed is not relivant.
43253  *
43254  * Fork - LGPL
43255  * <script type="text/javascript">
43256  */
43257  
43258 /**
43259  * @class Roo.form.Radio
43260  * @extends Roo.form.Checkbox
43261  * Single radio field.  Same as Checkbox, but provided as a convenience for automatically setting the input type.
43262  * Radio grouping is handled automatically by the browser if you give each radio in a group the same name.
43263  * @constructor
43264  * Creates a new Radio
43265  * @param {Object} config Configuration options
43266  */
43267 Roo.form.Radio = function(){
43268     Roo.form.Radio.superclass.constructor.apply(this, arguments);
43269 };
43270 Roo.extend(Roo.form.Radio, Roo.form.Checkbox, {
43271     inputType: 'radio',
43272
43273     /**
43274      * If this radio is part of a group, it will return the selected value
43275      * @return {String}
43276      */
43277     getGroupValue : function(){
43278         return this.el.up('form').child('input[name='+this.el.dom.name+']:checked', true).value;
43279     },
43280     
43281     
43282     onRender : function(ct, position){
43283         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
43284         
43285         if(this.inputValue !== undefined){
43286             this.el.dom.value = this.inputValue;
43287         }
43288          
43289         this.wrap = this.el.wrap({cls: "x-form-check-wrap"});
43290         //this.wrap = this.el.wrap({cls: 'x-menu-check-item '});
43291         //var viewEl = this.wrap.createChild({ 
43292         //    tag: 'img', cls: 'x-menu-item-icon', style: 'margin: 0px;' ,src : Roo.BLANK_IMAGE_URL });
43293         //this.viewEl = viewEl;   
43294         //this.wrap.on('click', this.onClick,  this); 
43295         
43296         //this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
43297         //this.el.on('propertychange', this.setFromHidden,  this);  //ie
43298         
43299         
43300         
43301         if(this.boxLabel){
43302             this.wrap.createChild({tag: 'label', htmlFor: this.el.id, cls: 'x-form-cb-label', html: this.boxLabel});
43303         //    viewEl.on('click', this.onClick,  this); 
43304         }
43305          if(this.checked){
43306             this.el.dom.checked =   'checked' ;
43307         }
43308          
43309     } 
43310     
43311     
43312 });//<script type="text/javascript">
43313
43314 /*
43315  * Based  Ext JS Library 1.1.1
43316  * Copyright(c) 2006-2007, Ext JS, LLC.
43317  * LGPL
43318  *
43319  */
43320  
43321 /**
43322  * @class Roo.HtmlEditorCore
43323  * @extends Roo.Component
43324  * Provides a the editing component for the HTML editors in Roo. (bootstrap and Roo.form)
43325  *
43326  * any element that has display set to 'none' can cause problems in Safari and Firefox.<br/><br/>
43327  */
43328
43329 Roo.HtmlEditorCore = function(config){
43330     
43331     
43332     Roo.HtmlEditorCore.superclass.constructor.call(this, config);
43333     
43334     
43335     this.addEvents({
43336         /**
43337          * @event initialize
43338          * Fires when the editor is fully initialized (including the iframe)
43339          * @param {Roo.HtmlEditorCore} this
43340          */
43341         initialize: true,
43342         /**
43343          * @event activate
43344          * Fires when the editor is first receives the focus. Any insertion must wait
43345          * until after this event.
43346          * @param {Roo.HtmlEditorCore} this
43347          */
43348         activate: true,
43349          /**
43350          * @event beforesync
43351          * Fires before the textarea is updated with content from the editor iframe. Return false
43352          * to cancel the sync.
43353          * @param {Roo.HtmlEditorCore} this
43354          * @param {String} html
43355          */
43356         beforesync: true,
43357          /**
43358          * @event beforepush
43359          * Fires before the iframe editor is updated with content from the textarea. Return false
43360          * to cancel the push.
43361          * @param {Roo.HtmlEditorCore} this
43362          * @param {String} html
43363          */
43364         beforepush: true,
43365          /**
43366          * @event sync
43367          * Fires when the textarea is updated with content from the editor iframe.
43368          * @param {Roo.HtmlEditorCore} this
43369          * @param {String} html
43370          */
43371         sync: true,
43372          /**
43373          * @event push
43374          * Fires when the iframe editor is updated with content from the textarea.
43375          * @param {Roo.HtmlEditorCore} this
43376          * @param {String} html
43377          */
43378         push: true,
43379         
43380         /**
43381          * @event editorevent
43382          * Fires when on any editor (mouse up/down cursor movement etc.) - used for toolbar hooks.
43383          * @param {Roo.HtmlEditorCore} this
43384          */
43385         editorevent: true
43386         
43387     });
43388     
43389     // at this point this.owner is set, so we can start working out the whitelisted / blacklisted elements
43390     
43391     // defaults : white / black...
43392     this.applyBlacklists();
43393     
43394     
43395     
43396 };
43397
43398
43399 Roo.extend(Roo.HtmlEditorCore, Roo.Component,  {
43400
43401
43402      /**
43403      * @cfg {Roo.form.HtmlEditor|Roo.bootstrap.HtmlEditor} the owner field 
43404      */
43405     
43406     owner : false,
43407     
43408      /**
43409      * @cfg {String} resizable  's' or 'se' or 'e' - wrapps the element in a
43410      *                        Roo.resizable.
43411      */
43412     resizable : false,
43413      /**
43414      * @cfg {Number} height (in pixels)
43415      */   
43416     height: 300,
43417    /**
43418      * @cfg {Number} width (in pixels)
43419      */   
43420     width: 500,
43421     
43422     /**
43423      * @cfg {Array} stylesheets url of stylesheets. set to [] to disable stylesheets.
43424      * 
43425      */
43426     stylesheets: false,
43427     
43428     // id of frame..
43429     frameId: false,
43430     
43431     // private properties
43432     validationEvent : false,
43433     deferHeight: true,
43434     initialized : false,
43435     activated : false,
43436     sourceEditMode : false,
43437     onFocus : Roo.emptyFn,
43438     iframePad:3,
43439     hideMode:'offsets',
43440     
43441     clearUp: true,
43442     
43443     // blacklist + whitelisted elements..
43444     black: false,
43445     white: false,
43446      
43447     bodyCls : '',
43448
43449     /**
43450      * Protected method that will not generally be called directly. It
43451      * is called when the editor initializes the iframe with HTML contents. Override this method if you
43452      * want to change the initialization markup of the iframe (e.g. to add stylesheets).
43453      */
43454     getDocMarkup : function(){
43455         // body styles..
43456         var st = '';
43457         
43458         // inherit styels from page...?? 
43459         if (this.stylesheets === false) {
43460             
43461             Roo.get(document.head).select('style').each(function(node) {
43462                 st += node.dom.outerHTML || new XMLSerializer().serializeToString(node.dom);
43463             });
43464             
43465             Roo.get(document.head).select('link').each(function(node) { 
43466                 st += node.dom.outerHTML || new XMLSerializer().serializeToString(node.dom);
43467             });
43468             
43469         } else if (!this.stylesheets.length) {
43470                 // simple..
43471                 st = '<style type="text/css">' +
43472                     'body{border:0;margin:0;padding:3px;height:98%;cursor:text;}' +
43473                    '</style>';
43474         } else { 
43475             st = '<style type="text/css">' +
43476                     this.stylesheets +
43477                 '</style>';
43478         }
43479         
43480         st +=  '<style type="text/css">' +
43481             'IMG { cursor: pointer } ' +
43482         '</style>';
43483
43484         var cls = 'roo-htmleditor-body';
43485         
43486         if(this.bodyCls.length){
43487             cls += ' ' + this.bodyCls;
43488         }
43489         
43490         return '<html><head>' + st  +
43491             //<style type="text/css">' +
43492             //'body{border:0;margin:0;padding:3px;height:98%;cursor:text;}' +
43493             //'</style>' +
43494             ' </head><body class="' +  cls + '"></body></html>';
43495     },
43496
43497     // private
43498     onRender : function(ct, position)
43499     {
43500         var _t = this;
43501         //Roo.HtmlEditorCore.superclass.onRender.call(this, ct, position);
43502         this.el = this.owner.inputEl ? this.owner.inputEl() : this.owner.el;
43503         
43504         
43505         this.el.dom.style.border = '0 none';
43506         this.el.dom.setAttribute('tabIndex', -1);
43507         this.el.addClass('x-hidden hide');
43508         
43509         
43510         
43511         if(Roo.isIE){ // fix IE 1px bogus margin
43512             this.el.applyStyles('margin-top:-1px;margin-bottom:-1px;')
43513         }
43514        
43515         
43516         this.frameId = Roo.id();
43517         
43518          
43519         
43520         var iframe = this.owner.wrap.createChild({
43521             tag: 'iframe',
43522             cls: 'form-control', // bootstrap..
43523             id: this.frameId,
43524             name: this.frameId,
43525             frameBorder : 'no',
43526             'src' : Roo.SSL_SECURE_URL ? Roo.SSL_SECURE_URL  :  "javascript:false"
43527         }, this.el
43528         );
43529         
43530         
43531         this.iframe = iframe.dom;
43532
43533          this.assignDocWin();
43534         
43535         this.doc.designMode = 'on';
43536        
43537         this.doc.open();
43538         this.doc.write(this.getDocMarkup());
43539         this.doc.close();
43540
43541         
43542         var task = { // must defer to wait for browser to be ready
43543             run : function(){
43544                 //console.log("run task?" + this.doc.readyState);
43545                 this.assignDocWin();
43546                 if(this.doc.body || this.doc.readyState == 'complete'){
43547                     try {
43548                         this.doc.designMode="on";
43549                     } catch (e) {
43550                         return;
43551                     }
43552                     Roo.TaskMgr.stop(task);
43553                     this.initEditor.defer(10, this);
43554                 }
43555             },
43556             interval : 10,
43557             duration: 10000,
43558             scope: this
43559         };
43560         Roo.TaskMgr.start(task);
43561
43562     },
43563
43564     // private
43565     onResize : function(w, h)
43566     {
43567          Roo.log('resize: ' +w + ',' + h );
43568         //Roo.HtmlEditorCore.superclass.onResize.apply(this, arguments);
43569         if(!this.iframe){
43570             return;
43571         }
43572         if(typeof w == 'number'){
43573             
43574             this.iframe.style.width = w + 'px';
43575         }
43576         if(typeof h == 'number'){
43577             
43578             this.iframe.style.height = h + 'px';
43579             if(this.doc){
43580                 (this.doc.body || this.doc.documentElement).style.height = (h - (this.iframePad*2)) + 'px';
43581             }
43582         }
43583         
43584     },
43585
43586     /**
43587      * Toggles the editor between standard and source edit mode.
43588      * @param {Boolean} sourceEdit (optional) True for source edit, false for standard
43589      */
43590     toggleSourceEdit : function(sourceEditMode){
43591         
43592         this.sourceEditMode = sourceEditMode === true;
43593         
43594         if(this.sourceEditMode){
43595  
43596             Roo.get(this.iframe).addClass(['x-hidden','hide']);     //FIXME - what's the BS styles for these
43597             
43598         }else{
43599             Roo.get(this.iframe).removeClass(['x-hidden','hide']);
43600             //this.iframe.className = '';
43601             this.deferFocus();
43602         }
43603         //this.setSize(this.owner.wrap.getSize());
43604         //this.fireEvent('editmodechange', this, this.sourceEditMode);
43605     },
43606
43607     
43608   
43609
43610     /**
43611      * Protected method that will not generally be called directly. If you need/want
43612      * custom HTML cleanup, this is the method you should override.
43613      * @param {String} html The HTML to be cleaned
43614      * return {String} The cleaned HTML
43615      */
43616     cleanHtml : function(html){
43617         html = String(html);
43618         if(html.length > 5){
43619             if(Roo.isSafari){ // strip safari nonsense
43620                 html = html.replace(/\sclass="(?:Apple-style-span|khtml-block-placeholder)"/gi, '');
43621             }
43622         }
43623         if(html == '&nbsp;'){
43624             html = '';
43625         }
43626         return html;
43627     },
43628
43629     /**
43630      * HTML Editor -> Textarea
43631      * Protected method that will not generally be called directly. Syncs the contents
43632      * of the editor iframe with the textarea.
43633      */
43634     syncValue : function(){
43635         if(this.initialized){
43636             var bd = (this.doc.body || this.doc.documentElement);
43637             //this.cleanUpPaste(); -- this is done else where and causes havoc..
43638             var html = bd.innerHTML;
43639             if(Roo.isSafari){
43640                 var bs = bd.getAttribute('style'); // Safari puts text-align styles on the body element!
43641                 var m = bs ? bs.match(/text-align:(.*?);/i) : false;
43642                 if(m && m[1]){
43643                     html = '<div style="'+m[0]+'">' + html + '</div>';
43644                 }
43645             }
43646             html = this.cleanHtml(html);
43647             // fix up the special chars.. normaly like back quotes in word...
43648             // however we do not want to do this with chinese..
43649             html = html.replace(/[\uD800-\uDBFF][\uDC00-\uDFFF]|[\u0080-\uFFFF]/g, function(match) {
43650                 
43651                 var cc = match.charCodeAt();
43652
43653                 // Get the character value, handling surrogate pairs
43654                 if (match.length == 2) {
43655                     // It's a surrogate pair, calculate the Unicode code point
43656                     var high = match.charCodeAt(0) - 0xD800;
43657                     var low  = match.charCodeAt(1) - 0xDC00;
43658                     cc = (high * 0x400) + low + 0x10000;
43659                 }  else if (
43660                     (cc >= 0x4E00 && cc < 0xA000 ) ||
43661                     (cc >= 0x3400 && cc < 0x4E00 ) ||
43662                     (cc >= 0xf900 && cc < 0xfb00 )
43663                 ) {
43664                         return match;
43665                 }  
43666          
43667                 // No, use a numeric entity. Here we brazenly (and possibly mistakenly)
43668                 return "&#" + cc + ";";
43669                 
43670                 
43671             });
43672             
43673             
43674              
43675             if(this.owner.fireEvent('beforesync', this, html) !== false){
43676                 this.el.dom.value = html;
43677                 this.owner.fireEvent('sync', this, html);
43678             }
43679         }
43680     },
43681
43682     /**
43683      * Protected method that will not generally be called directly. Pushes the value of the textarea
43684      * into the iframe editor.
43685      */
43686     pushValue : function(){
43687         if(this.initialized){
43688             var v = this.el.dom.value.trim();
43689             
43690 //            if(v.length < 1){
43691 //                v = '&#160;';
43692 //            }
43693             
43694             if(this.owner.fireEvent('beforepush', this, v) !== false){
43695                 var d = (this.doc.body || this.doc.documentElement);
43696                 d.innerHTML = v;
43697                 this.cleanUpPaste();
43698                 this.el.dom.value = d.innerHTML;
43699                 this.owner.fireEvent('push', this, v);
43700             }
43701         }
43702     },
43703
43704     // private
43705     deferFocus : function(){
43706         this.focus.defer(10, this);
43707     },
43708
43709     // doc'ed in Field
43710     focus : function(){
43711         if(this.win && !this.sourceEditMode){
43712             this.win.focus();
43713         }else{
43714             this.el.focus();
43715         }
43716     },
43717     
43718     assignDocWin: function()
43719     {
43720         var iframe = this.iframe;
43721         
43722          if(Roo.isIE){
43723             this.doc = iframe.contentWindow.document;
43724             this.win = iframe.contentWindow;
43725         } else {
43726 //            if (!Roo.get(this.frameId)) {
43727 //                return;
43728 //            }
43729 //            this.doc = (iframe.contentDocument || Roo.get(this.frameId).dom.document);
43730 //            this.win = Roo.get(this.frameId).dom.contentWindow;
43731             
43732             if (!Roo.get(this.frameId) && !iframe.contentDocument) {
43733                 return;
43734             }
43735             
43736             this.doc = (iframe.contentDocument || Roo.get(this.frameId).dom.document);
43737             this.win = (iframe.contentWindow || Roo.get(this.frameId).dom.contentWindow);
43738         }
43739     },
43740     
43741     // private
43742     initEditor : function(){
43743         //console.log("INIT EDITOR");
43744         this.assignDocWin();
43745         
43746         
43747         
43748         this.doc.designMode="on";
43749         this.doc.open();
43750         this.doc.write(this.getDocMarkup());
43751         this.doc.close();
43752         
43753         var dbody = (this.doc.body || this.doc.documentElement);
43754         //var ss = this.el.getStyles('font-size', 'font-family', 'background-image', 'background-repeat');
43755         // this copies styles from the containing element into thsi one..
43756         // not sure why we need all of this..
43757         //var ss = this.el.getStyles('font-size', 'background-image', 'background-repeat');
43758         
43759         //var ss = this.el.getStyles( 'background-image', 'background-repeat');
43760         //ss['background-attachment'] = 'fixed'; // w3c
43761         dbody.bgProperties = 'fixed'; // ie
43762         //Roo.DomHelper.applyStyles(dbody, ss);
43763         Roo.EventManager.on(this.doc, {
43764             //'mousedown': this.onEditorEvent,
43765             'mouseup': this.onEditorEvent,
43766             'dblclick': this.onEditorEvent,
43767             'click': this.onEditorEvent,
43768             'keyup': this.onEditorEvent,
43769             buffer:100,
43770             scope: this
43771         });
43772         if(Roo.isGecko){
43773             Roo.EventManager.on(this.doc, 'keypress', this.mozKeyPress, this);
43774         }
43775         if(Roo.isIE || Roo.isSafari || Roo.isOpera){
43776             Roo.EventManager.on(this.doc, 'keydown', this.fixKeys, this);
43777         }
43778         this.initialized = true;
43779
43780         this.owner.fireEvent('initialize', this);
43781         this.pushValue();
43782     },
43783
43784     // private
43785     onDestroy : function(){
43786         
43787         
43788         
43789         if(this.rendered){
43790             
43791             //for (var i =0; i < this.toolbars.length;i++) {
43792             //    // fixme - ask toolbars for heights?
43793             //    this.toolbars[i].onDestroy();
43794            // }
43795             
43796             //this.wrap.dom.innerHTML = '';
43797             //this.wrap.remove();
43798         }
43799     },
43800
43801     // private
43802     onFirstFocus : function(){
43803         
43804         this.assignDocWin();
43805         
43806         
43807         this.activated = true;
43808          
43809     
43810         if(Roo.isGecko){ // prevent silly gecko errors
43811             this.win.focus();
43812             var s = this.win.getSelection();
43813             if(!s.focusNode || s.focusNode.nodeType != 3){
43814                 var r = s.getRangeAt(0);
43815                 r.selectNodeContents((this.doc.body || this.doc.documentElement));
43816                 r.collapse(true);
43817                 this.deferFocus();
43818             }
43819             try{
43820                 this.execCmd('useCSS', true);
43821                 this.execCmd('styleWithCSS', false);
43822             }catch(e){}
43823         }
43824         this.owner.fireEvent('activate', this);
43825     },
43826
43827     // private
43828     adjustFont: function(btn){
43829         var adjust = btn.cmd == 'increasefontsize' ? 1 : -1;
43830         //if(Roo.isSafari){ // safari
43831         //    adjust *= 2;
43832        // }
43833         var v = parseInt(this.doc.queryCommandValue('FontSize')|| 3, 10);
43834         if(Roo.isSafari){ // safari
43835             var sm = { 10 : 1, 13: 2, 16:3, 18:4, 24: 5, 32:6, 48: 7 };
43836             v =  (v < 10) ? 10 : v;
43837             v =  (v > 48) ? 48 : v;
43838             v = typeof(sm[v]) == 'undefined' ? 1 : sm[v];
43839             
43840         }
43841         
43842         
43843         v = Math.max(1, v+adjust);
43844         
43845         this.execCmd('FontSize', v  );
43846     },
43847
43848     onEditorEvent : function(e)
43849     {
43850         this.owner.fireEvent('editorevent', this, e);
43851       //  this.updateToolbar();
43852         this.syncValue(); //we can not sync so often.. sync cleans, so this breaks stuff
43853     },
43854
43855     insertTag : function(tg)
43856     {
43857         // could be a bit smarter... -> wrap the current selected tRoo..
43858         if (tg.toLowerCase() == 'span' ||
43859             tg.toLowerCase() == 'code' ||
43860             tg.toLowerCase() == 'sup' ||
43861             tg.toLowerCase() == 'sub' 
43862             ) {
43863             
43864             range = this.createRange(this.getSelection());
43865             var wrappingNode = this.doc.createElement(tg.toLowerCase());
43866             wrappingNode.appendChild(range.extractContents());
43867             range.insertNode(wrappingNode);
43868
43869             return;
43870             
43871             
43872             
43873         }
43874         this.execCmd("formatblock",   tg);
43875         
43876     },
43877     
43878     insertText : function(txt)
43879     {
43880         
43881         
43882         var range = this.createRange();
43883         range.deleteContents();
43884                //alert(Sender.getAttribute('label'));
43885                
43886         range.insertNode(this.doc.createTextNode(txt));
43887     } ,
43888     
43889      
43890
43891     /**
43892      * Executes a Midas editor command on the editor document and performs necessary focus and
43893      * toolbar updates. <b>This should only be called after the editor is initialized.</b>
43894      * @param {String} cmd The Midas command
43895      * @param {String/Boolean} value (optional) The value to pass to the command (defaults to null)
43896      */
43897     relayCmd : function(cmd, value){
43898         this.win.focus();
43899         this.execCmd(cmd, value);
43900         this.owner.fireEvent('editorevent', this);
43901         //this.updateToolbar();
43902         this.owner.deferFocus();
43903     },
43904
43905     /**
43906      * Executes a Midas editor command directly on the editor document.
43907      * For visual commands, you should use {@link #relayCmd} instead.
43908      * <b>This should only be called after the editor is initialized.</b>
43909      * @param {String} cmd The Midas command
43910      * @param {String/Boolean} value (optional) The value to pass to the command (defaults to null)
43911      */
43912     execCmd : function(cmd, value){
43913         this.doc.execCommand(cmd, false, value === undefined ? null : value);
43914         this.syncValue();
43915     },
43916  
43917  
43918    
43919     /**
43920      * Inserts the passed text at the current cursor position. Note: the editor must be initialized and activated
43921      * to insert tRoo.
43922      * @param {String} text | dom node.. 
43923      */
43924     insertAtCursor : function(text)
43925     {
43926         
43927         if(!this.activated){
43928             return;
43929         }
43930         /*
43931         if(Roo.isIE){
43932             this.win.focus();
43933             var r = this.doc.selection.createRange();
43934             if(r){
43935                 r.collapse(true);
43936                 r.pasteHTML(text);
43937                 this.syncValue();
43938                 this.deferFocus();
43939             
43940             }
43941             return;
43942         }
43943         */
43944         if(Roo.isGecko || Roo.isOpera || Roo.isSafari){
43945             this.win.focus();
43946             
43947             
43948             // from jquery ui (MIT licenced)
43949             var range, node;
43950             var win = this.win;
43951             
43952             if (win.getSelection && win.getSelection().getRangeAt) {
43953                 range = win.getSelection().getRangeAt(0);
43954                 node = typeof(text) == 'string' ? range.createContextualFragment(text) : text;
43955                 range.insertNode(node);
43956             } else if (win.document.selection && win.document.selection.createRange) {
43957                 // no firefox support
43958                 var txt = typeof(text) == 'string' ? text : text.outerHTML;
43959                 win.document.selection.createRange().pasteHTML(txt);
43960             } else {
43961                 // no firefox support
43962                 var txt = typeof(text) == 'string' ? text : text.outerHTML;
43963                 this.execCmd('InsertHTML', txt);
43964             } 
43965             
43966             this.syncValue();
43967             
43968             this.deferFocus();
43969         }
43970     },
43971  // private
43972     mozKeyPress : function(e){
43973         if(e.ctrlKey){
43974             var c = e.getCharCode(), cmd;
43975           
43976             if(c > 0){
43977                 c = String.fromCharCode(c).toLowerCase();
43978                 switch(c){
43979                     case 'b':
43980                         cmd = 'bold';
43981                         break;
43982                     case 'i':
43983                         cmd = 'italic';
43984                         break;
43985                     
43986                     case 'u':
43987                         cmd = 'underline';
43988                         break;
43989                     
43990                     case 'v':
43991                         this.cleanUpPaste.defer(100, this);
43992                         return;
43993                         
43994                 }
43995                 if(cmd){
43996                     this.win.focus();
43997                     this.execCmd(cmd);
43998                     this.deferFocus();
43999                     e.preventDefault();
44000                 }
44001                 
44002             }
44003         }
44004     },
44005
44006     // private
44007     fixKeys : function(){ // load time branching for fastest keydown performance
44008         if(Roo.isIE){
44009             return function(e){
44010                 var k = e.getKey(), r;
44011                 if(k == e.TAB){
44012                     e.stopEvent();
44013                     r = this.doc.selection.createRange();
44014                     if(r){
44015                         r.collapse(true);
44016                         r.pasteHTML('&#160;&#160;&#160;&#160;');
44017                         this.deferFocus();
44018                     }
44019                     return;
44020                 }
44021                 
44022                 if(k == e.ENTER){
44023                     r = this.doc.selection.createRange();
44024                     if(r){
44025                         var target = r.parentElement();
44026                         if(!target || target.tagName.toLowerCase() != 'li'){
44027                             e.stopEvent();
44028                             r.pasteHTML('<br />');
44029                             r.collapse(false);
44030                             r.select();
44031                         }
44032                     }
44033                 }
44034                 if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44035                     this.cleanUpPaste.defer(100, this);
44036                     return;
44037                 }
44038                 
44039                 
44040             };
44041         }else if(Roo.isOpera){
44042             return function(e){
44043                 var k = e.getKey();
44044                 if(k == e.TAB){
44045                     e.stopEvent();
44046                     this.win.focus();
44047                     this.execCmd('InsertHTML','&#160;&#160;&#160;&#160;');
44048                     this.deferFocus();
44049                 }
44050                 if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44051                     this.cleanUpPaste.defer(100, this);
44052                     return;
44053                 }
44054                 
44055             };
44056         }else if(Roo.isSafari){
44057             return function(e){
44058                 var k = e.getKey();
44059                 
44060                 if(k == e.TAB){
44061                     e.stopEvent();
44062                     this.execCmd('InsertText','\t');
44063                     this.deferFocus();
44064                     return;
44065                 }
44066                if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44067                     this.cleanUpPaste.defer(100, this);
44068                     return;
44069                 }
44070                 
44071              };
44072         }
44073     }(),
44074     
44075     getAllAncestors: function()
44076     {
44077         var p = this.getSelectedNode();
44078         var a = [];
44079         if (!p) {
44080             a.push(p); // push blank onto stack..
44081             p = this.getParentElement();
44082         }
44083         
44084         
44085         while (p && (p.nodeType == 1) && (p.tagName.toLowerCase() != 'body')) {
44086             a.push(p);
44087             p = p.parentNode;
44088         }
44089         a.push(this.doc.body);
44090         return a;
44091     },
44092     lastSel : false,
44093     lastSelNode : false,
44094     
44095     
44096     getSelection : function() 
44097     {
44098         this.assignDocWin();
44099         return Roo.isIE ? this.doc.selection : this.win.getSelection();
44100     },
44101     
44102     getSelectedNode: function() 
44103     {
44104         // this may only work on Gecko!!!
44105         
44106         // should we cache this!!!!
44107         
44108         
44109         
44110          
44111         var range = this.createRange(this.getSelection()).cloneRange();
44112         
44113         if (Roo.isIE) {
44114             var parent = range.parentElement();
44115             while (true) {
44116                 var testRange = range.duplicate();
44117                 testRange.moveToElementText(parent);
44118                 if (testRange.inRange(range)) {
44119                     break;
44120                 }
44121                 if ((parent.nodeType != 1) || (parent.tagName.toLowerCase() == 'body')) {
44122                     break;
44123                 }
44124                 parent = parent.parentElement;
44125             }
44126             return parent;
44127         }
44128         
44129         // is ancestor a text element.
44130         var ac =  range.commonAncestorContainer;
44131         if (ac.nodeType == 3) {
44132             ac = ac.parentNode;
44133         }
44134         
44135         var ar = ac.childNodes;
44136          
44137         var nodes = [];
44138         var other_nodes = [];
44139         var has_other_nodes = false;
44140         for (var i=0;i<ar.length;i++) {
44141             if ((ar[i].nodeType == 3) && (!ar[i].data.length)) { // empty text ? 
44142                 continue;
44143             }
44144             // fullly contained node.
44145             
44146             if (this.rangeIntersectsNode(range,ar[i]) && this.rangeCompareNode(range,ar[i]) == 3) {
44147                 nodes.push(ar[i]);
44148                 continue;
44149             }
44150             
44151             // probably selected..
44152             if ((ar[i].nodeType == 1) && this.rangeIntersectsNode(range,ar[i]) && (this.rangeCompareNode(range,ar[i]) > 0)) {
44153                 other_nodes.push(ar[i]);
44154                 continue;
44155             }
44156             // outer..
44157             if (!this.rangeIntersectsNode(range,ar[i])|| (this.rangeCompareNode(range,ar[i]) == 0))  {
44158                 continue;
44159             }
44160             
44161             
44162             has_other_nodes = true;
44163         }
44164         if (!nodes.length && other_nodes.length) {
44165             nodes= other_nodes;
44166         }
44167         if (has_other_nodes || !nodes.length || (nodes.length > 1)) {
44168             return false;
44169         }
44170         
44171         return nodes[0];
44172     },
44173     createRange: function(sel)
44174     {
44175         // this has strange effects when using with 
44176         // top toolbar - not sure if it's a great idea.
44177         //this.editor.contentWindow.focus();
44178         if (typeof sel != "undefined") {
44179             try {
44180                 return sel.getRangeAt ? sel.getRangeAt(0) : sel.createRange();
44181             } catch(e) {
44182                 return this.doc.createRange();
44183             }
44184         } else {
44185             return this.doc.createRange();
44186         }
44187     },
44188     getParentElement: function()
44189     {
44190         
44191         this.assignDocWin();
44192         var sel = Roo.isIE ? this.doc.selection : this.win.getSelection();
44193         
44194         var range = this.createRange(sel);
44195          
44196         try {
44197             var p = range.commonAncestorContainer;
44198             while (p.nodeType == 3) { // text node
44199                 p = p.parentNode;
44200             }
44201             return p;
44202         } catch (e) {
44203             return null;
44204         }
44205     
44206     },
44207     /***
44208      *
44209      * Range intersection.. the hard stuff...
44210      *  '-1' = before
44211      *  '0' = hits..
44212      *  '1' = after.
44213      *         [ -- selected range --- ]
44214      *   [fail]                        [fail]
44215      *
44216      *    basically..
44217      *      if end is before start or  hits it. fail.
44218      *      if start is after end or hits it fail.
44219      *
44220      *   if either hits (but other is outside. - then it's not 
44221      *   
44222      *    
44223      **/
44224     
44225     
44226     // @see http://www.thismuchiknow.co.uk/?p=64.
44227     rangeIntersectsNode : function(range, node)
44228     {
44229         var nodeRange = node.ownerDocument.createRange();
44230         try {
44231             nodeRange.selectNode(node);
44232         } catch (e) {
44233             nodeRange.selectNodeContents(node);
44234         }
44235     
44236         var rangeStartRange = range.cloneRange();
44237         rangeStartRange.collapse(true);
44238     
44239         var rangeEndRange = range.cloneRange();
44240         rangeEndRange.collapse(false);
44241     
44242         var nodeStartRange = nodeRange.cloneRange();
44243         nodeStartRange.collapse(true);
44244     
44245         var nodeEndRange = nodeRange.cloneRange();
44246         nodeEndRange.collapse(false);
44247     
44248         return rangeStartRange.compareBoundaryPoints(
44249                  Range.START_TO_START, nodeEndRange) == -1 &&
44250                rangeEndRange.compareBoundaryPoints(
44251                  Range.START_TO_START, nodeStartRange) == 1;
44252         
44253          
44254     },
44255     rangeCompareNode : function(range, node)
44256     {
44257         var nodeRange = node.ownerDocument.createRange();
44258         try {
44259             nodeRange.selectNode(node);
44260         } catch (e) {
44261             nodeRange.selectNodeContents(node);
44262         }
44263         
44264         
44265         range.collapse(true);
44266     
44267         nodeRange.collapse(true);
44268      
44269         var ss = range.compareBoundaryPoints( Range.START_TO_START, nodeRange);
44270         var ee = range.compareBoundaryPoints(  Range.END_TO_END, nodeRange);
44271          
44272         //Roo.log(node.tagName + ': ss='+ss +', ee='+ee)
44273         
44274         var nodeIsBefore   =  ss == 1;
44275         var nodeIsAfter    = ee == -1;
44276         
44277         if (nodeIsBefore && nodeIsAfter) {
44278             return 0; // outer
44279         }
44280         if (!nodeIsBefore && nodeIsAfter) {
44281             return 1; //right trailed.
44282         }
44283         
44284         if (nodeIsBefore && !nodeIsAfter) {
44285             return 2;  // left trailed.
44286         }
44287         // fully contined.
44288         return 3;
44289     },
44290
44291     // private? - in a new class?
44292     cleanUpPaste :  function()
44293     {
44294         // cleans up the whole document..
44295         Roo.log('cleanuppaste');
44296         
44297         this.cleanUpChildren(this.doc.body);
44298         var clean = this.cleanWordChars(this.doc.body.innerHTML);
44299         if (clean != this.doc.body.innerHTML) {
44300             this.doc.body.innerHTML = clean;
44301         }
44302         
44303     },
44304     
44305     cleanWordChars : function(input) {// change the chars to hex code
44306         var he = Roo.HtmlEditorCore;
44307         
44308         var output = input;
44309         Roo.each(he.swapCodes, function(sw) { 
44310             var swapper = new RegExp("\\u" + sw[0].toString(16), "g"); // hex codes
44311             
44312             output = output.replace(swapper, sw[1]);
44313         });
44314         
44315         return output;
44316     },
44317     
44318     
44319     cleanUpChildren : function (n)
44320     {
44321         if (!n.childNodes.length) {
44322             return;
44323         }
44324         for (var i = n.childNodes.length-1; i > -1 ; i--) {
44325            this.cleanUpChild(n.childNodes[i]);
44326         }
44327     },
44328     
44329     
44330         
44331     
44332     cleanUpChild : function (node)
44333     {
44334         var ed = this;
44335         //console.log(node);
44336         if (node.nodeName == "#text") {
44337             // clean up silly Windows -- stuff?
44338             return; 
44339         }
44340         if (node.nodeName == "#comment") {
44341             node.parentNode.removeChild(node);
44342             // clean up silly Windows -- stuff?
44343             return; 
44344         }
44345         var lcname = node.tagName.toLowerCase();
44346         // we ignore whitelists... ?? = not really the way to go, but we probably have not got a full
44347         // whitelist of tags..
44348         
44349         if (this.black.indexOf(lcname) > -1 && this.clearUp ) {
44350             // remove node.
44351             node.parentNode.removeChild(node);
44352             return;
44353             
44354         }
44355         
44356         var remove_keep_children= Roo.HtmlEditorCore.remove.indexOf(node.tagName.toLowerCase()) > -1;
44357         
44358         // spans with no attributes - just remove them..
44359         if ((!node.attributes || !node.attributes.length) && lcname == 'span') { 
44360             remove_keep_children = true;
44361         }
44362         
44363         // remove <a name=....> as rendering on yahoo mailer is borked with this.
44364         // this will have to be flaged elsewhere - perhaps ablack=name... on the mailer..
44365         
44366         //if (node.tagName.toLowerCase() == 'a' && !node.hasAttribute('href')) {
44367         //    remove_keep_children = true;
44368         //}
44369         
44370         if (remove_keep_children) {
44371             this.cleanUpChildren(node);
44372             // inserts everything just before this node...
44373             while (node.childNodes.length) {
44374                 var cn = node.childNodes[0];
44375                 node.removeChild(cn);
44376                 node.parentNode.insertBefore(cn, node);
44377             }
44378             node.parentNode.removeChild(node);
44379             return;
44380         }
44381         
44382         if (!node.attributes || !node.attributes.length) {
44383             
44384           
44385             
44386             
44387             this.cleanUpChildren(node);
44388             return;
44389         }
44390         
44391         function cleanAttr(n,v)
44392         {
44393             
44394             if (v.match(/^\./) || v.match(/^\//)) {
44395                 return;
44396             }
44397             if (v.match(/^(http|https):\/\//) || v.match(/^mailto:/) || v.match(/^ftp:/)) {
44398                 return;
44399             }
44400             if (v.match(/^#/)) {
44401                 return;
44402             }
44403 //            Roo.log("(REMOVE TAG)"+ node.tagName +'.' + n + '=' + v);
44404             node.removeAttribute(n);
44405             
44406         }
44407         
44408         var cwhite = this.cwhite;
44409         var cblack = this.cblack;
44410             
44411         function cleanStyle(n,v)
44412         {
44413             if (v.match(/expression/)) { //XSS?? should we even bother..
44414                 node.removeAttribute(n);
44415                 return;
44416             }
44417             
44418             var parts = v.split(/;/);
44419             var clean = [];
44420             
44421             Roo.each(parts, function(p) {
44422                 p = p.replace(/^\s+/g,'').replace(/\s+$/g,'');
44423                 if (!p.length) {
44424                     return true;
44425                 }
44426                 var l = p.split(':').shift().replace(/\s+/g,'');
44427                 l = l.replace(/^\s+/g,'').replace(/\s+$/g,'');
44428                 
44429                 if ( cwhite.length && cblack.indexOf(l) > -1) {
44430 //                    Roo.log('(REMOVE CSS)' + node.tagName +'.' + n + ':'+l + '=' + v);
44431                     //node.removeAttribute(n);
44432                     return true;
44433                 }
44434                 //Roo.log()
44435                 // only allow 'c whitelisted system attributes'
44436                 if ( cwhite.length &&  cwhite.indexOf(l) < 0) {
44437 //                    Roo.log('(REMOVE CSS)' + node.tagName +'.' + n + ':'+l + '=' + v);
44438                     //node.removeAttribute(n);
44439                     return true;
44440                 }
44441                 
44442                 
44443                  
44444                 
44445                 clean.push(p);
44446                 return true;
44447             });
44448             if (clean.length) { 
44449                 node.setAttribute(n, clean.join(';'));
44450             } else {
44451                 node.removeAttribute(n);
44452             }
44453             
44454         }
44455         
44456         
44457         for (var i = node.attributes.length-1; i > -1 ; i--) {
44458             var a = node.attributes[i];
44459             //console.log(a);
44460             
44461             if (a.name.toLowerCase().substr(0,2)=='on')  {
44462                 node.removeAttribute(a.name);
44463                 continue;
44464             }
44465             if (Roo.HtmlEditorCore.ablack.indexOf(a.name.toLowerCase()) > -1) {
44466                 node.removeAttribute(a.name);
44467                 continue;
44468             }
44469             if (Roo.HtmlEditorCore.aclean.indexOf(a.name.toLowerCase()) > -1) {
44470                 cleanAttr(a.name,a.value); // fixme..
44471                 continue;
44472             }
44473             if (a.name == 'style') {
44474                 cleanStyle(a.name,a.value);
44475                 continue;
44476             }
44477             /// clean up MS crap..
44478             // tecnically this should be a list of valid class'es..
44479             
44480             
44481             if (a.name == 'class') {
44482                 if (a.value.match(/^Mso/)) {
44483                     node.removeAttribute('class');
44484                 }
44485                 
44486                 if (a.value.match(/^body$/)) {
44487                     node.removeAttribute('class');
44488                 }
44489                 continue;
44490             }
44491             
44492             // style cleanup!?
44493             // class cleanup?
44494             
44495         }
44496         
44497         
44498         this.cleanUpChildren(node);
44499         
44500         
44501     },
44502     
44503     /**
44504      * Clean up MS wordisms...
44505      */
44506     cleanWord : function(node)
44507     {
44508         if (!node) {
44509             this.cleanWord(this.doc.body);
44510             return;
44511         }
44512         
44513         if(
44514                 node.nodeName == 'SPAN' &&
44515                 !node.hasAttributes() &&
44516                 node.childNodes.length == 1 &&
44517                 node.firstChild.nodeName == "#text"  
44518         ) {
44519             var textNode = node.firstChild;
44520             node.removeChild(textNode);
44521             if (node.getAttribute('lang') != 'zh-CN') {   // do not space pad on chinese characters..
44522                 node.parentNode.insertBefore(node.ownerDocument.createTextNode(" "), node);
44523             }
44524             node.parentNode.insertBefore(textNode, node);
44525             if (node.getAttribute('lang') != 'zh-CN') {   // do not space pad on chinese characters..
44526                 node.parentNode.insertBefore(node.ownerDocument.createTextNode(" ") , node);
44527             }
44528             node.parentNode.removeChild(node);
44529         }
44530         
44531         if (node.nodeName == "#text") {
44532             // clean up silly Windows -- stuff?
44533             return; 
44534         }
44535         if (node.nodeName == "#comment") {
44536             node.parentNode.removeChild(node);
44537             // clean up silly Windows -- stuff?
44538             return; 
44539         }
44540         
44541         if (node.tagName.toLowerCase().match(/^(style|script|applet|embed|noframes|noscript)$/)) {
44542             node.parentNode.removeChild(node);
44543             return;
44544         }
44545         //Roo.log(node.tagName);
44546         // remove - but keep children..
44547         if (node.tagName.toLowerCase().match(/^(meta|link|\\?xml:|st1:|o:|v:|font)/)) {
44548             //Roo.log('-- removed');
44549             while (node.childNodes.length) {
44550                 var cn = node.childNodes[0];
44551                 node.removeChild(cn);
44552                 node.parentNode.insertBefore(cn, node);
44553                 // move node to parent - and clean it..
44554                 this.cleanWord(cn);
44555             }
44556             node.parentNode.removeChild(node);
44557             /// no need to iterate chidlren = it's got none..
44558             //this.iterateChildren(node, this.cleanWord);
44559             return;
44560         }
44561         // clean styles
44562         if (node.className.length) {
44563             
44564             var cn = node.className.split(/\W+/);
44565             var cna = [];
44566             Roo.each(cn, function(cls) {
44567                 if (cls.match(/Mso[a-zA-Z]+/)) {
44568                     return;
44569                 }
44570                 cna.push(cls);
44571             });
44572             node.className = cna.length ? cna.join(' ') : '';
44573             if (!cna.length) {
44574                 node.removeAttribute("class");
44575             }
44576         }
44577         
44578         if (node.hasAttribute("lang")) {
44579             node.removeAttribute("lang");
44580         }
44581         
44582         if (node.hasAttribute("style")) {
44583             
44584             var styles = node.getAttribute("style").split(";");
44585             var nstyle = [];
44586             Roo.each(styles, function(s) {
44587                 if (!s.match(/:/)) {
44588                     return;
44589                 }
44590                 var kv = s.split(":");
44591                 if (kv[0].match(/^(mso-|line|font|background|margin|padding|color)/)) {
44592                     return;
44593                 }
44594                 // what ever is left... we allow.
44595                 nstyle.push(s);
44596             });
44597             node.setAttribute("style", nstyle.length ? nstyle.join(';') : '');
44598             if (!nstyle.length) {
44599                 node.removeAttribute('style');
44600             }
44601         }
44602         this.iterateChildren(node, this.cleanWord);
44603         
44604         
44605         
44606     },
44607     /**
44608      * iterateChildren of a Node, calling fn each time, using this as the scole..
44609      * @param {DomNode} node node to iterate children of.
44610      * @param {Function} fn method of this class to call on each item.
44611      */
44612     iterateChildren : function(node, fn)
44613     {
44614         if (!node.childNodes.length) {
44615                 return;
44616         }
44617         for (var i = node.childNodes.length-1; i > -1 ; i--) {
44618            fn.call(this, node.childNodes[i])
44619         }
44620     },
44621     
44622     
44623     /**
44624      * cleanTableWidths.
44625      *
44626      * Quite often pasting from word etc.. results in tables with column and widths.
44627      * This does not work well on fluid HTML layouts - like emails. - so this code should hunt an destroy them..
44628      *
44629      */
44630     cleanTableWidths : function(node)
44631     {
44632          
44633          
44634         if (!node) {
44635             this.cleanTableWidths(this.doc.body);
44636             return;
44637         }
44638         
44639         // ignore list...
44640         if (node.nodeName == "#text" || node.nodeName == "#comment") {
44641             return; 
44642         }
44643         Roo.log(node.tagName);
44644         if (!node.tagName.toLowerCase().match(/^(table|td|tr)$/)) {
44645             this.iterateChildren(node, this.cleanTableWidths);
44646             return;
44647         }
44648         if (node.hasAttribute('width')) {
44649             node.removeAttribute('width');
44650         }
44651         
44652          
44653         if (node.hasAttribute("style")) {
44654             // pretty basic...
44655             
44656             var styles = node.getAttribute("style").split(";");
44657             var nstyle = [];
44658             Roo.each(styles, function(s) {
44659                 if (!s.match(/:/)) {
44660                     return;
44661                 }
44662                 var kv = s.split(":");
44663                 if (kv[0].match(/^\s*(width|min-width)\s*$/)) {
44664                     return;
44665                 }
44666                 // what ever is left... we allow.
44667                 nstyle.push(s);
44668             });
44669             node.setAttribute("style", nstyle.length ? nstyle.join(';') : '');
44670             if (!nstyle.length) {
44671                 node.removeAttribute('style');
44672             }
44673         }
44674         
44675         this.iterateChildren(node, this.cleanTableWidths);
44676         
44677         
44678     },
44679     
44680     
44681     
44682     
44683     domToHTML : function(currentElement, depth, nopadtext) {
44684         
44685         depth = depth || 0;
44686         nopadtext = nopadtext || false;
44687     
44688         if (!currentElement) {
44689             return this.domToHTML(this.doc.body);
44690         }
44691         
44692         //Roo.log(currentElement);
44693         var j;
44694         var allText = false;
44695         var nodeName = currentElement.nodeName;
44696         var tagName = Roo.util.Format.htmlEncode(currentElement.tagName);
44697         
44698         if  (nodeName == '#text') {
44699             
44700             return nopadtext ? currentElement.nodeValue : currentElement.nodeValue.trim();
44701         }
44702         
44703         
44704         var ret = '';
44705         if (nodeName != 'BODY') {
44706              
44707             var i = 0;
44708             // Prints the node tagName, such as <A>, <IMG>, etc
44709             if (tagName) {
44710                 var attr = [];
44711                 for(i = 0; i < currentElement.attributes.length;i++) {
44712                     // quoting?
44713                     var aname = currentElement.attributes.item(i).name;
44714                     if (!currentElement.attributes.item(i).value.length) {
44715                         continue;
44716                     }
44717                     attr.push(aname + '="' + Roo.util.Format.htmlEncode(currentElement.attributes.item(i).value) + '"' );
44718                 }
44719                 
44720                 ret = "<"+currentElement.tagName+ ( attr.length ? (' ' + attr.join(' ') ) : '') + ">";
44721             } 
44722             else {
44723                 
44724                 // eack
44725             }
44726         } else {
44727             tagName = false;
44728         }
44729         if (['IMG', 'BR', 'HR', 'INPUT'].indexOf(tagName) > -1) {
44730             return ret;
44731         }
44732         if (['PRE', 'TEXTAREA', 'TD', 'A', 'SPAN'].indexOf(tagName) > -1) { // or code?
44733             nopadtext = true;
44734         }
44735         
44736         
44737         // Traverse the tree
44738         i = 0;
44739         var currentElementChild = currentElement.childNodes.item(i);
44740         var allText = true;
44741         var innerHTML  = '';
44742         lastnode = '';
44743         while (currentElementChild) {
44744             // Formatting code (indent the tree so it looks nice on the screen)
44745             var nopad = nopadtext;
44746             if (lastnode == 'SPAN') {
44747                 nopad  = true;
44748             }
44749             // text
44750             if  (currentElementChild.nodeName == '#text') {
44751                 var toadd = Roo.util.Format.htmlEncode(currentElementChild.nodeValue);
44752                 toadd = nopadtext ? toadd : toadd.trim();
44753                 if (!nopad && toadd.length > 80) {
44754                     innerHTML  += "\n" + (new Array( depth + 1 )).join( "  "  );
44755                 }
44756                 innerHTML  += toadd;
44757                 
44758                 i++;
44759                 currentElementChild = currentElement.childNodes.item(i);
44760                 lastNode = '';
44761                 continue;
44762             }
44763             allText = false;
44764             
44765             innerHTML  += nopad ? '' : "\n" + (new Array( depth + 1 )).join( "  "  );
44766                 
44767             // Recursively traverse the tree structure of the child node
44768             innerHTML   += this.domToHTML(currentElementChild, depth+1, nopadtext);
44769             lastnode = currentElementChild.nodeName;
44770             i++;
44771             currentElementChild=currentElement.childNodes.item(i);
44772         }
44773         
44774         ret += innerHTML;
44775         
44776         if (!allText) {
44777                 // The remaining code is mostly for formatting the tree
44778             ret+= nopadtext ? '' : "\n" + (new Array( depth  )).join( "  "  );
44779         }
44780         
44781         
44782         if (tagName) {
44783             ret+= "</"+tagName+">";
44784         }
44785         return ret;
44786         
44787     },
44788         
44789     applyBlacklists : function()
44790     {
44791         var w = typeof(this.owner.white) != 'undefined' && this.owner.white ? this.owner.white  : [];
44792         var b = typeof(this.owner.black) != 'undefined' && this.owner.black ? this.owner.black :  [];
44793         
44794         this.white = [];
44795         this.black = [];
44796         Roo.each(Roo.HtmlEditorCore.white, function(tag) {
44797             if (b.indexOf(tag) > -1) {
44798                 return;
44799             }
44800             this.white.push(tag);
44801             
44802         }, this);
44803         
44804         Roo.each(w, function(tag) {
44805             if (b.indexOf(tag) > -1) {
44806                 return;
44807             }
44808             if (this.white.indexOf(tag) > -1) {
44809                 return;
44810             }
44811             this.white.push(tag);
44812             
44813         }, this);
44814         
44815         
44816         Roo.each(Roo.HtmlEditorCore.black, function(tag) {
44817             if (w.indexOf(tag) > -1) {
44818                 return;
44819             }
44820             this.black.push(tag);
44821             
44822         }, this);
44823         
44824         Roo.each(b, function(tag) {
44825             if (w.indexOf(tag) > -1) {
44826                 return;
44827             }
44828             if (this.black.indexOf(tag) > -1) {
44829                 return;
44830             }
44831             this.black.push(tag);
44832             
44833         }, this);
44834         
44835         
44836         w = typeof(this.owner.cwhite) != 'undefined' && this.owner.cwhite ? this.owner.cwhite  : [];
44837         b = typeof(this.owner.cblack) != 'undefined' && this.owner.cblack ? this.owner.cblack :  [];
44838         
44839         this.cwhite = [];
44840         this.cblack = [];
44841         Roo.each(Roo.HtmlEditorCore.cwhite, function(tag) {
44842             if (b.indexOf(tag) > -1) {
44843                 return;
44844             }
44845             this.cwhite.push(tag);
44846             
44847         }, this);
44848         
44849         Roo.each(w, function(tag) {
44850             if (b.indexOf(tag) > -1) {
44851                 return;
44852             }
44853             if (this.cwhite.indexOf(tag) > -1) {
44854                 return;
44855             }
44856             this.cwhite.push(tag);
44857             
44858         }, this);
44859         
44860         
44861         Roo.each(Roo.HtmlEditorCore.cblack, function(tag) {
44862             if (w.indexOf(tag) > -1) {
44863                 return;
44864             }
44865             this.cblack.push(tag);
44866             
44867         }, this);
44868         
44869         Roo.each(b, function(tag) {
44870             if (w.indexOf(tag) > -1) {
44871                 return;
44872             }
44873             if (this.cblack.indexOf(tag) > -1) {
44874                 return;
44875             }
44876             this.cblack.push(tag);
44877             
44878         }, this);
44879     },
44880     
44881     setStylesheets : function(stylesheets)
44882     {
44883         if(typeof(stylesheets) == 'string'){
44884             Roo.get(this.iframe.contentDocument.head).createChild({
44885                 tag : 'link',
44886                 rel : 'stylesheet',
44887                 type : 'text/css',
44888                 href : stylesheets
44889             });
44890             
44891             return;
44892         }
44893         var _this = this;
44894      
44895         Roo.each(stylesheets, function(s) {
44896             if(!s.length){
44897                 return;
44898             }
44899             
44900             Roo.get(_this.iframe.contentDocument.head).createChild({
44901                 tag : 'link',
44902                 rel : 'stylesheet',
44903                 type : 'text/css',
44904                 href : s
44905             });
44906         });
44907
44908         
44909     },
44910     
44911     removeStylesheets : function()
44912     {
44913         var _this = this;
44914         
44915         Roo.each(Roo.get(_this.iframe.contentDocument.head).select('link[rel=stylesheet]', true).elements, function(s){
44916             s.remove();
44917         });
44918     },
44919     
44920     setStyle : function(style)
44921     {
44922         Roo.get(this.iframe.contentDocument.head).createChild({
44923             tag : 'style',
44924             type : 'text/css',
44925             html : style
44926         });
44927
44928         return;
44929     }
44930     
44931     // hide stuff that is not compatible
44932     /**
44933      * @event blur
44934      * @hide
44935      */
44936     /**
44937      * @event change
44938      * @hide
44939      */
44940     /**
44941      * @event focus
44942      * @hide
44943      */
44944     /**
44945      * @event specialkey
44946      * @hide
44947      */
44948     /**
44949      * @cfg {String} fieldClass @hide
44950      */
44951     /**
44952      * @cfg {String} focusClass @hide
44953      */
44954     /**
44955      * @cfg {String} autoCreate @hide
44956      */
44957     /**
44958      * @cfg {String} inputType @hide
44959      */
44960     /**
44961      * @cfg {String} invalidClass @hide
44962      */
44963     /**
44964      * @cfg {String} invalidText @hide
44965      */
44966     /**
44967      * @cfg {String} msgFx @hide
44968      */
44969     /**
44970      * @cfg {String} validateOnBlur @hide
44971      */
44972 });
44973
44974 Roo.HtmlEditorCore.white = [
44975         'area', 'br', 'img', 'input', 'hr', 'wbr',
44976         
44977        'address', 'blockquote', 'center', 'dd',      'dir',       'div', 
44978        'dl',      'dt',         'h1',     'h2',      'h3',        'h4', 
44979        'h5',      'h6',         'hr',     'isindex', 'listing',   'marquee', 
44980        'menu',    'multicol',   'ol',     'p',       'plaintext', 'pre', 
44981        'table',   'ul',         'xmp', 
44982        
44983        'caption', 'col', 'colgroup', 'tbody', 'td', 'tfoot', 'th', 
44984       'thead',   'tr', 
44985      
44986       'dir', 'menu', 'ol', 'ul', 'dl',
44987        
44988       'embed',  'object'
44989 ];
44990
44991
44992 Roo.HtmlEditorCore.black = [
44993     //    'embed',  'object', // enable - backend responsiblity to clean thiese
44994         'applet', // 
44995         'base',   'basefont', 'bgsound', 'blink',  'body', 
44996         'frame',  'frameset', 'head',    'html',   'ilayer', 
44997         'iframe', 'layer',  'link',     'meta',    'object',   
44998         'script', 'style' ,'title',  'xml' // clean later..
44999 ];
45000 Roo.HtmlEditorCore.clean = [
45001     'script', 'style', 'title', 'xml'
45002 ];
45003 Roo.HtmlEditorCore.remove = [
45004     'font'
45005 ];
45006 // attributes..
45007
45008 Roo.HtmlEditorCore.ablack = [
45009     'on'
45010 ];
45011     
45012 Roo.HtmlEditorCore.aclean = [ 
45013     'action', 'background', 'codebase', 'dynsrc', 'href', 'lowsrc' 
45014 ];
45015
45016 // protocols..
45017 Roo.HtmlEditorCore.pwhite= [
45018         'http',  'https',  'mailto'
45019 ];
45020
45021 // white listed style attributes.
45022 Roo.HtmlEditorCore.cwhite= [
45023       //  'text-align', /// default is to allow most things..
45024       
45025          
45026 //        'font-size'//??
45027 ];
45028
45029 // black listed style attributes.
45030 Roo.HtmlEditorCore.cblack= [
45031       //  'font-size' -- this can be set by the project 
45032 ];
45033
45034
45035 Roo.HtmlEditorCore.swapCodes   =[ 
45036     [    8211, "--" ], 
45037     [    8212, "--" ], 
45038     [    8216,  "'" ],  
45039     [    8217, "'" ],  
45040     [    8220, '"' ],  
45041     [    8221, '"' ],  
45042     [    8226, "*" ],  
45043     [    8230, "..." ]
45044 ]; 
45045
45046     //<script type="text/javascript">
45047
45048 /*
45049  * Ext JS Library 1.1.1
45050  * Copyright(c) 2006-2007, Ext JS, LLC.
45051  * Licence LGPL
45052  * 
45053  */
45054  
45055  
45056 Roo.form.HtmlEditor = function(config){
45057     
45058     
45059     
45060     Roo.form.HtmlEditor.superclass.constructor.call(this, config);
45061     
45062     if (!this.toolbars) {
45063         this.toolbars = [];
45064     }
45065     this.editorcore = new Roo.HtmlEditorCore(Roo.apply({ owner : this} , config));
45066     
45067     
45068 };
45069
45070 /**
45071  * @class Roo.form.HtmlEditor
45072  * @extends Roo.form.Field
45073  * Provides a lightweight HTML Editor component.
45074  *
45075  * This has been tested on Fireforx / Chrome.. IE may not be so great..
45076  * 
45077  * <br><br><b>Note: The focus/blur and validation marking functionality inherited from Ext.form.Field is NOT
45078  * supported by this editor.</b><br/><br/>
45079  * An Editor is a sensitive component that can't be used in all spots standard fields can be used. Putting an Editor within
45080  * any element that has display set to 'none' can cause problems in Safari and Firefox.<br/><br/>
45081  */
45082 Roo.extend(Roo.form.HtmlEditor, Roo.form.Field, {
45083     /**
45084      * @cfg {Boolean} clearUp
45085      */
45086     clearUp : true,
45087       /**
45088      * @cfg {Array} toolbars Array of toolbars. - defaults to just the Standard one
45089      */
45090     toolbars : false,
45091    
45092      /**
45093      * @cfg {String} resizable  's' or 'se' or 'e' - wrapps the element in a
45094      *                        Roo.resizable.
45095      */
45096     resizable : false,
45097      /**
45098      * @cfg {Number} height (in pixels)
45099      */   
45100     height: 300,
45101    /**
45102      * @cfg {Number} width (in pixels)
45103      */   
45104     width: 500,
45105     
45106     /**
45107      * @cfg {Array} stylesheets url of stylesheets. set to [] to disable stylesheets.
45108      * 
45109      */
45110     stylesheets: false,
45111     
45112     
45113      /**
45114      * @cfg {Array} blacklist of css styles style attributes (blacklist overrides whitelist)
45115      * 
45116      */
45117     cblack: false,
45118     /**
45119      * @cfg {Array} whitelist of css styles style attributes (blacklist overrides whitelist)
45120      * 
45121      */
45122     cwhite: false,
45123     
45124      /**
45125      * @cfg {Array} blacklist of html tags - in addition to standard blacklist.
45126      * 
45127      */
45128     black: false,
45129     /**
45130      * @cfg {Array} whitelist of html tags - in addition to statndard whitelist
45131      * 
45132      */
45133     white: false,
45134     
45135     // id of frame..
45136     frameId: false,
45137     
45138     // private properties
45139     validationEvent : false,
45140     deferHeight: true,
45141     initialized : false,
45142     activated : false,
45143     
45144     onFocus : Roo.emptyFn,
45145     iframePad:3,
45146     hideMode:'offsets',
45147     
45148     actionMode : 'container', // defaults to hiding it...
45149     
45150     defaultAutoCreate : { // modified by initCompnoent..
45151         tag: "textarea",
45152         style:"width:500px;height:300px;",
45153         autocomplete: "new-password"
45154     },
45155
45156     // private
45157     initComponent : function(){
45158         this.addEvents({
45159             /**
45160              * @event initialize
45161              * Fires when the editor is fully initialized (including the iframe)
45162              * @param {HtmlEditor} this
45163              */
45164             initialize: true,
45165             /**
45166              * @event activate
45167              * Fires when the editor is first receives the focus. Any insertion must wait
45168              * until after this event.
45169              * @param {HtmlEditor} this
45170              */
45171             activate: true,
45172              /**
45173              * @event beforesync
45174              * Fires before the textarea is updated with content from the editor iframe. Return false
45175              * to cancel the sync.
45176              * @param {HtmlEditor} this
45177              * @param {String} html
45178              */
45179             beforesync: true,
45180              /**
45181              * @event beforepush
45182              * Fires before the iframe editor is updated with content from the textarea. Return false
45183              * to cancel the push.
45184              * @param {HtmlEditor} this
45185              * @param {String} html
45186              */
45187             beforepush: true,
45188              /**
45189              * @event sync
45190              * Fires when the textarea is updated with content from the editor iframe.
45191              * @param {HtmlEditor} this
45192              * @param {String} html
45193              */
45194             sync: true,
45195              /**
45196              * @event push
45197              * Fires when the iframe editor is updated with content from the textarea.
45198              * @param {HtmlEditor} this
45199              * @param {String} html
45200              */
45201             push: true,
45202              /**
45203              * @event editmodechange
45204              * Fires when the editor switches edit modes
45205              * @param {HtmlEditor} this
45206              * @param {Boolean} sourceEdit True if source edit, false if standard editing.
45207              */
45208             editmodechange: true,
45209             /**
45210              * @event editorevent
45211              * Fires when on any editor (mouse up/down cursor movement etc.) - used for toolbar hooks.
45212              * @param {HtmlEditor} this
45213              */
45214             editorevent: true,
45215             /**
45216              * @event firstfocus
45217              * Fires when on first focus - needed by toolbars..
45218              * @param {HtmlEditor} this
45219              */
45220             firstfocus: true,
45221             /**
45222              * @event autosave
45223              * Auto save the htmlEditor value as a file into Events
45224              * @param {HtmlEditor} this
45225              */
45226             autosave: true,
45227             /**
45228              * @event savedpreview
45229              * preview the saved version of htmlEditor
45230              * @param {HtmlEditor} this
45231              */
45232             savedpreview: true,
45233             
45234             /**
45235             * @event stylesheetsclick
45236             * Fires when press the Sytlesheets button
45237             * @param {Roo.HtmlEditorCore} this
45238             */
45239             stylesheetsclick: true
45240         });
45241         this.defaultAutoCreate =  {
45242             tag: "textarea",
45243             style:'width: ' + this.width + 'px;height: ' + this.height + 'px;',
45244             autocomplete: "new-password"
45245         };
45246     },
45247
45248     /**
45249      * Protected method that will not generally be called directly. It
45250      * is called when the editor creates its toolbar. Override this method if you need to
45251      * add custom toolbar buttons.
45252      * @param {HtmlEditor} editor
45253      */
45254     createToolbar : function(editor){
45255         Roo.log("create toolbars");
45256         if (!editor.toolbars || !editor.toolbars.length) {
45257             editor.toolbars = [ new Roo.form.HtmlEditor.ToolbarStandard() ]; // can be empty?
45258         }
45259         
45260         for (var i =0 ; i < editor.toolbars.length;i++) {
45261             editor.toolbars[i] = Roo.factory(
45262                     typeof(editor.toolbars[i]) == 'string' ?
45263                         { xtype: editor.toolbars[i]} : editor.toolbars[i],
45264                 Roo.form.HtmlEditor);
45265             editor.toolbars[i].init(editor);
45266         }
45267          
45268         
45269     },
45270
45271      
45272     // private
45273     onRender : function(ct, position)
45274     {
45275         var _t = this;
45276         Roo.form.HtmlEditor.superclass.onRender.call(this, ct, position);
45277         
45278         this.wrap = this.el.wrap({
45279             cls:'x-html-editor-wrap', cn:{cls:'x-html-editor-tb'}
45280         });
45281         
45282         this.editorcore.onRender(ct, position);
45283          
45284         if (this.resizable) {
45285             this.resizeEl = new Roo.Resizable(this.wrap, {
45286                 pinned : true,
45287                 wrap: true,
45288                 dynamic : true,
45289                 minHeight : this.height,
45290                 height: this.height,
45291                 handles : this.resizable,
45292                 width: this.width,
45293                 listeners : {
45294                     resize : function(r, w, h) {
45295                         _t.onResize(w,h); // -something
45296                     }
45297                 }
45298             });
45299             
45300         }
45301         this.createToolbar(this);
45302        
45303         
45304         if(!this.width){
45305             this.setSize(this.wrap.getSize());
45306         }
45307         if (this.resizeEl) {
45308             this.resizeEl.resizeTo.defer(100, this.resizeEl,[ this.width,this.height ] );
45309             // should trigger onReize..
45310         }
45311         
45312         this.keyNav = new Roo.KeyNav(this.el, {
45313             
45314             "tab" : function(e){
45315                 e.preventDefault();
45316                 
45317                 var value = this.getValue();
45318                 
45319                 var start = this.el.dom.selectionStart;
45320                 var end = this.el.dom.selectionEnd;
45321                 
45322                 if(!e.shiftKey){
45323                     
45324                     this.setValue(value.substring(0, start) + "\t" + value.substring(end));
45325                     this.el.dom.setSelectionRange(end + 1, end + 1);
45326                     return;
45327                 }
45328                 
45329                 var f = value.substring(0, start).split("\t");
45330                 
45331                 if(f.pop().length != 0){
45332                     return;
45333                 }
45334                 
45335                 this.setValue(f.join("\t") + value.substring(end));
45336                 this.el.dom.setSelectionRange(start - 1, start - 1);
45337                 
45338             },
45339             
45340             "home" : function(e){
45341                 e.preventDefault();
45342                 
45343                 var curr = this.el.dom.selectionStart;
45344                 var lines = this.getValue().split("\n");
45345                 
45346                 if(!lines.length){
45347                     return;
45348                 }
45349                 
45350                 if(e.ctrlKey){
45351                     this.el.dom.setSelectionRange(0, 0);
45352                     return;
45353                 }
45354                 
45355                 var pos = 0;
45356                 
45357                 for (var i = 0; i < lines.length;i++) {
45358                     pos += lines[i].length;
45359                     
45360                     if(i != 0){
45361                         pos += 1;
45362                     }
45363                     
45364                     if(pos < curr){
45365                         continue;
45366                     }
45367                     
45368                     pos -= lines[i].length;
45369                     
45370                     break;
45371                 }
45372                 
45373                 if(!e.shiftKey){
45374                     this.el.dom.setSelectionRange(pos, pos);
45375                     return;
45376                 }
45377                 
45378                 this.el.dom.selectionStart = pos;
45379                 this.el.dom.selectionEnd = curr;
45380             },
45381             
45382             "end" : function(e){
45383                 e.preventDefault();
45384                 
45385                 var curr = this.el.dom.selectionStart;
45386                 var lines = this.getValue().split("\n");
45387                 
45388                 if(!lines.length){
45389                     return;
45390                 }
45391                 
45392                 if(e.ctrlKey){
45393                     this.el.dom.setSelectionRange(this.getValue().length, this.getValue().length);
45394                     return;
45395                 }
45396                 
45397                 var pos = 0;
45398                 
45399                 for (var i = 0; i < lines.length;i++) {
45400                     
45401                     pos += lines[i].length;
45402                     
45403                     if(i != 0){
45404                         pos += 1;
45405                     }
45406                     
45407                     if(pos < curr){
45408                         continue;
45409                     }
45410                     
45411                     break;
45412                 }
45413                 
45414                 if(!e.shiftKey){
45415                     this.el.dom.setSelectionRange(pos, pos);
45416                     return;
45417                 }
45418                 
45419                 this.el.dom.selectionStart = curr;
45420                 this.el.dom.selectionEnd = pos;
45421             },
45422
45423             scope : this,
45424
45425             doRelay : function(foo, bar, hname){
45426                 return Roo.KeyNav.prototype.doRelay.apply(this, arguments);
45427             },
45428
45429             forceKeyDown: true
45430         });
45431         
45432 //        if(this.autosave && this.w){
45433 //            this.autoSaveFn = setInterval(this.autosave, 1000);
45434 //        }
45435     },
45436
45437     // private
45438     onResize : function(w, h)
45439     {
45440         Roo.form.HtmlEditor.superclass.onResize.apply(this, arguments);
45441         var ew = false;
45442         var eh = false;
45443         
45444         if(this.el ){
45445             if(typeof w == 'number'){
45446                 var aw = w - this.wrap.getFrameWidth('lr');
45447                 this.el.setWidth(this.adjustWidth('textarea', aw));
45448                 ew = aw;
45449             }
45450             if(typeof h == 'number'){
45451                 var tbh = 0;
45452                 for (var i =0; i < this.toolbars.length;i++) {
45453                     // fixme - ask toolbars for heights?
45454                     tbh += this.toolbars[i].tb.el.getHeight();
45455                     if (this.toolbars[i].footer) {
45456                         tbh += this.toolbars[i].footer.el.getHeight();
45457                     }
45458                 }
45459                 
45460                 
45461                 
45462                 
45463                 var ah = h - this.wrap.getFrameWidth('tb') - tbh;// this.tb.el.getHeight();
45464                 ah -= 5; // knock a few pixes off for look..
45465 //                Roo.log(ah);
45466                 this.el.setHeight(this.adjustWidth('textarea', ah));
45467                 var eh = ah;
45468             }
45469         }
45470         Roo.log('onResize:' + [w,h,ew,eh].join(',') );
45471         this.editorcore.onResize(ew,eh);
45472         
45473     },
45474
45475     /**
45476      * Toggles the editor between standard and source edit mode.
45477      * @param {Boolean} sourceEdit (optional) True for source edit, false for standard
45478      */
45479     toggleSourceEdit : function(sourceEditMode)
45480     {
45481         this.editorcore.toggleSourceEdit(sourceEditMode);
45482         
45483         if(this.editorcore.sourceEditMode){
45484             Roo.log('editor - showing textarea');
45485             
45486 //            Roo.log('in');
45487 //            Roo.log(this.syncValue());
45488             this.editorcore.syncValue();
45489             this.el.removeClass('x-hidden');
45490             this.el.dom.removeAttribute('tabIndex');
45491             this.el.focus();
45492             
45493             for (var i = 0; i < this.toolbars.length; i++) {
45494                 if(this.toolbars[i] instanceof Roo.form.HtmlEditor.ToolbarContext){
45495                     this.toolbars[i].tb.hide();
45496                     this.toolbars[i].footer.hide();
45497                 }
45498             }
45499             
45500         }else{
45501             Roo.log('editor - hiding textarea');
45502 //            Roo.log('out')
45503 //            Roo.log(this.pushValue()); 
45504             this.editorcore.pushValue();
45505             
45506             this.el.addClass('x-hidden');
45507             this.el.dom.setAttribute('tabIndex', -1);
45508             
45509             for (var i = 0; i < this.toolbars.length; i++) {
45510                 if(this.toolbars[i] instanceof Roo.form.HtmlEditor.ToolbarContext){
45511                     this.toolbars[i].tb.show();
45512                     this.toolbars[i].footer.show();
45513                 }
45514             }
45515             
45516             //this.deferFocus();
45517         }
45518         
45519         this.setSize(this.wrap.getSize());
45520         this.onResize(this.wrap.getSize().width, this.wrap.getSize().height);
45521         
45522         this.fireEvent('editmodechange', this, this.editorcore.sourceEditMode);
45523     },
45524  
45525     // private (for BoxComponent)
45526     adjustSize : Roo.BoxComponent.prototype.adjustSize,
45527
45528     // private (for BoxComponent)
45529     getResizeEl : function(){
45530         return this.wrap;
45531     },
45532
45533     // private (for BoxComponent)
45534     getPositionEl : function(){
45535         return this.wrap;
45536     },
45537
45538     // private
45539     initEvents : function(){
45540         this.originalValue = this.getValue();
45541     },
45542
45543     /**
45544      * Overridden and disabled. The editor element does not support standard valid/invalid marking. @hide
45545      * @method
45546      */
45547     markInvalid : Roo.emptyFn,
45548     /**
45549      * Overridden and disabled. The editor element does not support standard valid/invalid marking. @hide
45550      * @method
45551      */
45552     clearInvalid : Roo.emptyFn,
45553
45554     setValue : function(v){
45555         Roo.form.HtmlEditor.superclass.setValue.call(this, v);
45556         this.editorcore.pushValue();
45557     },
45558
45559      
45560     // private
45561     deferFocus : function(){
45562         this.focus.defer(10, this);
45563     },
45564
45565     // doc'ed in Field
45566     focus : function(){
45567         this.editorcore.focus();
45568         
45569     },
45570       
45571
45572     // private
45573     onDestroy : function(){
45574         
45575         
45576         
45577         if(this.rendered){
45578             
45579             for (var i =0; i < this.toolbars.length;i++) {
45580                 // fixme - ask toolbars for heights?
45581                 this.toolbars[i].onDestroy();
45582             }
45583             
45584             this.wrap.dom.innerHTML = '';
45585             this.wrap.remove();
45586         }
45587     },
45588
45589     // private
45590     onFirstFocus : function(){
45591         //Roo.log("onFirstFocus");
45592         this.editorcore.onFirstFocus();
45593          for (var i =0; i < this.toolbars.length;i++) {
45594             this.toolbars[i].onFirstFocus();
45595         }
45596         
45597     },
45598     
45599     // private
45600     syncValue : function()
45601     {
45602         this.editorcore.syncValue();
45603     },
45604     
45605     pushValue : function()
45606     {
45607         this.editorcore.pushValue();
45608     },
45609     
45610     setStylesheets : function(stylesheets)
45611     {
45612         this.editorcore.setStylesheets(stylesheets);
45613     },
45614     
45615     removeStylesheets : function()
45616     {
45617         this.editorcore.removeStylesheets();
45618     }
45619      
45620     
45621     // hide stuff that is not compatible
45622     /**
45623      * @event blur
45624      * @hide
45625      */
45626     /**
45627      * @event change
45628      * @hide
45629      */
45630     /**
45631      * @event focus
45632      * @hide
45633      */
45634     /**
45635      * @event specialkey
45636      * @hide
45637      */
45638     /**
45639      * @cfg {String} fieldClass @hide
45640      */
45641     /**
45642      * @cfg {String} focusClass @hide
45643      */
45644     /**
45645      * @cfg {String} autoCreate @hide
45646      */
45647     /**
45648      * @cfg {String} inputType @hide
45649      */
45650     /**
45651      * @cfg {String} invalidClass @hide
45652      */
45653     /**
45654      * @cfg {String} invalidText @hide
45655      */
45656     /**
45657      * @cfg {String} msgFx @hide
45658      */
45659     /**
45660      * @cfg {String} validateOnBlur @hide
45661      */
45662 });
45663  
45664     // <script type="text/javascript">
45665 /*
45666  * Based on
45667  * Ext JS Library 1.1.1
45668  * Copyright(c) 2006-2007, Ext JS, LLC.
45669  *  
45670  
45671  */
45672
45673 /**
45674  * @class Roo.form.HtmlEditorToolbar1
45675  * Basic Toolbar
45676  * 
45677  * Usage:
45678  *
45679  new Roo.form.HtmlEditor({
45680     ....
45681     toolbars : [
45682         new Roo.form.HtmlEditorToolbar1({
45683             disable : { fonts: 1 , format: 1, ..., ... , ...],
45684             btns : [ .... ]
45685         })
45686     }
45687      
45688  * 
45689  * @cfg {Object} disable List of elements to disable..
45690  * @cfg {Array} btns List of additional buttons.
45691  * 
45692  * 
45693  * NEEDS Extra CSS? 
45694  * .x-html-editor-tb .x-edit-none .x-btn-text { background: none; }
45695  */
45696  
45697 Roo.form.HtmlEditor.ToolbarStandard = function(config)
45698 {
45699     
45700     Roo.apply(this, config);
45701     
45702     // default disabled, based on 'good practice'..
45703     this.disable = this.disable || {};
45704     Roo.applyIf(this.disable, {
45705         fontSize : true,
45706         colors : true,
45707         specialElements : true
45708     });
45709     
45710     
45711     //Roo.form.HtmlEditorToolbar1.superclass.constructor.call(this, editor.wrap.dom.firstChild, [], config);
45712     // dont call parent... till later.
45713 }
45714
45715 Roo.apply(Roo.form.HtmlEditor.ToolbarStandard.prototype,  {
45716     
45717     tb: false,
45718     
45719     rendered: false,
45720     
45721     editor : false,
45722     editorcore : false,
45723     /**
45724      * @cfg {Object} disable  List of toolbar elements to disable
45725          
45726      */
45727     disable : false,
45728     
45729     
45730      /**
45731      * @cfg {String} createLinkText The default text for the create link prompt
45732      */
45733     createLinkText : 'Please enter the URL for the link:',
45734     /**
45735      * @cfg {String} defaultLinkValue The default value for the create link prompt (defaults to http:/ /)
45736      */
45737     defaultLinkValue : 'http:/'+'/',
45738    
45739     
45740       /**
45741      * @cfg {Array} fontFamilies An array of available font families
45742      */
45743     fontFamilies : [
45744         'Arial',
45745         'Courier New',
45746         'Tahoma',
45747         'Times New Roman',
45748         'Verdana'
45749     ],
45750     
45751     specialChars : [
45752            "&#169;",
45753           "&#174;",     
45754           "&#8482;",    
45755           "&#163;" ,    
45756          // "&#8212;",    
45757           "&#8230;",    
45758           "&#247;" ,    
45759         //  "&#225;" ,     ?? a acute?
45760            "&#8364;"    , //Euro
45761        //   "&#8220;"    ,
45762         //  "&#8221;"    ,
45763         //  "&#8226;"    ,
45764           "&#176;"  //   , // degrees
45765
45766          // "&#233;"     , // e ecute
45767          // "&#250;"     , // u ecute?
45768     ],
45769     
45770     specialElements : [
45771         {
45772             text: "Insert Table",
45773             xtype: 'MenuItem',
45774             xns : Roo.Menu,
45775             ihtml :  '<table><tr><td>Cell</td></tr></table>' 
45776                 
45777         },
45778         {    
45779             text: "Insert Image",
45780             xtype: 'MenuItem',
45781             xns : Roo.Menu,
45782             ihtml : '<img src="about:blank"/>'
45783             
45784         }
45785         
45786          
45787     ],
45788     
45789     
45790     inputElements : [ 
45791             "form", "input:text", "input:hidden", "input:checkbox", "input:radio", "input:password", 
45792             "input:submit", "input:button", "select", "textarea", "label" ],
45793     formats : [
45794         ["p"] ,  
45795         ["h1"],["h2"],["h3"],["h4"],["h5"],["h6"], 
45796         ["pre"],[ "code"], 
45797         ["abbr"],[ "acronym"],[ "address"],[ "cite"],[ "samp"],[ "var"],
45798         ['div'],['span'],
45799         ['sup'],['sub']
45800     ],
45801     
45802     cleanStyles : [
45803         "font-size"
45804     ],
45805      /**
45806      * @cfg {String} defaultFont default font to use.
45807      */
45808     defaultFont: 'tahoma',
45809    
45810     fontSelect : false,
45811     
45812     
45813     formatCombo : false,
45814     
45815     init : function(editor)
45816     {
45817         this.editor = editor;
45818         this.editorcore = editor.editorcore ? editor.editorcore : editor;
45819         var editorcore = this.editorcore;
45820         
45821         var _t = this;
45822         
45823         var fid = editorcore.frameId;
45824         var etb = this;
45825         function btn(id, toggle, handler){
45826             var xid = fid + '-'+ id ;
45827             return {
45828                 id : xid,
45829                 cmd : id,
45830                 cls : 'x-btn-icon x-edit-'+id,
45831                 enableToggle:toggle !== false,
45832                 scope: _t, // was editor...
45833                 handler:handler||_t.relayBtnCmd,
45834                 clickEvent:'mousedown',
45835                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
45836                 tabIndex:-1
45837             };
45838         }
45839         
45840         
45841         
45842         var tb = new Roo.Toolbar(editor.wrap.dom.firstChild);
45843         this.tb = tb;
45844          // stop form submits
45845         tb.el.on('click', function(e){
45846             e.preventDefault(); // what does this do?
45847         });
45848
45849         if(!this.disable.font) { // && !Roo.isSafari){
45850             /* why no safari for fonts 
45851             editor.fontSelect = tb.el.createChild({
45852                 tag:'select',
45853                 tabIndex: -1,
45854                 cls:'x-font-select',
45855                 html: this.createFontOptions()
45856             });
45857             
45858             editor.fontSelect.on('change', function(){
45859                 var font = editor.fontSelect.dom.value;
45860                 editor.relayCmd('fontname', font);
45861                 editor.deferFocus();
45862             }, editor);
45863             
45864             tb.add(
45865                 editor.fontSelect.dom,
45866                 '-'
45867             );
45868             */
45869             
45870         };
45871         if(!this.disable.formats){
45872             this.formatCombo = new Roo.form.ComboBox({
45873                 store: new Roo.data.SimpleStore({
45874                     id : 'tag',
45875                     fields: ['tag'],
45876                     data : this.formats // from states.js
45877                 }),
45878                 blockFocus : true,
45879                 name : '',
45880                 //autoCreate : {tag: "div",  size: "20"},
45881                 displayField:'tag',
45882                 typeAhead: false,
45883                 mode: 'local',
45884                 editable : false,
45885                 triggerAction: 'all',
45886                 emptyText:'Add tag',
45887                 selectOnFocus:true,
45888                 width:135,
45889                 listeners : {
45890                     'select': function(c, r, i) {
45891                         editorcore.insertTag(r.get('tag'));
45892                         editor.focus();
45893                     }
45894                 }
45895
45896             });
45897             tb.addField(this.formatCombo);
45898             
45899         }
45900         
45901         if(!this.disable.format){
45902             tb.add(
45903                 btn('bold'),
45904                 btn('italic'),
45905                 btn('underline'),
45906                 btn('strikethrough')
45907             );
45908         };
45909         if(!this.disable.fontSize){
45910             tb.add(
45911                 '-',
45912                 
45913                 
45914                 btn('increasefontsize', false, editorcore.adjustFont),
45915                 btn('decreasefontsize', false, editorcore.adjustFont)
45916             );
45917         };
45918         
45919         
45920         if(!this.disable.colors){
45921             tb.add(
45922                 '-', {
45923                     id:editorcore.frameId +'-forecolor',
45924                     cls:'x-btn-icon x-edit-forecolor',
45925                     clickEvent:'mousedown',
45926                     tooltip: this.buttonTips['forecolor'] || undefined,
45927                     tabIndex:-1,
45928                     menu : new Roo.menu.ColorMenu({
45929                         allowReselect: true,
45930                         focus: Roo.emptyFn,
45931                         value:'000000',
45932                         plain:true,
45933                         selectHandler: function(cp, color){
45934                             editorcore.execCmd('forecolor', Roo.isSafari || Roo.isIE ? '#'+color : color);
45935                             editor.deferFocus();
45936                         },
45937                         scope: editorcore,
45938                         clickEvent:'mousedown'
45939                     })
45940                 }, {
45941                     id:editorcore.frameId +'backcolor',
45942                     cls:'x-btn-icon x-edit-backcolor',
45943                     clickEvent:'mousedown',
45944                     tooltip: this.buttonTips['backcolor'] || undefined,
45945                     tabIndex:-1,
45946                     menu : new Roo.menu.ColorMenu({
45947                         focus: Roo.emptyFn,
45948                         value:'FFFFFF',
45949                         plain:true,
45950                         allowReselect: true,
45951                         selectHandler: function(cp, color){
45952                             if(Roo.isGecko){
45953                                 editorcore.execCmd('useCSS', false);
45954                                 editorcore.execCmd('hilitecolor', color);
45955                                 editorcore.execCmd('useCSS', true);
45956                                 editor.deferFocus();
45957                             }else{
45958                                 editorcore.execCmd(Roo.isOpera ? 'hilitecolor' : 'backcolor', 
45959                                     Roo.isSafari || Roo.isIE ? '#'+color : color);
45960                                 editor.deferFocus();
45961                             }
45962                         },
45963                         scope:editorcore,
45964                         clickEvent:'mousedown'
45965                     })
45966                 }
45967             );
45968         };
45969         // now add all the items...
45970         
45971
45972         if(!this.disable.alignments){
45973             tb.add(
45974                 '-',
45975                 btn('justifyleft'),
45976                 btn('justifycenter'),
45977                 btn('justifyright')
45978             );
45979         };
45980
45981         //if(!Roo.isSafari){
45982             if(!this.disable.links){
45983                 tb.add(
45984                     '-',
45985                     btn('createlink', false, this.createLink)    /// MOVE TO HERE?!!?!?!?!
45986                 );
45987             };
45988
45989             if(!this.disable.lists){
45990                 tb.add(
45991                     '-',
45992                     btn('insertorderedlist'),
45993                     btn('insertunorderedlist')
45994                 );
45995             }
45996             if(!this.disable.sourceEdit){
45997                 tb.add(
45998                     '-',
45999                     btn('sourceedit', true, function(btn){
46000                         this.toggleSourceEdit(btn.pressed);
46001                     })
46002                 );
46003             }
46004         //}
46005         
46006         var smenu = { };
46007         // special menu.. - needs to be tidied up..
46008         if (!this.disable.special) {
46009             smenu = {
46010                 text: "&#169;",
46011                 cls: 'x-edit-none',
46012                 
46013                 menu : {
46014                     items : []
46015                 }
46016             };
46017             for (var i =0; i < this.specialChars.length; i++) {
46018                 smenu.menu.items.push({
46019                     
46020                     html: this.specialChars[i],
46021                     handler: function(a,b) {
46022                         editorcore.insertAtCursor(String.fromCharCode(a.html.replace('&#','').replace(';', '')));
46023                         //editor.insertAtCursor(a.html);
46024                         
46025                     },
46026                     tabIndex:-1
46027                 });
46028             }
46029             
46030             
46031             tb.add(smenu);
46032             
46033             
46034         }
46035         
46036         var cmenu = { };
46037         if (!this.disable.cleanStyles) {
46038             cmenu = {
46039                 cls: 'x-btn-icon x-btn-clear',
46040                 
46041                 menu : {
46042                     items : []
46043                 }
46044             };
46045             for (var i =0; i < this.cleanStyles.length; i++) {
46046                 cmenu.menu.items.push({
46047                     actiontype : this.cleanStyles[i],
46048                     html: 'Remove ' + this.cleanStyles[i],
46049                     handler: function(a,b) {
46050 //                        Roo.log(a);
46051 //                        Roo.log(b);
46052                         var c = Roo.get(editorcore.doc.body);
46053                         c.select('[style]').each(function(s) {
46054                             s.dom.style.removeProperty(a.actiontype);
46055                         });
46056                         editorcore.syncValue();
46057                     },
46058                     tabIndex:-1
46059                 });
46060             }
46061              cmenu.menu.items.push({
46062                 actiontype : 'tablewidths',
46063                 html: 'Remove Table Widths',
46064                 handler: function(a,b) {
46065                     editorcore.cleanTableWidths();
46066                     editorcore.syncValue();
46067                 },
46068                 tabIndex:-1
46069             });
46070             cmenu.menu.items.push({
46071                 actiontype : 'word',
46072                 html: 'Remove MS Word Formating',
46073                 handler: function(a,b) {
46074                     editorcore.cleanWord();
46075                     editorcore.syncValue();
46076                 },
46077                 tabIndex:-1
46078             });
46079             
46080             cmenu.menu.items.push({
46081                 actiontype : 'all',
46082                 html: 'Remove All Styles',
46083                 handler: function(a,b) {
46084                     
46085                     var c = Roo.get(editorcore.doc.body);
46086                     c.select('[style]').each(function(s) {
46087                         s.dom.removeAttribute('style');
46088                     });
46089                     editorcore.syncValue();
46090                 },
46091                 tabIndex:-1
46092             });
46093             
46094             cmenu.menu.items.push({
46095                 actiontype : 'all',
46096                 html: 'Remove All CSS Classes',
46097                 handler: function(a,b) {
46098                     
46099                     var c = Roo.get(editorcore.doc.body);
46100                     c.select('[class]').each(function(s) {
46101                         s.dom.removeAttribute('class');
46102                     });
46103                     editorcore.cleanWord();
46104                     editorcore.syncValue();
46105                 },
46106                 tabIndex:-1
46107             });
46108             
46109              cmenu.menu.items.push({
46110                 actiontype : 'tidy',
46111                 html: 'Tidy HTML Source',
46112                 handler: function(a,b) {
46113                     editorcore.doc.body.innerHTML = editorcore.domToHTML();
46114                     editorcore.syncValue();
46115                 },
46116                 tabIndex:-1
46117             });
46118             
46119             
46120             tb.add(cmenu);
46121         }
46122          
46123         if (!this.disable.specialElements) {
46124             var semenu = {
46125                 text: "Other;",
46126                 cls: 'x-edit-none',
46127                 menu : {
46128                     items : []
46129                 }
46130             };
46131             for (var i =0; i < this.specialElements.length; i++) {
46132                 semenu.menu.items.push(
46133                     Roo.apply({ 
46134                         handler: function(a,b) {
46135                             editor.insertAtCursor(this.ihtml);
46136                         }
46137                     }, this.specialElements[i])
46138                 );
46139                     
46140             }
46141             
46142             tb.add(semenu);
46143             
46144             
46145         }
46146          
46147         
46148         if (this.btns) {
46149             for(var i =0; i< this.btns.length;i++) {
46150                 var b = Roo.factory(this.btns[i],Roo.form);
46151                 b.cls =  'x-edit-none';
46152                 
46153                 if(typeof(this.btns[i].cls) != 'undefined' && this.btns[i].cls.indexOf('x-init-enable') !== -1){
46154                     b.cls += ' x-init-enable';
46155                 }
46156                 
46157                 b.scope = editorcore;
46158                 tb.add(b);
46159             }
46160         
46161         }
46162         
46163         
46164         
46165         // disable everything...
46166         
46167         this.tb.items.each(function(item){
46168             
46169            if(
46170                 item.id != editorcore.frameId+ '-sourceedit' && 
46171                 (typeof(item.cls) != 'undefined' && item.cls.indexOf('x-init-enable') === -1)
46172             ){
46173                 
46174                 item.disable();
46175             }
46176         });
46177         this.rendered = true;
46178         
46179         // the all the btns;
46180         editor.on('editorevent', this.updateToolbar, this);
46181         // other toolbars need to implement this..
46182         //editor.on('editmodechange', this.updateToolbar, this);
46183     },
46184     
46185     
46186     relayBtnCmd : function(btn) {
46187         this.editorcore.relayCmd(btn.cmd);
46188     },
46189     // private used internally
46190     createLink : function(){
46191         Roo.log("create link?");
46192         var url = prompt(this.createLinkText, this.defaultLinkValue);
46193         if(url && url != 'http:/'+'/'){
46194             this.editorcore.relayCmd('createlink', url);
46195         }
46196     },
46197
46198     
46199     /**
46200      * Protected method that will not generally be called directly. It triggers
46201      * a toolbar update by reading the markup state of the current selection in the editor.
46202      */
46203     updateToolbar: function(){
46204
46205         if(!this.editorcore.activated){
46206             this.editor.onFirstFocus();
46207             return;
46208         }
46209
46210         var btns = this.tb.items.map, 
46211             doc = this.editorcore.doc,
46212             frameId = this.editorcore.frameId;
46213
46214         if(!this.disable.font && !Roo.isSafari){
46215             /*
46216             var name = (doc.queryCommandValue('FontName')||this.editor.defaultFont).toLowerCase();
46217             if(name != this.fontSelect.dom.value){
46218                 this.fontSelect.dom.value = name;
46219             }
46220             */
46221         }
46222         if(!this.disable.format){
46223             btns[frameId + '-bold'].toggle(doc.queryCommandState('bold'));
46224             btns[frameId + '-italic'].toggle(doc.queryCommandState('italic'));
46225             btns[frameId + '-underline'].toggle(doc.queryCommandState('underline'));
46226             btns[frameId + '-strikethrough'].toggle(doc.queryCommandState('strikethrough'));
46227         }
46228         if(!this.disable.alignments){
46229             btns[frameId + '-justifyleft'].toggle(doc.queryCommandState('justifyleft'));
46230             btns[frameId + '-justifycenter'].toggle(doc.queryCommandState('justifycenter'));
46231             btns[frameId + '-justifyright'].toggle(doc.queryCommandState('justifyright'));
46232         }
46233         if(!Roo.isSafari && !this.disable.lists){
46234             btns[frameId + '-insertorderedlist'].toggle(doc.queryCommandState('insertorderedlist'));
46235             btns[frameId + '-insertunorderedlist'].toggle(doc.queryCommandState('insertunorderedlist'));
46236         }
46237         
46238         var ans = this.editorcore.getAllAncestors();
46239         if (this.formatCombo) {
46240             
46241             
46242             var store = this.formatCombo.store;
46243             this.formatCombo.setValue("");
46244             for (var i =0; i < ans.length;i++) {
46245                 if (ans[i] && store.query('tag',ans[i].tagName.toLowerCase(), false).length) {
46246                     // select it..
46247                     this.formatCombo.setValue(ans[i].tagName.toLowerCase());
46248                     break;
46249                 }
46250             }
46251         }
46252         
46253         
46254         
46255         // hides menus... - so this cant be on a menu...
46256         Roo.menu.MenuMgr.hideAll();
46257
46258         //this.editorsyncValue();
46259     },
46260    
46261     
46262     createFontOptions : function(){
46263         var buf = [], fs = this.fontFamilies, ff, lc;
46264         
46265         
46266         
46267         for(var i = 0, len = fs.length; i< len; i++){
46268             ff = fs[i];
46269             lc = ff.toLowerCase();
46270             buf.push(
46271                 '<option value="',lc,'" style="font-family:',ff,';"',
46272                     (this.defaultFont == lc ? ' selected="true">' : '>'),
46273                     ff,
46274                 '</option>'
46275             );
46276         }
46277         return buf.join('');
46278     },
46279     
46280     toggleSourceEdit : function(sourceEditMode){
46281         
46282         Roo.log("toolbar toogle");
46283         if(sourceEditMode === undefined){
46284             sourceEditMode = !this.sourceEditMode;
46285         }
46286         this.sourceEditMode = sourceEditMode === true;
46287         var btn = this.tb.items.get(this.editorcore.frameId +'-sourceedit');
46288         // just toggle the button?
46289         if(btn.pressed !== this.sourceEditMode){
46290             btn.toggle(this.sourceEditMode);
46291             return;
46292         }
46293         
46294         if(sourceEditMode){
46295             Roo.log("disabling buttons");
46296             this.tb.items.each(function(item){
46297                 if(item.cmd != 'sourceedit' && (typeof(item.cls) != 'undefined' && item.cls.indexOf('x-init-enable') === -1)){
46298                     item.disable();
46299                 }
46300             });
46301           
46302         }else{
46303             Roo.log("enabling buttons");
46304             if(this.editorcore.initialized){
46305                 this.tb.items.each(function(item){
46306                     item.enable();
46307                 });
46308             }
46309             
46310         }
46311         Roo.log("calling toggole on editor");
46312         // tell the editor that it's been pressed..
46313         this.editor.toggleSourceEdit(sourceEditMode);
46314        
46315     },
46316      /**
46317      * Object collection of toolbar tooltips for the buttons in the editor. The key
46318      * is the command id associated with that button and the value is a valid QuickTips object.
46319      * For example:
46320 <pre><code>
46321 {
46322     bold : {
46323         title: 'Bold (Ctrl+B)',
46324         text: 'Make the selected text bold.',
46325         cls: 'x-html-editor-tip'
46326     },
46327     italic : {
46328         title: 'Italic (Ctrl+I)',
46329         text: 'Make the selected text italic.',
46330         cls: 'x-html-editor-tip'
46331     },
46332     ...
46333 </code></pre>
46334     * @type Object
46335      */
46336     buttonTips : {
46337         bold : {
46338             title: 'Bold (Ctrl+B)',
46339             text: 'Make the selected text bold.',
46340             cls: 'x-html-editor-tip'
46341         },
46342         italic : {
46343             title: 'Italic (Ctrl+I)',
46344             text: 'Make the selected text italic.',
46345             cls: 'x-html-editor-tip'
46346         },
46347         underline : {
46348             title: 'Underline (Ctrl+U)',
46349             text: 'Underline the selected text.',
46350             cls: 'x-html-editor-tip'
46351         },
46352         strikethrough : {
46353             title: 'Strikethrough',
46354             text: 'Strikethrough the selected text.',
46355             cls: 'x-html-editor-tip'
46356         },
46357         increasefontsize : {
46358             title: 'Grow Text',
46359             text: 'Increase the font size.',
46360             cls: 'x-html-editor-tip'
46361         },
46362         decreasefontsize : {
46363             title: 'Shrink Text',
46364             text: 'Decrease the font size.',
46365             cls: 'x-html-editor-tip'
46366         },
46367         backcolor : {
46368             title: 'Text Highlight Color',
46369             text: 'Change the background color of the selected text.',
46370             cls: 'x-html-editor-tip'
46371         },
46372         forecolor : {
46373             title: 'Font Color',
46374             text: 'Change the color of the selected text.',
46375             cls: 'x-html-editor-tip'
46376         },
46377         justifyleft : {
46378             title: 'Align Text Left',
46379             text: 'Align text to the left.',
46380             cls: 'x-html-editor-tip'
46381         },
46382         justifycenter : {
46383             title: 'Center Text',
46384             text: 'Center text in the editor.',
46385             cls: 'x-html-editor-tip'
46386         },
46387         justifyright : {
46388             title: 'Align Text Right',
46389             text: 'Align text to the right.',
46390             cls: 'x-html-editor-tip'
46391         },
46392         insertunorderedlist : {
46393             title: 'Bullet List',
46394             text: 'Start a bulleted list.',
46395             cls: 'x-html-editor-tip'
46396         },
46397         insertorderedlist : {
46398             title: 'Numbered List',
46399             text: 'Start a numbered list.',
46400             cls: 'x-html-editor-tip'
46401         },
46402         createlink : {
46403             title: 'Hyperlink',
46404             text: 'Make the selected text a hyperlink.',
46405             cls: 'x-html-editor-tip'
46406         },
46407         sourceedit : {
46408             title: 'Source Edit',
46409             text: 'Switch to source editing mode.',
46410             cls: 'x-html-editor-tip'
46411         }
46412     },
46413     // private
46414     onDestroy : function(){
46415         if(this.rendered){
46416             
46417             this.tb.items.each(function(item){
46418                 if(item.menu){
46419                     item.menu.removeAll();
46420                     if(item.menu.el){
46421                         item.menu.el.destroy();
46422                     }
46423                 }
46424                 item.destroy();
46425             });
46426              
46427         }
46428     },
46429     onFirstFocus: function() {
46430         this.tb.items.each(function(item){
46431            item.enable();
46432         });
46433     }
46434 });
46435
46436
46437
46438
46439 // <script type="text/javascript">
46440 /*
46441  * Based on
46442  * Ext JS Library 1.1.1
46443  * Copyright(c) 2006-2007, Ext JS, LLC.
46444  *  
46445  
46446  */
46447
46448  
46449 /**
46450  * @class Roo.form.HtmlEditor.ToolbarContext
46451  * Context Toolbar
46452  * 
46453  * Usage:
46454  *
46455  new Roo.form.HtmlEditor({
46456     ....
46457     toolbars : [
46458         { xtype: 'ToolbarStandard', styles : {} }
46459         { xtype: 'ToolbarContext', disable : {} }
46460     ]
46461 })
46462
46463      
46464  * 
46465  * @config : {Object} disable List of elements to disable.. (not done yet.)
46466  * @config : {Object} styles  Map of styles available.
46467  * 
46468  */
46469
46470 Roo.form.HtmlEditor.ToolbarContext = function(config)
46471 {
46472     
46473     Roo.apply(this, config);
46474     //Roo.form.HtmlEditorToolbar1.superclass.constructor.call(this, editor.wrap.dom.firstChild, [], config);
46475     // dont call parent... till later.
46476     this.styles = this.styles || {};
46477 }
46478
46479  
46480
46481 Roo.form.HtmlEditor.ToolbarContext.types = {
46482     'IMG' : {
46483         width : {
46484             title: "Width",
46485             width: 40
46486         },
46487         height:  {
46488             title: "Height",
46489             width: 40
46490         },
46491         align: {
46492             title: "Align",
46493             opts : [ [""],[ "left"],[ "right"],[ "center"],[ "top"]],
46494             width : 80
46495             
46496         },
46497         border: {
46498             title: "Border",
46499             width: 40
46500         },
46501         alt: {
46502             title: "Alt",
46503             width: 120
46504         },
46505         src : {
46506             title: "Src",
46507             width: 220
46508         }
46509         
46510     },
46511     'A' : {
46512         name : {
46513             title: "Name",
46514             width: 50
46515         },
46516         target:  {
46517             title: "Target",
46518             width: 120
46519         },
46520         href:  {
46521             title: "Href",
46522             width: 220
46523         } // border?
46524         
46525     },
46526     'TABLE' : {
46527         rows : {
46528             title: "Rows",
46529             width: 20
46530         },
46531         cols : {
46532             title: "Cols",
46533             width: 20
46534         },
46535         width : {
46536             title: "Width",
46537             width: 40
46538         },
46539         height : {
46540             title: "Height",
46541             width: 40
46542         },
46543         border : {
46544             title: "Border",
46545             width: 20
46546         }
46547     },
46548     'TD' : {
46549         width : {
46550             title: "Width",
46551             width: 40
46552         },
46553         height : {
46554             title: "Height",
46555             width: 40
46556         },   
46557         align: {
46558             title: "Align",
46559             opts : [[""],[ "left"],[ "center"],[ "right"],[ "justify"],[ "char"]],
46560             width: 80
46561         },
46562         valign: {
46563             title: "Valign",
46564             opts : [[""],[ "top"],[ "middle"],[ "bottom"],[ "baseline"]],
46565             width: 80
46566         },
46567         colspan: {
46568             title: "Colspan",
46569             width: 20
46570             
46571         },
46572          'font-family'  : {
46573             title : "Font",
46574             style : 'fontFamily',
46575             displayField: 'display',
46576             optname : 'font-family',
46577             width: 140
46578         }
46579     },
46580     'INPUT' : {
46581         name : {
46582             title: "name",
46583             width: 120
46584         },
46585         value : {
46586             title: "Value",
46587             width: 120
46588         },
46589         width : {
46590             title: "Width",
46591             width: 40
46592         }
46593     },
46594     'LABEL' : {
46595         'for' : {
46596             title: "For",
46597             width: 120
46598         }
46599     },
46600     'TEXTAREA' : {
46601           name : {
46602             title: "name",
46603             width: 120
46604         },
46605         rows : {
46606             title: "Rows",
46607             width: 20
46608         },
46609         cols : {
46610             title: "Cols",
46611             width: 20
46612         }
46613     },
46614     'SELECT' : {
46615         name : {
46616             title: "name",
46617             width: 120
46618         },
46619         selectoptions : {
46620             title: "Options",
46621             width: 200
46622         }
46623     },
46624     
46625     // should we really allow this??
46626     // should this just be 
46627     'BODY' : {
46628         title : {
46629             title: "Title",
46630             width: 200,
46631             disabled : true
46632         }
46633     },
46634     'SPAN' : {
46635         'font-family'  : {
46636             title : "Font",
46637             style : 'fontFamily',
46638             displayField: 'display',
46639             optname : 'font-family',
46640             width: 140
46641         }
46642     },
46643     'DIV' : {
46644         'font-family'  : {
46645             title : "Font",
46646             style : 'fontFamily',
46647             displayField: 'display',
46648             optname : 'font-family',
46649             width: 140
46650         }
46651     },
46652      'P' : {
46653         'font-family'  : {
46654             title : "Font",
46655             style : 'fontFamily',
46656             displayField: 'display',
46657             optname : 'font-family',
46658             width: 140
46659         }
46660     },
46661     
46662     '*' : {
46663         // empty..
46664     }
46665
46666 };
46667
46668 // this should be configurable.. - you can either set it up using stores, or modify options somehwere..
46669 Roo.form.HtmlEditor.ToolbarContext.stores = false;
46670
46671 Roo.form.HtmlEditor.ToolbarContext.options = {
46672         'font-family'  : [ 
46673                 [ 'Helvetica,Arial,sans-serif', 'Helvetica'],
46674                 [ 'Courier New', 'Courier New'],
46675                 [ 'Tahoma', 'Tahoma'],
46676                 [ 'Times New Roman,serif', 'Times'],
46677                 [ 'Verdana','Verdana' ]
46678         ]
46679 };
46680
46681 // fixme - these need to be configurable..
46682  
46683
46684 //Roo.form.HtmlEditor.ToolbarContext.types
46685
46686
46687 Roo.apply(Roo.form.HtmlEditor.ToolbarContext.prototype,  {
46688     
46689     tb: false,
46690     
46691     rendered: false,
46692     
46693     editor : false,
46694     editorcore : false,
46695     /**
46696      * @cfg {Object} disable  List of toolbar elements to disable
46697          
46698      */
46699     disable : false,
46700     /**
46701      * @cfg {Object} styles List of styles 
46702      *    eg. { '*' : [ 'headline' ] , 'TD' : [ 'underline', 'double-underline' ] } 
46703      *
46704      * These must be defined in the page, so they get rendered correctly..
46705      * .headline { }
46706      * TD.underline { }
46707      * 
46708      */
46709     styles : false,
46710     
46711     options: false,
46712     
46713     toolbars : false,
46714     
46715     init : function(editor)
46716     {
46717         this.editor = editor;
46718         this.editorcore = editor.editorcore ? editor.editorcore : editor;
46719         var editorcore = this.editorcore;
46720         
46721         var fid = editorcore.frameId;
46722         var etb = this;
46723         function btn(id, toggle, handler){
46724             var xid = fid + '-'+ id ;
46725             return {
46726                 id : xid,
46727                 cmd : id,
46728                 cls : 'x-btn-icon x-edit-'+id,
46729                 enableToggle:toggle !== false,
46730                 scope: editorcore, // was editor...
46731                 handler:handler||editorcore.relayBtnCmd,
46732                 clickEvent:'mousedown',
46733                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
46734                 tabIndex:-1
46735             };
46736         }
46737         // create a new element.
46738         var wdiv = editor.wrap.createChild({
46739                 tag: 'div'
46740             }, editor.wrap.dom.firstChild.nextSibling, true);
46741         
46742         // can we do this more than once??
46743         
46744          // stop form submits
46745       
46746  
46747         // disable everything...
46748         var ty= Roo.form.HtmlEditor.ToolbarContext.types;
46749         this.toolbars = {};
46750            
46751         for (var i in  ty) {
46752           
46753             this.toolbars[i] = this.buildToolbar(ty[i],i);
46754         }
46755         this.tb = this.toolbars.BODY;
46756         this.tb.el.show();
46757         this.buildFooter();
46758         this.footer.show();
46759         editor.on('hide', function( ) { this.footer.hide() }, this);
46760         editor.on('show', function( ) { this.footer.show() }, this);
46761         
46762          
46763         this.rendered = true;
46764         
46765         // the all the btns;
46766         editor.on('editorevent', this.updateToolbar, this);
46767         // other toolbars need to implement this..
46768         //editor.on('editmodechange', this.updateToolbar, this);
46769     },
46770     
46771     
46772     
46773     /**
46774      * Protected method that will not generally be called directly. It triggers
46775      * a toolbar update by reading the markup state of the current selection in the editor.
46776      *
46777      * Note you can force an update by calling on('editorevent', scope, false)
46778      */
46779     updateToolbar: function(editor,ev,sel){
46780
46781         //Roo.log(ev);
46782         // capture mouse up - this is handy for selecting images..
46783         // perhaps should go somewhere else...
46784         if(!this.editorcore.activated){
46785              this.editor.onFirstFocus();
46786             return;
46787         }
46788         
46789         
46790         
46791         // http://developer.yahoo.com/yui/docs/simple-editor.js.html
46792         // selectNode - might want to handle IE?
46793         if (ev &&
46794             (ev.type == 'mouseup' || ev.type == 'click' ) &&
46795             ev.target && ev.target.tagName == 'IMG') {
46796             // they have click on an image...
46797             // let's see if we can change the selection...
46798             sel = ev.target;
46799          
46800               var nodeRange = sel.ownerDocument.createRange();
46801             try {
46802                 nodeRange.selectNode(sel);
46803             } catch (e) {
46804                 nodeRange.selectNodeContents(sel);
46805             }
46806             //nodeRange.collapse(true);
46807             var s = this.editorcore.win.getSelection();
46808             s.removeAllRanges();
46809             s.addRange(nodeRange);
46810         }  
46811         
46812       
46813         var updateFooter = sel ? false : true;
46814         
46815         
46816         var ans = this.editorcore.getAllAncestors();
46817         
46818         // pick
46819         var ty= Roo.form.HtmlEditor.ToolbarContext.types;
46820         
46821         if (!sel) { 
46822             sel = ans.length ? (ans[0] ?  ans[0]  : ans[1]) : this.editorcore.doc.body;
46823             sel = sel ? sel : this.editorcore.doc.body;
46824             sel = sel.tagName.length ? sel : this.editorcore.doc.body;
46825             
46826         }
46827         // pick a menu that exists..
46828         var tn = sel.tagName.toUpperCase();
46829         //sel = typeof(ty[tn]) != 'undefined' ? sel : this.editor.doc.body;
46830         
46831         tn = sel.tagName.toUpperCase();
46832         
46833         var lastSel = this.tb.selectedNode;
46834         
46835         this.tb.selectedNode = sel;
46836         
46837         // if current menu does not match..
46838         
46839         if ((this.tb.name != tn) || (lastSel != this.tb.selectedNode) || ev === false) {
46840                 
46841             this.tb.el.hide();
46842             ///console.log("show: " + tn);
46843             this.tb =  typeof(ty[tn]) != 'undefined' ? this.toolbars[tn] : this.toolbars['*'];
46844             this.tb.el.show();
46845             // update name
46846             this.tb.items.first().el.innerHTML = tn + ':&nbsp;';
46847             
46848             
46849             // update attributes
46850             if (this.tb.fields) {
46851                 this.tb.fields.each(function(e) {
46852                     if (e.stylename) {
46853                         e.setValue(sel.style[e.stylename]);
46854                         return;
46855                     } 
46856                    e.setValue(sel.getAttribute(e.attrname));
46857                 });
46858             }
46859             
46860             var hasStyles = false;
46861             for(var i in this.styles) {
46862                 hasStyles = true;
46863                 break;
46864             }
46865             
46866             // update styles
46867             if (hasStyles) { 
46868                 var st = this.tb.fields.item(0);
46869                 
46870                 st.store.removeAll();
46871                
46872                 
46873                 var cn = sel.className.split(/\s+/);
46874                 
46875                 var avs = [];
46876                 if (this.styles['*']) {
46877                     
46878                     Roo.each(this.styles['*'], function(v) {
46879                         avs.push( [ v , cn.indexOf(v) > -1 ? 1 : 0 ] );         
46880                     });
46881                 }
46882                 if (this.styles[tn]) { 
46883                     Roo.each(this.styles[tn], function(v) {
46884                         avs.push( [ v , cn.indexOf(v) > -1 ? 1 : 0 ] );         
46885                     });
46886                 }
46887                 
46888                 st.store.loadData(avs);
46889                 st.collapse();
46890                 st.setValue(cn);
46891             }
46892             // flag our selected Node.
46893             this.tb.selectedNode = sel;
46894            
46895            
46896             Roo.menu.MenuMgr.hideAll();
46897
46898         }
46899         
46900         if (!updateFooter) {
46901             //this.footDisp.dom.innerHTML = ''; 
46902             return;
46903         }
46904         // update the footer
46905         //
46906         var html = '';
46907         
46908         this.footerEls = ans.reverse();
46909         Roo.each(this.footerEls, function(a,i) {
46910             if (!a) { return; }
46911             html += html.length ? ' &gt; '  :  '';
46912             
46913             html += '<span class="x-ed-loc-' + i + '">' + a.tagName + '</span>';
46914             
46915         });
46916        
46917         // 
46918         var sz = this.footDisp.up('td').getSize();
46919         this.footDisp.dom.style.width = (sz.width -10) + 'px';
46920         this.footDisp.dom.style.marginLeft = '5px';
46921         
46922         this.footDisp.dom.style.overflow = 'hidden';
46923         
46924         this.footDisp.dom.innerHTML = html;
46925             
46926         //this.editorsyncValue();
46927     },
46928      
46929     
46930    
46931        
46932     // private
46933     onDestroy : function(){
46934         if(this.rendered){
46935             
46936             this.tb.items.each(function(item){
46937                 if(item.menu){
46938                     item.menu.removeAll();
46939                     if(item.menu.el){
46940                         item.menu.el.destroy();
46941                     }
46942                 }
46943                 item.destroy();
46944             });
46945              
46946         }
46947     },
46948     onFirstFocus: function() {
46949         // need to do this for all the toolbars..
46950         this.tb.items.each(function(item){
46951            item.enable();
46952         });
46953     },
46954     buildToolbar: function(tlist, nm)
46955     {
46956         var editor = this.editor;
46957         var editorcore = this.editorcore;
46958          // create a new element.
46959         var wdiv = editor.wrap.createChild({
46960                 tag: 'div'
46961             }, editor.wrap.dom.firstChild.nextSibling, true);
46962         
46963        
46964         var tb = new Roo.Toolbar(wdiv);
46965         // add the name..
46966         
46967         tb.add(nm+ ":&nbsp;");
46968         
46969         var styles = [];
46970         for(var i in this.styles) {
46971             styles.push(i);
46972         }
46973         
46974         // styles...
46975         if (styles && styles.length) {
46976             
46977             // this needs a multi-select checkbox...
46978             tb.addField( new Roo.form.ComboBox({
46979                 store: new Roo.data.SimpleStore({
46980                     id : 'val',
46981                     fields: ['val', 'selected'],
46982                     data : [] 
46983                 }),
46984                 name : '-roo-edit-className',
46985                 attrname : 'className',
46986                 displayField: 'val',
46987                 typeAhead: false,
46988                 mode: 'local',
46989                 editable : false,
46990                 triggerAction: 'all',
46991                 emptyText:'Select Style',
46992                 selectOnFocus:true,
46993                 width: 130,
46994                 listeners : {
46995                     'select': function(c, r, i) {
46996                         // initial support only for on class per el..
46997                         tb.selectedNode.className =  r ? r.get('val') : '';
46998                         editorcore.syncValue();
46999                     }
47000                 }
47001     
47002             }));
47003         }
47004         
47005         var tbc = Roo.form.HtmlEditor.ToolbarContext;
47006         var tbops = tbc.options;
47007         
47008         for (var i in tlist) {
47009             
47010             var item = tlist[i];
47011             tb.add(item.title + ":&nbsp;");
47012             
47013             
47014             //optname == used so you can configure the options available..
47015             var opts = item.opts ? item.opts : false;
47016             if (item.optname) {
47017                 opts = tbops[item.optname];
47018            
47019             }
47020             
47021             if (opts) {
47022                 // opts == pulldown..
47023                 tb.addField( new Roo.form.ComboBox({
47024                     store:   typeof(tbc.stores[i]) != 'undefined' ?  Roo.factory(tbc.stores[i],Roo.data) : new Roo.data.SimpleStore({
47025                         id : 'val',
47026                         fields: ['val', 'display'],
47027                         data : opts  
47028                     }),
47029                     name : '-roo-edit-' + i,
47030                     attrname : i,
47031                     stylename : item.style ? item.style : false,
47032                     displayField: item.displayField ? item.displayField : 'val',
47033                     valueField :  'val',
47034                     typeAhead: false,
47035                     mode: typeof(tbc.stores[i]) != 'undefined'  ? 'remote' : 'local',
47036                     editable : false,
47037                     triggerAction: 'all',
47038                     emptyText:'Select',
47039                     selectOnFocus:true,
47040                     width: item.width ? item.width  : 130,
47041                     listeners : {
47042                         'select': function(c, r, i) {
47043                             if (c.stylename) {
47044                                 tb.selectedNode.style[c.stylename] =  r.get('val');
47045                                 return;
47046                             }
47047                             tb.selectedNode.setAttribute(c.attrname, r.get('val'));
47048                         }
47049                     }
47050
47051                 }));
47052                 continue;
47053                     
47054                  
47055                 
47056                 tb.addField( new Roo.form.TextField({
47057                     name: i,
47058                     width: 100,
47059                     //allowBlank:false,
47060                     value: ''
47061                 }));
47062                 continue;
47063             }
47064             tb.addField( new Roo.form.TextField({
47065                 name: '-roo-edit-' + i,
47066                 attrname : i,
47067                 
47068                 width: item.width,
47069                 //allowBlank:true,
47070                 value: '',
47071                 listeners: {
47072                     'change' : function(f, nv, ov) {
47073                         tb.selectedNode.setAttribute(f.attrname, nv);
47074                         editorcore.syncValue();
47075                     }
47076                 }
47077             }));
47078              
47079         }
47080         
47081         var _this = this;
47082         
47083         if(nm == 'BODY'){
47084             tb.addSeparator();
47085         
47086             tb.addButton( {
47087                 text: 'Stylesheets',
47088
47089                 listeners : {
47090                     click : function ()
47091                     {
47092                         _this.editor.fireEvent('stylesheetsclick', _this.editor);
47093                     }
47094                 }
47095             });
47096         }
47097         
47098         tb.addFill();
47099         tb.addButton( {
47100             text: 'Remove Tag',
47101     
47102             listeners : {
47103                 click : function ()
47104                 {
47105                     // remove
47106                     // undo does not work.
47107                      
47108                     var sn = tb.selectedNode;
47109                     
47110                     var pn = sn.parentNode;
47111                     
47112                     var stn =  sn.childNodes[0];
47113                     var en = sn.childNodes[sn.childNodes.length - 1 ];
47114                     while (sn.childNodes.length) {
47115                         var node = sn.childNodes[0];
47116                         sn.removeChild(node);
47117                         //Roo.log(node);
47118                         pn.insertBefore(node, sn);
47119                         
47120                     }
47121                     pn.removeChild(sn);
47122                     var range = editorcore.createRange();
47123         
47124                     range.setStart(stn,0);
47125                     range.setEnd(en,0); //????
47126                     //range.selectNode(sel);
47127                     
47128                     
47129                     var selection = editorcore.getSelection();
47130                     selection.removeAllRanges();
47131                     selection.addRange(range);
47132                     
47133                     
47134                     
47135                     //_this.updateToolbar(null, null, pn);
47136                     _this.updateToolbar(null, null, null);
47137                     _this.footDisp.dom.innerHTML = ''; 
47138                 }
47139             }
47140             
47141                     
47142                 
47143             
47144         });
47145         
47146         
47147         tb.el.on('click', function(e){
47148             e.preventDefault(); // what does this do?
47149         });
47150         tb.el.setVisibilityMode( Roo.Element.DISPLAY);
47151         tb.el.hide();
47152         tb.name = nm;
47153         // dont need to disable them... as they will get hidden
47154         return tb;
47155          
47156         
47157     },
47158     buildFooter : function()
47159     {
47160         
47161         var fel = this.editor.wrap.createChild();
47162         this.footer = new Roo.Toolbar(fel);
47163         // toolbar has scrolly on left / right?
47164         var footDisp= new Roo.Toolbar.Fill();
47165         var _t = this;
47166         this.footer.add(
47167             {
47168                 text : '&lt;',
47169                 xtype: 'Button',
47170                 handler : function() {
47171                     _t.footDisp.scrollTo('left',0,true)
47172                 }
47173             }
47174         );
47175         this.footer.add( footDisp );
47176         this.footer.add( 
47177             {
47178                 text : '&gt;',
47179                 xtype: 'Button',
47180                 handler : function() {
47181                     // no animation..
47182                     _t.footDisp.select('span').last().scrollIntoView(_t.footDisp,true);
47183                 }
47184             }
47185         );
47186         var fel = Roo.get(footDisp.el);
47187         fel.addClass('x-editor-context');
47188         this.footDispWrap = fel; 
47189         this.footDispWrap.overflow  = 'hidden';
47190         
47191         this.footDisp = fel.createChild();
47192         this.footDispWrap.on('click', this.onContextClick, this)
47193         
47194         
47195     },
47196     onContextClick : function (ev,dom)
47197     {
47198         ev.preventDefault();
47199         var  cn = dom.className;
47200         //Roo.log(cn);
47201         if (!cn.match(/x-ed-loc-/)) {
47202             return;
47203         }
47204         var n = cn.split('-').pop();
47205         var ans = this.footerEls;
47206         var sel = ans[n];
47207         
47208          // pick
47209         var range = this.editorcore.createRange();
47210         
47211         range.selectNodeContents(sel);
47212         //range.selectNode(sel);
47213         
47214         
47215         var selection = this.editorcore.getSelection();
47216         selection.removeAllRanges();
47217         selection.addRange(range);
47218         
47219         
47220         
47221         this.updateToolbar(null, null, sel);
47222         
47223         
47224     }
47225     
47226     
47227     
47228     
47229     
47230 });
47231
47232
47233
47234
47235
47236 /*
47237  * Based on:
47238  * Ext JS Library 1.1.1
47239  * Copyright(c) 2006-2007, Ext JS, LLC.
47240  *
47241  * Originally Released Under LGPL - original licence link has changed is not relivant.
47242  *
47243  * Fork - LGPL
47244  * <script type="text/javascript">
47245  */
47246  
47247 /**
47248  * @class Roo.form.BasicForm
47249  * @extends Roo.util.Observable
47250  * Supplies the functionality to do "actions" on forms and initialize Roo.form.Field types on existing markup.
47251  * @constructor
47252  * @param {String/HTMLElement/Roo.Element} el The form element or its id
47253  * @param {Object} config Configuration options
47254  */
47255 Roo.form.BasicForm = function(el, config){
47256     this.allItems = [];
47257     this.childForms = [];
47258     Roo.apply(this, config);
47259     /*
47260      * The Roo.form.Field items in this form.
47261      * @type MixedCollection
47262      */
47263      
47264      
47265     this.items = new Roo.util.MixedCollection(false, function(o){
47266         return o.id || (o.id = Roo.id());
47267     });
47268     this.addEvents({
47269         /**
47270          * @event beforeaction
47271          * Fires before any action is performed. Return false to cancel the action.
47272          * @param {Form} this
47273          * @param {Action} action The action to be performed
47274          */
47275         beforeaction: true,
47276         /**
47277          * @event actionfailed
47278          * Fires when an action fails.
47279          * @param {Form} this
47280          * @param {Action} action The action that failed
47281          */
47282         actionfailed : true,
47283         /**
47284          * @event actioncomplete
47285          * Fires when an action is completed.
47286          * @param {Form} this
47287          * @param {Action} action The action that completed
47288          */
47289         actioncomplete : true
47290     });
47291     if(el){
47292         this.initEl(el);
47293     }
47294     Roo.form.BasicForm.superclass.constructor.call(this);
47295     
47296     Roo.form.BasicForm.popover.apply();
47297 };
47298
47299 Roo.extend(Roo.form.BasicForm, Roo.util.Observable, {
47300     /**
47301      * @cfg {String} method
47302      * The request method to use (GET or POST) for form actions if one isn't supplied in the action options.
47303      */
47304     /**
47305      * @cfg {DataReader} reader
47306      * An Roo.data.DataReader (e.g. {@link Roo.data.XmlReader}) to be used to read data when executing "load" actions.
47307      * This is optional as there is built-in support for processing JSON.
47308      */
47309     /**
47310      * @cfg {DataReader} errorReader
47311      * An Roo.data.DataReader (e.g. {@link Roo.data.XmlReader}) to be used to read data when reading validation errors on "submit" actions.
47312      * This is completely optional as there is built-in support for processing JSON.
47313      */
47314     /**
47315      * @cfg {String} url
47316      * The URL to use for form actions if one isn't supplied in the action options.
47317      */
47318     /**
47319      * @cfg {Boolean} fileUpload
47320      * Set to true if this form is a file upload.
47321      */
47322      
47323     /**
47324      * @cfg {Object} baseParams
47325      * Parameters to pass with all requests. e.g. baseParams: {id: '123', foo: 'bar'}.
47326      */
47327      /**
47328      
47329     /**
47330      * @cfg {Number} timeout Timeout for form actions in seconds (default is 30 seconds).
47331      */
47332     timeout: 30,
47333
47334     // private
47335     activeAction : null,
47336
47337     /**
47338      * @cfg {Boolean} trackResetOnLoad If set to true, form.reset() resets to the last loaded
47339      * or setValues() data instead of when the form was first created.
47340      */
47341     trackResetOnLoad : false,
47342     
47343     
47344     /**
47345      * childForms - used for multi-tab forms
47346      * @type {Array}
47347      */
47348     childForms : false,
47349     
47350     /**
47351      * allItems - full list of fields.
47352      * @type {Array}
47353      */
47354     allItems : false,
47355     
47356     /**
47357      * By default wait messages are displayed with Roo.MessageBox.wait. You can target a specific
47358      * element by passing it or its id or mask the form itself by passing in true.
47359      * @type Mixed
47360      */
47361     waitMsgTarget : false,
47362     
47363     /**
47364      * @type Boolean
47365      */
47366     disableMask : false,
47367     
47368     /**
47369      * @cfg {Boolean} errorMask (true|false) default false
47370      */
47371     errorMask : false,
47372     
47373     /**
47374      * @cfg {Number} maskOffset Default 100
47375      */
47376     maskOffset : 100,
47377
47378     // private
47379     initEl : function(el){
47380         this.el = Roo.get(el);
47381         this.id = this.el.id || Roo.id();
47382         this.el.on('submit', this.onSubmit, this);
47383         this.el.addClass('x-form');
47384     },
47385
47386     // private
47387     onSubmit : function(e){
47388         e.stopEvent();
47389     },
47390
47391     /**
47392      * Returns true if client-side validation on the form is successful.
47393      * @return Boolean
47394      */
47395     isValid : function(){
47396         var valid = true;
47397         var target = false;
47398         this.items.each(function(f){
47399             if(f.validate()){
47400                 return;
47401             }
47402             
47403             valid = false;
47404                 
47405             if(!target && f.el.isVisible(true)){
47406                 target = f;
47407             }
47408         });
47409         
47410         if(this.errorMask && !valid){
47411             Roo.form.BasicForm.popover.mask(this, target);
47412         }
47413         
47414         return valid;
47415     },
47416
47417     /**
47418      * DEPRICATED Returns true if any fields in this form have changed since their original load. 
47419      * @return Boolean
47420      */
47421     isDirty : function(){
47422         var dirty = false;
47423         this.items.each(function(f){
47424            if(f.isDirty()){
47425                dirty = true;
47426                return false;
47427            }
47428         });
47429         return dirty;
47430     },
47431     
47432     /**
47433      * Returns true if any fields in this form have changed since their original load. (New version)
47434      * @return Boolean
47435      */
47436     
47437     hasChanged : function()
47438     {
47439         var dirty = false;
47440         this.items.each(function(f){
47441            if(f.hasChanged()){
47442                dirty = true;
47443                return false;
47444            }
47445         });
47446         return dirty;
47447         
47448     },
47449     /**
47450      * Resets all hasChanged to 'false' -
47451      * The old 'isDirty' used 'original value..' however this breaks reset() and a few other things.
47452      * So hasChanged storage is only to be used for this purpose
47453      * @return Boolean
47454      */
47455     resetHasChanged : function()
47456     {
47457         this.items.each(function(f){
47458            f.resetHasChanged();
47459         });
47460         
47461     },
47462     
47463     
47464     /**
47465      * Performs a predefined action (submit or load) or custom actions you define on this form.
47466      * @param {String} actionName The name of the action type
47467      * @param {Object} options (optional) The options to pass to the action.  All of the config options listed
47468      * below are supported by both the submit and load actions unless otherwise noted (custom actions could also
47469      * accept other config options):
47470      * <pre>
47471 Property          Type             Description
47472 ----------------  ---------------  ----------------------------------------------------------------------------------
47473 url               String           The url for the action (defaults to the form's url)
47474 method            String           The form method to use (defaults to the form's method, or POST if not defined)
47475 params            String/Object    The params to pass (defaults to the form's baseParams, or none if not defined)
47476 clientValidation  Boolean          Applies to submit only.  Pass true to call form.isValid() prior to posting to
47477                                    validate the form on the client (defaults to false)
47478      * </pre>
47479      * @return {BasicForm} this
47480      */
47481     doAction : function(action, options){
47482         if(typeof action == 'string'){
47483             action = new Roo.form.Action.ACTION_TYPES[action](this, options);
47484         }
47485         if(this.fireEvent('beforeaction', this, action) !== false){
47486             this.beforeAction(action);
47487             action.run.defer(100, action);
47488         }
47489         return this;
47490     },
47491
47492     /**
47493      * Shortcut to do a submit action.
47494      * @param {Object} options The options to pass to the action (see {@link #doAction} for details)
47495      * @return {BasicForm} this
47496      */
47497     submit : function(options){
47498         this.doAction('submit', options);
47499         return this;
47500     },
47501
47502     /**
47503      * Shortcut to do a load action.
47504      * @param {Object} options The options to pass to the action (see {@link #doAction} for details)
47505      * @return {BasicForm} this
47506      */
47507     load : function(options){
47508         this.doAction('load', options);
47509         return this;
47510     },
47511
47512     /**
47513      * Persists the values in this form into the passed Roo.data.Record object in a beginEdit/endEdit block.
47514      * @param {Record} record The record to edit
47515      * @return {BasicForm} this
47516      */
47517     updateRecord : function(record){
47518         record.beginEdit();
47519         var fs = record.fields;
47520         fs.each(function(f){
47521             var field = this.findField(f.name);
47522             if(field){
47523                 record.set(f.name, field.getValue());
47524             }
47525         }, this);
47526         record.endEdit();
47527         return this;
47528     },
47529
47530     /**
47531      * Loads an Roo.data.Record into this form.
47532      * @param {Record} record The record to load
47533      * @return {BasicForm} this
47534      */
47535     loadRecord : function(record){
47536         this.setValues(record.data);
47537         return this;
47538     },
47539
47540     // private
47541     beforeAction : function(action){
47542         var o = action.options;
47543         
47544         if(!this.disableMask) {
47545             if(this.waitMsgTarget === true){
47546                 this.el.mask(o.waitMsg || "Sending", 'x-mask-loading');
47547             }else if(this.waitMsgTarget){
47548                 this.waitMsgTarget = Roo.get(this.waitMsgTarget);
47549                 this.waitMsgTarget.mask(o.waitMsg || "Sending", 'x-mask-loading');
47550             }else {
47551                 Roo.MessageBox.wait(o.waitMsg || "Sending", o.waitTitle || this.waitTitle || 'Please Wait...');
47552             }
47553         }
47554         
47555          
47556     },
47557
47558     // private
47559     afterAction : function(action, success){
47560         this.activeAction = null;
47561         var o = action.options;
47562         
47563         if(!this.disableMask) {
47564             if(this.waitMsgTarget === true){
47565                 this.el.unmask();
47566             }else if(this.waitMsgTarget){
47567                 this.waitMsgTarget.unmask();
47568             }else{
47569                 Roo.MessageBox.updateProgress(1);
47570                 Roo.MessageBox.hide();
47571             }
47572         }
47573         
47574         if(success){
47575             if(o.reset){
47576                 this.reset();
47577             }
47578             Roo.callback(o.success, o.scope, [this, action]);
47579             this.fireEvent('actioncomplete', this, action);
47580             
47581         }else{
47582             
47583             // failure condition..
47584             // we have a scenario where updates need confirming.
47585             // eg. if a locking scenario exists..
47586             // we look for { errors : { needs_confirm : true }} in the response.
47587             if (
47588                 (typeof(action.result) != 'undefined')  &&
47589                 (typeof(action.result.errors) != 'undefined')  &&
47590                 (typeof(action.result.errors.needs_confirm) != 'undefined')
47591            ){
47592                 var _t = this;
47593                 Roo.MessageBox.confirm(
47594                     "Change requires confirmation",
47595                     action.result.errorMsg,
47596                     function(r) {
47597                         if (r != 'yes') {
47598                             return;
47599                         }
47600                         _t.doAction('submit', { params :  { _submit_confirmed : 1 } }  );
47601                     }
47602                     
47603                 );
47604                 
47605                 
47606                 
47607                 return;
47608             }
47609             
47610             Roo.callback(o.failure, o.scope, [this, action]);
47611             // show an error message if no failed handler is set..
47612             if (!this.hasListener('actionfailed')) {
47613                 Roo.MessageBox.alert("Error",
47614                     (typeof(action.result) != 'undefined' && typeof(action.result.errorMsg) != 'undefined') ?
47615                         action.result.errorMsg :
47616                         "Saving Failed, please check your entries or try again"
47617                 );
47618             }
47619             
47620             this.fireEvent('actionfailed', this, action);
47621         }
47622         
47623     },
47624
47625     /**
47626      * Find a Roo.form.Field in this form by id, dataIndex, name or hiddenName
47627      * @param {String} id The value to search for
47628      * @return Field
47629      */
47630     findField : function(id){
47631         var field = this.items.get(id);
47632         if(!field){
47633             this.items.each(function(f){
47634                 if(f.isFormField && (f.dataIndex == id || f.id == id || f.getName() == id)){
47635                     field = f;
47636                     return false;
47637                 }
47638             });
47639         }
47640         return field || null;
47641     },
47642
47643     /**
47644      * Add a secondary form to this one, 
47645      * Used to provide tabbed forms. One form is primary, with hidden values 
47646      * which mirror the elements from the other forms.
47647      * 
47648      * @param {Roo.form.Form} form to add.
47649      * 
47650      */
47651     addForm : function(form)
47652     {
47653        
47654         if (this.childForms.indexOf(form) > -1) {
47655             // already added..
47656             return;
47657         }
47658         this.childForms.push(form);
47659         var n = '';
47660         Roo.each(form.allItems, function (fe) {
47661             
47662             n = typeof(fe.getName) == 'undefined' ? fe.name : fe.getName();
47663             if (this.findField(n)) { // already added..
47664                 return;
47665             }
47666             var add = new Roo.form.Hidden({
47667                 name : n
47668             });
47669             add.render(this.el);
47670             
47671             this.add( add );
47672         }, this);
47673         
47674     },
47675     /**
47676      * Mark fields in this form invalid in bulk.
47677      * @param {Array/Object} errors Either an array in the form [{id:'fieldId', msg:'The message'},...] or an object hash of {id: msg, id2: msg2}
47678      * @return {BasicForm} this
47679      */
47680     markInvalid : function(errors){
47681         if(errors instanceof Array){
47682             for(var i = 0, len = errors.length; i < len; i++){
47683                 var fieldError = errors[i];
47684                 var f = this.findField(fieldError.id);
47685                 if(f){
47686                     f.markInvalid(fieldError.msg);
47687                 }
47688             }
47689         }else{
47690             var field, id;
47691             for(id in errors){
47692                 if(typeof errors[id] != 'function' && (field = this.findField(id))){
47693                     field.markInvalid(errors[id]);
47694                 }
47695             }
47696         }
47697         Roo.each(this.childForms || [], function (f) {
47698             f.markInvalid(errors);
47699         });
47700         
47701         return this;
47702     },
47703
47704     /**
47705      * Set values for fields in this form in bulk.
47706      * @param {Array/Object} values Either an array in the form [{id:'fieldId', value:'foo'},...] or an object hash of {id: value, id2: value2}
47707      * @return {BasicForm} this
47708      */
47709     setValues : function(values){
47710         if(values instanceof Array){ // array of objects
47711             for(var i = 0, len = values.length; i < len; i++){
47712                 var v = values[i];
47713                 var f = this.findField(v.id);
47714                 if(f){
47715                     f.setValue(v.value);
47716                     if(this.trackResetOnLoad){
47717                         f.originalValue = f.getValue();
47718                     }
47719                 }
47720             }
47721         }else{ // object hash
47722             var field, id;
47723             for(id in values){
47724                 if(typeof values[id] != 'function' && (field = this.findField(id))){
47725                     
47726                     if (field.setFromData && 
47727                         field.valueField && 
47728                         field.displayField &&
47729                         // combos' with local stores can 
47730                         // be queried via setValue()
47731                         // to set their value..
47732                         (field.store && !field.store.isLocal)
47733                         ) {
47734                         // it's a combo
47735                         var sd = { };
47736                         sd[field.valueField] = typeof(values[field.hiddenName]) == 'undefined' ? '' : values[field.hiddenName];
47737                         sd[field.displayField] = typeof(values[field.name]) == 'undefined' ? '' : values[field.name];
47738                         field.setFromData(sd);
47739                         
47740                     } else {
47741                         field.setValue(values[id]);
47742                     }
47743                     
47744                     
47745                     if(this.trackResetOnLoad){
47746                         field.originalValue = field.getValue();
47747                     }
47748                 }
47749             }
47750         }
47751         this.resetHasChanged();
47752         
47753         
47754         Roo.each(this.childForms || [], function (f) {
47755             f.setValues(values);
47756             f.resetHasChanged();
47757         });
47758                 
47759         return this;
47760     },
47761  
47762     /**
47763      * Returns the fields in this form as an object with key/value pairs. If multiple fields exist with the same name
47764      * they are returned as an array.
47765      * @param {Boolean} asString
47766      * @return {Object}
47767      */
47768     getValues : function(asString){
47769         if (this.childForms) {
47770             // copy values from the child forms
47771             Roo.each(this.childForms, function (f) {
47772                 this.setValues(f.getValues());
47773             }, this);
47774         }
47775         
47776         // use formdata
47777         if (typeof(FormData) != 'undefined' && asString !== true) {
47778             var fd = (new FormData(this.el.dom)).entries();
47779             var ret = {};
47780             var ent = fd.next();
47781             while (!ent.done) {
47782                 ret[ent.value[0]] = ent.value[1]; // not sure how this will handle duplicates..
47783                 ent = fd.next();
47784             };
47785             return ret;
47786         }
47787         
47788         
47789         var fs = Roo.lib.Ajax.serializeForm(this.el.dom);
47790         if(asString === true){
47791             return fs;
47792         }
47793         return Roo.urlDecode(fs);
47794     },
47795     
47796     /**
47797      * Returns the fields in this form as an object with key/value pairs. 
47798      * This differs from getValues as it calls getValue on each child item, rather than using dom data.
47799      * @return {Object}
47800      */
47801     getFieldValues : function(with_hidden)
47802     {
47803         if (this.childForms) {
47804             // copy values from the child forms
47805             // should this call getFieldValues - probably not as we do not currently copy
47806             // hidden fields when we generate..
47807             Roo.each(this.childForms, function (f) {
47808                 this.setValues(f.getValues());
47809             }, this);
47810         }
47811         
47812         var ret = {};
47813         this.items.each(function(f){
47814             if (!f.getName()) {
47815                 return;
47816             }
47817             var v = f.getValue();
47818             if (f.inputType =='radio') {
47819                 if (typeof(ret[f.getName()]) == 'undefined') {
47820                     ret[f.getName()] = ''; // empty..
47821                 }
47822                 
47823                 if (!f.el.dom.checked) {
47824                     return;
47825                     
47826                 }
47827                 v = f.el.dom.value;
47828                 
47829             }
47830             
47831             // not sure if this supported any more..
47832             if ((typeof(v) == 'object') && f.getRawValue) {
47833                 v = f.getRawValue() ; // dates..
47834             }
47835             // combo boxes where name != hiddenName...
47836             if (f.name != f.getName()) {
47837                 ret[f.name] = f.getRawValue();
47838             }
47839             ret[f.getName()] = v;
47840         });
47841         
47842         return ret;
47843     },
47844
47845     /**
47846      * Clears all invalid messages in this form.
47847      * @return {BasicForm} this
47848      */
47849     clearInvalid : function(){
47850         this.items.each(function(f){
47851            f.clearInvalid();
47852         });
47853         
47854         Roo.each(this.childForms || [], function (f) {
47855             f.clearInvalid();
47856         });
47857         
47858         
47859         return this;
47860     },
47861
47862     /**
47863      * Resets this form.
47864      * @return {BasicForm} this
47865      */
47866     reset : function(){
47867         this.items.each(function(f){
47868             f.reset();
47869         });
47870         
47871         Roo.each(this.childForms || [], function (f) {
47872             f.reset();
47873         });
47874         this.resetHasChanged();
47875         
47876         return this;
47877     },
47878
47879     /**
47880      * Add Roo.form components to this form.
47881      * @param {Field} field1
47882      * @param {Field} field2 (optional)
47883      * @param {Field} etc (optional)
47884      * @return {BasicForm} this
47885      */
47886     add : function(){
47887         this.items.addAll(Array.prototype.slice.call(arguments, 0));
47888         return this;
47889     },
47890
47891
47892     /**
47893      * Removes a field from the items collection (does NOT remove its markup).
47894      * @param {Field} field
47895      * @return {BasicForm} this
47896      */
47897     remove : function(field){
47898         this.items.remove(field);
47899         return this;
47900     },
47901
47902     /**
47903      * Looks at the fields in this form, checks them for an id attribute,
47904      * and calls applyTo on the existing dom element with that id.
47905      * @return {BasicForm} this
47906      */
47907     render : function(){
47908         this.items.each(function(f){
47909             if(f.isFormField && !f.rendered && document.getElementById(f.id)){ // if the element exists
47910                 f.applyTo(f.id);
47911             }
47912         });
47913         return this;
47914     },
47915
47916     /**
47917      * Calls {@link Ext#apply} for all fields in this form with the passed object.
47918      * @param {Object} values
47919      * @return {BasicForm} this
47920      */
47921     applyToFields : function(o){
47922         this.items.each(function(f){
47923            Roo.apply(f, o);
47924         });
47925         return this;
47926     },
47927
47928     /**
47929      * Calls {@link Ext#applyIf} for all field in this form with the passed object.
47930      * @param {Object} values
47931      * @return {BasicForm} this
47932      */
47933     applyIfToFields : function(o){
47934         this.items.each(function(f){
47935            Roo.applyIf(f, o);
47936         });
47937         return this;
47938     }
47939 });
47940
47941 // back compat
47942 Roo.BasicForm = Roo.form.BasicForm;
47943
47944 Roo.apply(Roo.form.BasicForm, {
47945     
47946     popover : {
47947         
47948         padding : 5,
47949         
47950         isApplied : false,
47951         
47952         isMasked : false,
47953         
47954         form : false,
47955         
47956         target : false,
47957         
47958         intervalID : false,
47959         
47960         maskEl : false,
47961         
47962         apply : function()
47963         {
47964             if(this.isApplied){
47965                 return;
47966             }
47967             
47968             this.maskEl = {
47969                 top : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-top-mask" }, true),
47970                 left : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-left-mask" }, true),
47971                 bottom : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-bottom-mask" }, true),
47972                 right : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-right-mask" }, true)
47973             };
47974             
47975             this.maskEl.top.enableDisplayMode("block");
47976             this.maskEl.left.enableDisplayMode("block");
47977             this.maskEl.bottom.enableDisplayMode("block");
47978             this.maskEl.right.enableDisplayMode("block");
47979             
47980             Roo.get(document.body).on('click', function(){
47981                 this.unmask();
47982             }, this);
47983             
47984             Roo.get(document.body).on('touchstart', function(){
47985                 this.unmask();
47986             }, this);
47987             
47988             this.isApplied = true
47989         },
47990         
47991         mask : function(form, target)
47992         {
47993             this.form = form;
47994             
47995             this.target = target;
47996             
47997             if(!this.form.errorMask || !target.el){
47998                 return;
47999             }
48000             
48001             var scrollable = this.target.el.findScrollableParent() || this.target.el.findParent('div.x-layout-active-content', 100, true) || Roo.get(document.body);
48002             
48003             var ot = this.target.el.calcOffsetsTo(scrollable);
48004             
48005             var scrollTo = ot[1] - this.form.maskOffset;
48006             
48007             scrollTo = Math.min(scrollTo, scrollable.dom.scrollHeight);
48008             
48009             scrollable.scrollTo('top', scrollTo);
48010             
48011             var el = this.target.wrap || this.target.el;
48012             
48013             var box = el.getBox();
48014             
48015             this.maskEl.top.setStyle('position', 'absolute');
48016             this.maskEl.top.setStyle('z-index', 10000);
48017             this.maskEl.top.setSize(Roo.lib.Dom.getDocumentWidth(), box.y - this.padding);
48018             this.maskEl.top.setLeft(0);
48019             this.maskEl.top.setTop(0);
48020             this.maskEl.top.show();
48021             
48022             this.maskEl.left.setStyle('position', 'absolute');
48023             this.maskEl.left.setStyle('z-index', 10000);
48024             this.maskEl.left.setSize(box.x - this.padding, box.height + this.padding * 2);
48025             this.maskEl.left.setLeft(0);
48026             this.maskEl.left.setTop(box.y - this.padding);
48027             this.maskEl.left.show();
48028
48029             this.maskEl.bottom.setStyle('position', 'absolute');
48030             this.maskEl.bottom.setStyle('z-index', 10000);
48031             this.maskEl.bottom.setSize(Roo.lib.Dom.getDocumentWidth(), Roo.lib.Dom.getDocumentHeight() - box.bottom - this.padding);
48032             this.maskEl.bottom.setLeft(0);
48033             this.maskEl.bottom.setTop(box.bottom + this.padding);
48034             this.maskEl.bottom.show();
48035
48036             this.maskEl.right.setStyle('position', 'absolute');
48037             this.maskEl.right.setStyle('z-index', 10000);
48038             this.maskEl.right.setSize(Roo.lib.Dom.getDocumentWidth() - box.right - this.padding, box.height + this.padding * 2);
48039             this.maskEl.right.setLeft(box.right + this.padding);
48040             this.maskEl.right.setTop(box.y - this.padding);
48041             this.maskEl.right.show();
48042
48043             this.intervalID = window.setInterval(function() {
48044                 Roo.form.BasicForm.popover.unmask();
48045             }, 10000);
48046
48047             window.onwheel = function(){ return false;};
48048             
48049             (function(){ this.isMasked = true; }).defer(500, this);
48050             
48051         },
48052         
48053         unmask : function()
48054         {
48055             if(!this.isApplied || !this.isMasked || !this.form || !this.target || !this.form.errorMask){
48056                 return;
48057             }
48058             
48059             this.maskEl.top.setStyle('position', 'absolute');
48060             this.maskEl.top.setSize(0, 0).setXY([0, 0]);
48061             this.maskEl.top.hide();
48062
48063             this.maskEl.left.setStyle('position', 'absolute');
48064             this.maskEl.left.setSize(0, 0).setXY([0, 0]);
48065             this.maskEl.left.hide();
48066
48067             this.maskEl.bottom.setStyle('position', 'absolute');
48068             this.maskEl.bottom.setSize(0, 0).setXY([0, 0]);
48069             this.maskEl.bottom.hide();
48070
48071             this.maskEl.right.setStyle('position', 'absolute');
48072             this.maskEl.right.setSize(0, 0).setXY([0, 0]);
48073             this.maskEl.right.hide();
48074             
48075             window.onwheel = function(){ return true;};
48076             
48077             if(this.intervalID){
48078                 window.clearInterval(this.intervalID);
48079                 this.intervalID = false;
48080             }
48081             
48082             this.isMasked = false;
48083             
48084         }
48085         
48086     }
48087     
48088 });/*
48089  * Based on:
48090  * Ext JS Library 1.1.1
48091  * Copyright(c) 2006-2007, Ext JS, LLC.
48092  *
48093  * Originally Released Under LGPL - original licence link has changed is not relivant.
48094  *
48095  * Fork - LGPL
48096  * <script type="text/javascript">
48097  */
48098
48099 /**
48100  * @class Roo.form.Form
48101  * @extends Roo.form.BasicForm
48102  * Adds the ability to dynamically render forms with JavaScript to {@link Roo.form.BasicForm}.
48103  * @constructor
48104  * @param {Object} config Configuration options
48105  */
48106 Roo.form.Form = function(config){
48107     var xitems =  [];
48108     if (config.items) {
48109         xitems = config.items;
48110         delete config.items;
48111     }
48112    
48113     
48114     Roo.form.Form.superclass.constructor.call(this, null, config);
48115     this.url = this.url || this.action;
48116     if(!this.root){
48117         this.root = new Roo.form.Layout(Roo.applyIf({
48118             id: Roo.id()
48119         }, config));
48120     }
48121     this.active = this.root;
48122     /**
48123      * Array of all the buttons that have been added to this form via {@link addButton}
48124      * @type Array
48125      */
48126     this.buttons = [];
48127     this.allItems = [];
48128     this.addEvents({
48129         /**
48130          * @event clientvalidation
48131          * If the monitorValid config option is true, this event fires repetitively to notify of valid state
48132          * @param {Form} this
48133          * @param {Boolean} valid true if the form has passed client-side validation
48134          */
48135         clientvalidation: true,
48136         /**
48137          * @event rendered
48138          * Fires when the form is rendered
48139          * @param {Roo.form.Form} form
48140          */
48141         rendered : true
48142     });
48143     
48144     if (this.progressUrl) {
48145             // push a hidden field onto the list of fields..
48146             this.addxtype( {
48147                     xns: Roo.form, 
48148                     xtype : 'Hidden', 
48149                     name : 'UPLOAD_IDENTIFIER' 
48150             });
48151         }
48152         
48153     
48154     Roo.each(xitems, this.addxtype, this);
48155     
48156 };
48157
48158 Roo.extend(Roo.form.Form, Roo.form.BasicForm, {
48159     /**
48160      * @cfg {Number} labelWidth The width of labels. This property cascades to child containers.
48161      */
48162     /**
48163      * @cfg {String} itemCls A css class to apply to the x-form-item of fields. This property cascades to child containers.
48164      */
48165     /**
48166      * @cfg {String} buttonAlign Valid values are "left," "center" and "right" (defaults to "center")
48167      */
48168     buttonAlign:'center',
48169
48170     /**
48171      * @cfg {Number} minButtonWidth Minimum width of all buttons in pixels (defaults to 75)
48172      */
48173     minButtonWidth:75,
48174
48175     /**
48176      * @cfg {String} labelAlign Valid values are "left," "top" and "right" (defaults to "left").
48177      * This property cascades to child containers if not set.
48178      */
48179     labelAlign:'left',
48180
48181     /**
48182      * @cfg {Boolean} monitorValid If true the form monitors its valid state <b>client-side</b> and
48183      * fires a looping event with that state. This is required to bind buttons to the valid
48184      * state using the config value formBind:true on the button.
48185      */
48186     monitorValid : false,
48187
48188     /**
48189      * @cfg {Number} monitorPoll The milliseconds to poll valid state, ignored if monitorValid is not true (defaults to 200)
48190      */
48191     monitorPoll : 200,
48192     
48193     /**
48194      * @cfg {String} progressUrl - Url to return progress data 
48195      */
48196     
48197     progressUrl : false,
48198     /**
48199      * @cfg {boolean|FormData} formData - true to use new 'FormData' post, or set to a new FormData({dom form}) Object, if
48200      * sending a formdata with extra parameters - eg uploaded elements.
48201      */
48202     
48203     formData : false,
48204     
48205     /**
48206      * Opens a new {@link Roo.form.Column} container in the layout stack. If fields are passed after the config, the
48207      * fields are added and the column is closed. If no fields are passed the column remains open
48208      * until end() is called.
48209      * @param {Object} config The config to pass to the column
48210      * @param {Field} field1 (optional)
48211      * @param {Field} field2 (optional)
48212      * @param {Field} etc (optional)
48213      * @return Column The column container object
48214      */
48215     column : function(c){
48216         var col = new Roo.form.Column(c);
48217         this.start(col);
48218         if(arguments.length > 1){ // duplicate code required because of Opera
48219             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48220             this.end();
48221         }
48222         return col;
48223     },
48224
48225     /**
48226      * Opens a new {@link Roo.form.FieldSet} container in the layout stack. If fields are passed after the config, the
48227      * fields are added and the fieldset is closed. If no fields are passed the fieldset remains open
48228      * until end() is called.
48229      * @param {Object} config The config to pass to the fieldset
48230      * @param {Field} field1 (optional)
48231      * @param {Field} field2 (optional)
48232      * @param {Field} etc (optional)
48233      * @return FieldSet The fieldset container object
48234      */
48235     fieldset : function(c){
48236         var fs = new Roo.form.FieldSet(c);
48237         this.start(fs);
48238         if(arguments.length > 1){ // duplicate code required because of Opera
48239             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48240             this.end();
48241         }
48242         return fs;
48243     },
48244
48245     /**
48246      * Opens a new {@link Roo.form.Layout} container in the layout stack. If fields are passed after the config, the
48247      * fields are added and the container is closed. If no fields are passed the container remains open
48248      * until end() is called.
48249      * @param {Object} config The config to pass to the Layout
48250      * @param {Field} field1 (optional)
48251      * @param {Field} field2 (optional)
48252      * @param {Field} etc (optional)
48253      * @return Layout The container object
48254      */
48255     container : function(c){
48256         var l = new Roo.form.Layout(c);
48257         this.start(l);
48258         if(arguments.length > 1){ // duplicate code required because of Opera
48259             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48260             this.end();
48261         }
48262         return l;
48263     },
48264
48265     /**
48266      * Opens the passed container in the layout stack. The container can be any {@link Roo.form.Layout} or subclass.
48267      * @param {Object} container A Roo.form.Layout or subclass of Layout
48268      * @return {Form} this
48269      */
48270     start : function(c){
48271         // cascade label info
48272         Roo.applyIf(c, {'labelAlign': this.active.labelAlign, 'labelWidth': this.active.labelWidth, 'itemCls': this.active.itemCls});
48273         this.active.stack.push(c);
48274         c.ownerCt = this.active;
48275         this.active = c;
48276         return this;
48277     },
48278
48279     /**
48280      * Closes the current open container
48281      * @return {Form} this
48282      */
48283     end : function(){
48284         if(this.active == this.root){
48285             return this;
48286         }
48287         this.active = this.active.ownerCt;
48288         return this;
48289     },
48290
48291     /**
48292      * Add Roo.form components to the current open container (e.g. column, fieldset, etc.).  Fields added via this method
48293      * can also be passed with an additional property of fieldLabel, which if supplied, will provide the text to display
48294      * as the label of the field.
48295      * @param {Field} field1
48296      * @param {Field} field2 (optional)
48297      * @param {Field} etc. (optional)
48298      * @return {Form} this
48299      */
48300     add : function(){
48301         this.active.stack.push.apply(this.active.stack, arguments);
48302         this.allItems.push.apply(this.allItems,arguments);
48303         var r = [];
48304         for(var i = 0, a = arguments, len = a.length; i < len; i++) {
48305             if(a[i].isFormField){
48306                 r.push(a[i]);
48307             }
48308         }
48309         if(r.length > 0){
48310             Roo.form.Form.superclass.add.apply(this, r);
48311         }
48312         return this;
48313     },
48314     
48315
48316     
48317     
48318     
48319      /**
48320      * Find any element that has been added to a form, using it's ID or name
48321      * This can include framesets, columns etc. along with regular fields..
48322      * @param {String} id - id or name to find.
48323      
48324      * @return {Element} e - or false if nothing found.
48325      */
48326     findbyId : function(id)
48327     {
48328         var ret = false;
48329         if (!id) {
48330             return ret;
48331         }
48332         Roo.each(this.allItems, function(f){
48333             if (f.id == id || f.name == id ){
48334                 ret = f;
48335                 return false;
48336             }
48337         });
48338         return ret;
48339     },
48340
48341     
48342     
48343     /**
48344      * Render this form into the passed container. This should only be called once!
48345      * @param {String/HTMLElement/Element} container The element this component should be rendered into
48346      * @return {Form} this
48347      */
48348     render : function(ct)
48349     {
48350         
48351         
48352         
48353         ct = Roo.get(ct);
48354         var o = this.autoCreate || {
48355             tag: 'form',
48356             method : this.method || 'POST',
48357             id : this.id || Roo.id()
48358         };
48359         this.initEl(ct.createChild(o));
48360
48361         this.root.render(this.el);
48362         
48363        
48364              
48365         this.items.each(function(f){
48366             f.render('x-form-el-'+f.id);
48367         });
48368
48369         if(this.buttons.length > 0){
48370             // tables are required to maintain order and for correct IE layout
48371             var tb = this.el.createChild({cls:'x-form-btns-ct', cn: {
48372                 cls:"x-form-btns x-form-btns-"+this.buttonAlign,
48373                 html:'<table cellspacing="0"><tbody><tr></tr></tbody></table><div class="x-clear"></div>'
48374             }}, null, true);
48375             var tr = tb.getElementsByTagName('tr')[0];
48376             for(var i = 0, len = this.buttons.length; i < len; i++) {
48377                 var b = this.buttons[i];
48378                 var td = document.createElement('td');
48379                 td.className = 'x-form-btn-td';
48380                 b.render(tr.appendChild(td));
48381             }
48382         }
48383         if(this.monitorValid){ // initialize after render
48384             this.startMonitoring();
48385         }
48386         this.fireEvent('rendered', this);
48387         return this;
48388     },
48389
48390     /**
48391      * Adds a button to the footer of the form - this <b>must</b> be called before the form is rendered.
48392      * @param {String/Object} config A string becomes the button text, an object can either be a Button config
48393      * object or a valid Roo.DomHelper element config
48394      * @param {Function} handler The function called when the button is clicked
48395      * @param {Object} scope (optional) The scope of the handler function
48396      * @return {Roo.Button}
48397      */
48398     addButton : function(config, handler, scope){
48399         var bc = {
48400             handler: handler,
48401             scope: scope,
48402             minWidth: this.minButtonWidth,
48403             hideParent:true
48404         };
48405         if(typeof config == "string"){
48406             bc.text = config;
48407         }else{
48408             Roo.apply(bc, config);
48409         }
48410         var btn = new Roo.Button(null, bc);
48411         this.buttons.push(btn);
48412         return btn;
48413     },
48414
48415      /**
48416      * Adds a series of form elements (using the xtype property as the factory method.
48417      * Valid xtypes are:  TextField, TextArea .... Button, Layout, FieldSet, Column, (and 'end' to close a block)
48418      * @param {Object} config 
48419      */
48420     
48421     addxtype : function()
48422     {
48423         var ar = Array.prototype.slice.call(arguments, 0);
48424         var ret = false;
48425         for(var i = 0; i < ar.length; i++) {
48426             if (!ar[i]) {
48427                 continue; // skip -- if this happends something invalid got sent, we 
48428                 // should ignore it, as basically that interface element will not show up
48429                 // and that should be pretty obvious!!
48430             }
48431             
48432             if (Roo.form[ar[i].xtype]) {
48433                 ar[i].form = this;
48434                 var fe = Roo.factory(ar[i], Roo.form);
48435                 if (!ret) {
48436                     ret = fe;
48437                 }
48438                 fe.form = this;
48439                 if (fe.store) {
48440                     fe.store.form = this;
48441                 }
48442                 if (fe.isLayout) {  
48443                          
48444                     this.start(fe);
48445                     this.allItems.push(fe);
48446                     if (fe.items && fe.addxtype) {
48447                         fe.addxtype.apply(fe, fe.items);
48448                         delete fe.items;
48449                     }
48450                      this.end();
48451                     continue;
48452                 }
48453                 
48454                 
48455                  
48456                 this.add(fe);
48457               //  console.log('adding ' + ar[i].xtype);
48458             }
48459             if (ar[i].xtype == 'Button') {  
48460                 //console.log('adding button');
48461                 //console.log(ar[i]);
48462                 this.addButton(ar[i]);
48463                 this.allItems.push(fe);
48464                 continue;
48465             }
48466             
48467             if (ar[i].xtype == 'end') { // so we can add fieldsets... / layout etc.
48468                 alert('end is not supported on xtype any more, use items');
48469             //    this.end();
48470             //    //console.log('adding end');
48471             }
48472             
48473         }
48474         return ret;
48475     },
48476     
48477     /**
48478      * Starts monitoring of the valid state of this form. Usually this is done by passing the config
48479      * option "monitorValid"
48480      */
48481     startMonitoring : function(){
48482         if(!this.bound){
48483             this.bound = true;
48484             Roo.TaskMgr.start({
48485                 run : this.bindHandler,
48486                 interval : this.monitorPoll || 200,
48487                 scope: this
48488             });
48489         }
48490     },
48491
48492     /**
48493      * Stops monitoring of the valid state of this form
48494      */
48495     stopMonitoring : function(){
48496         this.bound = false;
48497     },
48498
48499     // private
48500     bindHandler : function(){
48501         if(!this.bound){
48502             return false; // stops binding
48503         }
48504         var valid = true;
48505         this.items.each(function(f){
48506             if(!f.isValid(true)){
48507                 valid = false;
48508                 return false;
48509             }
48510         });
48511         for(var i = 0, len = this.buttons.length; i < len; i++){
48512             var btn = this.buttons[i];
48513             if(btn.formBind === true && btn.disabled === valid){
48514                 btn.setDisabled(!valid);
48515             }
48516         }
48517         this.fireEvent('clientvalidation', this, valid);
48518     }
48519     
48520     
48521     
48522     
48523     
48524     
48525     
48526     
48527 });
48528
48529
48530 // back compat
48531 Roo.Form = Roo.form.Form;
48532 /*
48533  * Based on:
48534  * Ext JS Library 1.1.1
48535  * Copyright(c) 2006-2007, Ext JS, LLC.
48536  *
48537  * Originally Released Under LGPL - original licence link has changed is not relivant.
48538  *
48539  * Fork - LGPL
48540  * <script type="text/javascript">
48541  */
48542
48543 // as we use this in bootstrap.
48544 Roo.namespace('Roo.form');
48545  /**
48546  * @class Roo.form.Action
48547  * Internal Class used to handle form actions
48548  * @constructor
48549  * @param {Roo.form.BasicForm} el The form element or its id
48550  * @param {Object} config Configuration options
48551  */
48552
48553  
48554  
48555 // define the action interface
48556 Roo.form.Action = function(form, options){
48557     this.form = form;
48558     this.options = options || {};
48559 };
48560 /**
48561  * Client Validation Failed
48562  * @const 
48563  */
48564 Roo.form.Action.CLIENT_INVALID = 'client';
48565 /**
48566  * Server Validation Failed
48567  * @const 
48568  */
48569 Roo.form.Action.SERVER_INVALID = 'server';
48570  /**
48571  * Connect to Server Failed
48572  * @const 
48573  */
48574 Roo.form.Action.CONNECT_FAILURE = 'connect';
48575 /**
48576  * Reading Data from Server Failed
48577  * @const 
48578  */
48579 Roo.form.Action.LOAD_FAILURE = 'load';
48580
48581 Roo.form.Action.prototype = {
48582     type : 'default',
48583     failureType : undefined,
48584     response : undefined,
48585     result : undefined,
48586
48587     // interface method
48588     run : function(options){
48589
48590     },
48591
48592     // interface method
48593     success : function(response){
48594
48595     },
48596
48597     // interface method
48598     handleResponse : function(response){
48599
48600     },
48601
48602     // default connection failure
48603     failure : function(response){
48604         
48605         this.response = response;
48606         this.failureType = Roo.form.Action.CONNECT_FAILURE;
48607         this.form.afterAction(this, false);
48608     },
48609
48610     processResponse : function(response){
48611         this.response = response;
48612         if(!response.responseText){
48613             return true;
48614         }
48615         this.result = this.handleResponse(response);
48616         return this.result;
48617     },
48618
48619     // utility functions used internally
48620     getUrl : function(appendParams){
48621         var url = this.options.url || this.form.url || this.form.el.dom.action;
48622         if(appendParams){
48623             var p = this.getParams();
48624             if(p){
48625                 url += (url.indexOf('?') != -1 ? '&' : '?') + p;
48626             }
48627         }
48628         return url;
48629     },
48630
48631     getMethod : function(){
48632         return (this.options.method || this.form.method || this.form.el.dom.method || 'POST').toUpperCase();
48633     },
48634
48635     getParams : function(){
48636         var bp = this.form.baseParams;
48637         var p = this.options.params;
48638         if(p){
48639             if(typeof p == "object"){
48640                 p = Roo.urlEncode(Roo.applyIf(p, bp));
48641             }else if(typeof p == 'string' && bp){
48642                 p += '&' + Roo.urlEncode(bp);
48643             }
48644         }else if(bp){
48645             p = Roo.urlEncode(bp);
48646         }
48647         return p;
48648     },
48649
48650     createCallback : function(){
48651         return {
48652             success: this.success,
48653             failure: this.failure,
48654             scope: this,
48655             timeout: (this.form.timeout*1000),
48656             upload: this.form.fileUpload ? this.success : undefined
48657         };
48658     }
48659 };
48660
48661 Roo.form.Action.Submit = function(form, options){
48662     Roo.form.Action.Submit.superclass.constructor.call(this, form, options);
48663 };
48664
48665 Roo.extend(Roo.form.Action.Submit, Roo.form.Action, {
48666     type : 'submit',
48667
48668     haveProgress : false,
48669     uploadComplete : false,
48670     
48671     // uploadProgress indicator.
48672     uploadProgress : function()
48673     {
48674         if (!this.form.progressUrl) {
48675             return;
48676         }
48677         
48678         if (!this.haveProgress) {
48679             Roo.MessageBox.progress("Uploading", "Uploading");
48680         }
48681         if (this.uploadComplete) {
48682            Roo.MessageBox.hide();
48683            return;
48684         }
48685         
48686         this.haveProgress = true;
48687    
48688         var uid = this.form.findField('UPLOAD_IDENTIFIER').getValue();
48689         
48690         var c = new Roo.data.Connection();
48691         c.request({
48692             url : this.form.progressUrl,
48693             params: {
48694                 id : uid
48695             },
48696             method: 'GET',
48697             success : function(req){
48698                //console.log(data);
48699                 var rdata = false;
48700                 var edata;
48701                 try  {
48702                    rdata = Roo.decode(req.responseText)
48703                 } catch (e) {
48704                     Roo.log("Invalid data from server..");
48705                     Roo.log(edata);
48706                     return;
48707                 }
48708                 if (!rdata || !rdata.success) {
48709                     Roo.log(rdata);
48710                     Roo.MessageBox.alert(Roo.encode(rdata));
48711                     return;
48712                 }
48713                 var data = rdata.data;
48714                 
48715                 if (this.uploadComplete) {
48716                    Roo.MessageBox.hide();
48717                    return;
48718                 }
48719                    
48720                 if (data){
48721                     Roo.MessageBox.updateProgress(data.bytes_uploaded/data.bytes_total,
48722                        Math.floor((data.bytes_total - data.bytes_uploaded)/1000) + 'k remaining'
48723                     );
48724                 }
48725                 this.uploadProgress.defer(2000,this);
48726             },
48727        
48728             failure: function(data) {
48729                 Roo.log('progress url failed ');
48730                 Roo.log(data);
48731             },
48732             scope : this
48733         });
48734            
48735     },
48736     
48737     
48738     run : function()
48739     {
48740         // run get Values on the form, so it syncs any secondary forms.
48741         this.form.getValues();
48742         
48743         var o = this.options;
48744         var method = this.getMethod();
48745         var isPost = method == 'POST';
48746         if(o.clientValidation === false || this.form.isValid()){
48747             
48748             if (this.form.progressUrl) {
48749                 this.form.findField('UPLOAD_IDENTIFIER').setValue(
48750                     (new Date() * 1) + '' + Math.random());
48751                     
48752             } 
48753             
48754             
48755             Roo.Ajax.request(Roo.apply(this.createCallback(), {
48756                 form:this.form.el.dom,
48757                 url:this.getUrl(!isPost),
48758                 method: method,
48759                 params:isPost ? this.getParams() : null,
48760                 isUpload: this.form.fileUpload,
48761                 formData : this.form.formData
48762             }));
48763             
48764             this.uploadProgress();
48765
48766         }else if (o.clientValidation !== false){ // client validation failed
48767             this.failureType = Roo.form.Action.CLIENT_INVALID;
48768             this.form.afterAction(this, false);
48769         }
48770     },
48771
48772     success : function(response)
48773     {
48774         this.uploadComplete= true;
48775         if (this.haveProgress) {
48776             Roo.MessageBox.hide();
48777         }
48778         
48779         
48780         var result = this.processResponse(response);
48781         if(result === true || result.success){
48782             this.form.afterAction(this, true);
48783             return;
48784         }
48785         if(result.errors){
48786             this.form.markInvalid(result.errors);
48787             this.failureType = Roo.form.Action.SERVER_INVALID;
48788         }
48789         this.form.afterAction(this, false);
48790     },
48791     failure : function(response)
48792     {
48793         this.uploadComplete= true;
48794         if (this.haveProgress) {
48795             Roo.MessageBox.hide();
48796         }
48797         
48798         this.response = response;
48799         this.failureType = Roo.form.Action.CONNECT_FAILURE;
48800         this.form.afterAction(this, false);
48801     },
48802     
48803     handleResponse : function(response){
48804         if(this.form.errorReader){
48805             var rs = this.form.errorReader.read(response);
48806             var errors = [];
48807             if(rs.records){
48808                 for(var i = 0, len = rs.records.length; i < len; i++) {
48809                     var r = rs.records[i];
48810                     errors[i] = r.data;
48811                 }
48812             }
48813             if(errors.length < 1){
48814                 errors = null;
48815             }
48816             return {
48817                 success : rs.success,
48818                 errors : errors
48819             };
48820         }
48821         var ret = false;
48822         try {
48823             ret = Roo.decode(response.responseText);
48824         } catch (e) {
48825             ret = {
48826                 success: false,
48827                 errorMsg: "Failed to read server message: " + (response ? response.responseText : ' - no message'),
48828                 errors : []
48829             };
48830         }
48831         return ret;
48832         
48833     }
48834 });
48835
48836
48837 Roo.form.Action.Load = function(form, options){
48838     Roo.form.Action.Load.superclass.constructor.call(this, form, options);
48839     this.reader = this.form.reader;
48840 };
48841
48842 Roo.extend(Roo.form.Action.Load, Roo.form.Action, {
48843     type : 'load',
48844
48845     run : function(){
48846         
48847         Roo.Ajax.request(Roo.apply(
48848                 this.createCallback(), {
48849                     method:this.getMethod(),
48850                     url:this.getUrl(false),
48851                     params:this.getParams()
48852         }));
48853     },
48854
48855     success : function(response){
48856         
48857         var result = this.processResponse(response);
48858         if(result === true || !result.success || !result.data){
48859             this.failureType = Roo.form.Action.LOAD_FAILURE;
48860             this.form.afterAction(this, false);
48861             return;
48862         }
48863         this.form.clearInvalid();
48864         this.form.setValues(result.data);
48865         this.form.afterAction(this, true);
48866     },
48867
48868     handleResponse : function(response){
48869         if(this.form.reader){
48870             var rs = this.form.reader.read(response);
48871             var data = rs.records && rs.records[0] ? rs.records[0].data : null;
48872             return {
48873                 success : rs.success,
48874                 data : data
48875             };
48876         }
48877         return Roo.decode(response.responseText);
48878     }
48879 });
48880
48881 Roo.form.Action.ACTION_TYPES = {
48882     'load' : Roo.form.Action.Load,
48883     'submit' : Roo.form.Action.Submit
48884 };/*
48885  * Based on:
48886  * Ext JS Library 1.1.1
48887  * Copyright(c) 2006-2007, Ext JS, LLC.
48888  *
48889  * Originally Released Under LGPL - original licence link has changed is not relivant.
48890  *
48891  * Fork - LGPL
48892  * <script type="text/javascript">
48893  */
48894  
48895 /**
48896  * @class Roo.form.Layout
48897  * @extends Roo.Component
48898  * Creates a container for layout and rendering of fields in an {@link Roo.form.Form}.
48899  * @constructor
48900  * @param {Object} config Configuration options
48901  */
48902 Roo.form.Layout = function(config){
48903     var xitems = [];
48904     if (config.items) {
48905         xitems = config.items;
48906         delete config.items;
48907     }
48908     Roo.form.Layout.superclass.constructor.call(this, config);
48909     this.stack = [];
48910     Roo.each(xitems, this.addxtype, this);
48911      
48912 };
48913
48914 Roo.extend(Roo.form.Layout, Roo.Component, {
48915     /**
48916      * @cfg {String/Object} autoCreate
48917      * A DomHelper element spec used to autocreate the layout (defaults to {tag: 'div', cls: 'x-form-ct'})
48918      */
48919     /**
48920      * @cfg {String/Object/Function} style
48921      * A style specification string, e.g. "width:100px", or object in the form {width:"100px"}, or
48922      * a function which returns such a specification.
48923      */
48924     /**
48925      * @cfg {String} labelAlign
48926      * Valid values are "left," "top" and "right" (defaults to "left")
48927      */
48928     /**
48929      * @cfg {Number} labelWidth
48930      * Fixed width in pixels of all field labels (defaults to undefined)
48931      */
48932     /**
48933      * @cfg {Boolean} clear
48934      * True to add a clearing element at the end of this layout, equivalent to CSS clear: both (defaults to true)
48935      */
48936     clear : true,
48937     /**
48938      * @cfg {String} labelSeparator
48939      * The separator to use after field labels (defaults to ':')
48940      */
48941     labelSeparator : ':',
48942     /**
48943      * @cfg {Boolean} hideLabels
48944      * True to suppress the display of field labels in this layout (defaults to false)
48945      */
48946     hideLabels : false,
48947
48948     // private
48949     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct'},
48950     
48951     isLayout : true,
48952     
48953     // private
48954     onRender : function(ct, position){
48955         if(this.el){ // from markup
48956             this.el = Roo.get(this.el);
48957         }else {  // generate
48958             var cfg = this.getAutoCreate();
48959             this.el = ct.createChild(cfg, position);
48960         }
48961         if(this.style){
48962             this.el.applyStyles(this.style);
48963         }
48964         if(this.labelAlign){
48965             this.el.addClass('x-form-label-'+this.labelAlign);
48966         }
48967         if(this.hideLabels){
48968             this.labelStyle = "display:none";
48969             this.elementStyle = "padding-left:0;";
48970         }else{
48971             if(typeof this.labelWidth == 'number'){
48972                 this.labelStyle = "width:"+this.labelWidth+"px;";
48973                 this.elementStyle = "padding-left:"+((this.labelWidth+(typeof this.labelPad == 'number' ? this.labelPad : 5))+'px')+";";
48974             }
48975             if(this.labelAlign == 'top'){
48976                 this.labelStyle = "width:auto;";
48977                 this.elementStyle = "padding-left:0;";
48978             }
48979         }
48980         var stack = this.stack;
48981         var slen = stack.length;
48982         if(slen > 0){
48983             if(!this.fieldTpl){
48984                 var t = new Roo.Template(
48985                     '<div class="x-form-item {5}">',
48986                         '<label for="{0}" style="{2}">{1}{4}</label>',
48987                         '<div class="x-form-element" id="x-form-el-{0}" style="{3}">',
48988                         '</div>',
48989                     '</div><div class="x-form-clear-left"></div>'
48990                 );
48991                 t.disableFormats = true;
48992                 t.compile();
48993                 Roo.form.Layout.prototype.fieldTpl = t;
48994             }
48995             for(var i = 0; i < slen; i++) {
48996                 if(stack[i].isFormField){
48997                     this.renderField(stack[i]);
48998                 }else{
48999                     this.renderComponent(stack[i]);
49000                 }
49001             }
49002         }
49003         if(this.clear){
49004             this.el.createChild({cls:'x-form-clear'});
49005         }
49006     },
49007
49008     // private
49009     renderField : function(f){
49010         f.fieldEl = Roo.get(this.fieldTpl.append(this.el, [
49011                f.id, //0
49012                f.fieldLabel, //1
49013                f.labelStyle||this.labelStyle||'', //2
49014                this.elementStyle||'', //3
49015                typeof f.labelSeparator == 'undefined' ? this.labelSeparator : f.labelSeparator, //4
49016                f.itemCls||this.itemCls||''  //5
49017        ], true).getPrevSibling());
49018     },
49019
49020     // private
49021     renderComponent : function(c){
49022         c.render(c.isLayout ? this.el : this.el.createChild());    
49023     },
49024     /**
49025      * Adds a object form elements (using the xtype property as the factory method.)
49026      * Valid xtypes are:  TextField, TextArea .... Button, Layout, FieldSet, Column
49027      * @param {Object} config 
49028      */
49029     addxtype : function(o)
49030     {
49031         // create the lement.
49032         o.form = this.form;
49033         var fe = Roo.factory(o, Roo.form);
49034         this.form.allItems.push(fe);
49035         this.stack.push(fe);
49036         
49037         if (fe.isFormField) {
49038             this.form.items.add(fe);
49039         }
49040          
49041         return fe;
49042     }
49043 });
49044
49045 /**
49046  * @class Roo.form.Column
49047  * @extends Roo.form.Layout
49048  * Creates a column container for layout and rendering of fields in an {@link Roo.form.Form}.
49049  * @constructor
49050  * @param {Object} config Configuration options
49051  */
49052 Roo.form.Column = function(config){
49053     Roo.form.Column.superclass.constructor.call(this, config);
49054 };
49055
49056 Roo.extend(Roo.form.Column, Roo.form.Layout, {
49057     /**
49058      * @cfg {Number/String} width
49059      * The fixed width of the column in pixels or CSS value (defaults to "auto")
49060      */
49061     /**
49062      * @cfg {String/Object} autoCreate
49063      * A DomHelper element spec used to autocreate the column (defaults to {tag: 'div', cls: 'x-form-ct x-form-column'})
49064      */
49065
49066     // private
49067     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct x-form-column'},
49068
49069     // private
49070     onRender : function(ct, position){
49071         Roo.form.Column.superclass.onRender.call(this, ct, position);
49072         if(this.width){
49073             this.el.setWidth(this.width);
49074         }
49075     }
49076 });
49077
49078
49079 /**
49080  * @class Roo.form.Row
49081  * @extends Roo.form.Layout
49082  * Creates a row container for layout and rendering of fields in an {@link Roo.form.Form}.
49083  * @constructor
49084  * @param {Object} config Configuration options
49085  */
49086
49087  
49088 Roo.form.Row = function(config){
49089     Roo.form.Row.superclass.constructor.call(this, config);
49090 };
49091  
49092 Roo.extend(Roo.form.Row, Roo.form.Layout, {
49093       /**
49094      * @cfg {Number/String} width
49095      * The fixed width of the column in pixels or CSS value (defaults to "auto")
49096      */
49097     /**
49098      * @cfg {Number/String} height
49099      * The fixed height of the column in pixels or CSS value (defaults to "auto")
49100      */
49101     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct x-form-row'},
49102     
49103     padWidth : 20,
49104     // private
49105     onRender : function(ct, position){
49106         //console.log('row render');
49107         if(!this.rowTpl){
49108             var t = new Roo.Template(
49109                 '<div class="x-form-item {5}" style="float:left;width:{6}px">',
49110                     '<label for="{0}" style="{2}">{1}{4}</label>',
49111                     '<div class="x-form-element" id="x-form-el-{0}" style="{3}">',
49112                     '</div>',
49113                 '</div>'
49114             );
49115             t.disableFormats = true;
49116             t.compile();
49117             Roo.form.Layout.prototype.rowTpl = t;
49118         }
49119         this.fieldTpl = this.rowTpl;
49120         
49121         //console.log('lw' + this.labelWidth +', la:' + this.labelAlign);
49122         var labelWidth = 100;
49123         
49124         if ((this.labelAlign != 'top')) {
49125             if (typeof this.labelWidth == 'number') {
49126                 labelWidth = this.labelWidth
49127             }
49128             this.padWidth =  20 + labelWidth;
49129             
49130         }
49131         
49132         Roo.form.Column.superclass.onRender.call(this, ct, position);
49133         if(this.width){
49134             this.el.setWidth(this.width);
49135         }
49136         if(this.height){
49137             this.el.setHeight(this.height);
49138         }
49139     },
49140     
49141     // private
49142     renderField : function(f){
49143         f.fieldEl = this.fieldTpl.append(this.el, [
49144                f.id, f.fieldLabel,
49145                f.labelStyle||this.labelStyle||'',
49146                this.elementStyle||'',
49147                typeof f.labelSeparator == 'undefined' ? this.labelSeparator : f.labelSeparator,
49148                f.itemCls||this.itemCls||'',
49149                f.width ? f.width + this.padWidth : 160 + this.padWidth
49150        ],true);
49151     }
49152 });
49153  
49154
49155 /**
49156  * @class Roo.form.FieldSet
49157  * @extends Roo.form.Layout
49158  * Creates a fieldset container for layout and rendering of fields in an {@link Roo.form.Form}.
49159  * @constructor
49160  * @param {Object} config Configuration options
49161  */
49162 Roo.form.FieldSet = function(config){
49163     Roo.form.FieldSet.superclass.constructor.call(this, config);
49164 };
49165
49166 Roo.extend(Roo.form.FieldSet, Roo.form.Layout, {
49167     /**
49168      * @cfg {String} legend
49169      * The text to display as the legend for the FieldSet (defaults to '')
49170      */
49171     /**
49172      * @cfg {String/Object} autoCreate
49173      * A DomHelper element spec used to autocreate the fieldset (defaults to {tag: 'fieldset', cn: {tag:'legend'}})
49174      */
49175
49176     // private
49177     defaultAutoCreate : {tag: 'fieldset', cn: {tag:'legend'}},
49178
49179     // private
49180     onRender : function(ct, position){
49181         Roo.form.FieldSet.superclass.onRender.call(this, ct, position);
49182         if(this.legend){
49183             this.setLegend(this.legend);
49184         }
49185     },
49186
49187     // private
49188     setLegend : function(text){
49189         if(this.rendered){
49190             this.el.child('legend').update(text);
49191         }
49192     }
49193 });/*
49194  * Based on:
49195  * Ext JS Library 1.1.1
49196  * Copyright(c) 2006-2007, Ext JS, LLC.
49197  *
49198  * Originally Released Under LGPL - original licence link has changed is not relivant.
49199  *
49200  * Fork - LGPL
49201  * <script type="text/javascript">
49202  */
49203 /**
49204  * @class Roo.form.VTypes
49205  * Overridable validation definitions. The validations provided are basic and intended to be easily customizable and extended.
49206  * @singleton
49207  */
49208 Roo.form.VTypes = function(){
49209     // closure these in so they are only created once.
49210     var alpha = /^[a-zA-Z_]+$/;
49211     var alphanum = /^[a-zA-Z0-9_]+$/;
49212     var email = /^([\w]+)(.[\w]+)*@([\w-]+\.){1,5}([A-Za-z]){2,24}$/;
49213     var url = /(((https?)|(ftp)):\/\/([\-\w]+\.)+\w{2,3}(\/[%\-\w]+(\.\w{2,})?)*(([\w\-\.\?\\\/+@&#;`~=%!]*)(\.\w{2,})?)*\/?)/i;
49214
49215     // All these messages and functions are configurable
49216     return {
49217         /**
49218          * The function used to validate email addresses
49219          * @param {String} value The email address
49220          */
49221         'email' : function(v){
49222             return email.test(v);
49223         },
49224         /**
49225          * The error text to display when the email validation function returns false
49226          * @type String
49227          */
49228         'emailText' : 'This field should be an e-mail address in the format "user@domain.com"',
49229         /**
49230          * The keystroke filter mask to be applied on email input
49231          * @type RegExp
49232          */
49233         'emailMask' : /[a-z0-9_\.\-@]/i,
49234
49235         /**
49236          * The function used to validate URLs
49237          * @param {String} value The URL
49238          */
49239         'url' : function(v){
49240             return url.test(v);
49241         },
49242         /**
49243          * The error text to display when the url validation function returns false
49244          * @type String
49245          */
49246         'urlText' : 'This field should be a URL in the format "http:/'+'/www.domain.com"',
49247         
49248         /**
49249          * The function used to validate alpha values
49250          * @param {String} value The value
49251          */
49252         'alpha' : function(v){
49253             return alpha.test(v);
49254         },
49255         /**
49256          * The error text to display when the alpha validation function returns false
49257          * @type String
49258          */
49259         'alphaText' : 'This field should only contain letters and _',
49260         /**
49261          * The keystroke filter mask to be applied on alpha input
49262          * @type RegExp
49263          */
49264         'alphaMask' : /[a-z_]/i,
49265
49266         /**
49267          * The function used to validate alphanumeric values
49268          * @param {String} value The value
49269          */
49270         'alphanum' : function(v){
49271             return alphanum.test(v);
49272         },
49273         /**
49274          * The error text to display when the alphanumeric validation function returns false
49275          * @type String
49276          */
49277         'alphanumText' : 'This field should only contain letters, numbers and _',
49278         /**
49279          * The keystroke filter mask to be applied on alphanumeric input
49280          * @type RegExp
49281          */
49282         'alphanumMask' : /[a-z0-9_]/i
49283     };
49284 }();//<script type="text/javascript">
49285
49286 /**
49287  * @class Roo.form.FCKeditor
49288  * @extends Roo.form.TextArea
49289  * Wrapper around the FCKEditor http://www.fckeditor.net
49290  * @constructor
49291  * Creates a new FCKeditor
49292  * @param {Object} config Configuration options
49293  */
49294 Roo.form.FCKeditor = function(config){
49295     Roo.form.FCKeditor.superclass.constructor.call(this, config);
49296     this.addEvents({
49297          /**
49298          * @event editorinit
49299          * Fired when the editor is initialized - you can add extra handlers here..
49300          * @param {FCKeditor} this
49301          * @param {Object} the FCK object.
49302          */
49303         editorinit : true
49304     });
49305     
49306     
49307 };
49308 Roo.form.FCKeditor.editors = { };
49309 Roo.extend(Roo.form.FCKeditor, Roo.form.TextArea,
49310 {
49311     //defaultAutoCreate : {
49312     //    tag : "textarea",style   : "width:100px;height:60px;" ,autocomplete    : "off"
49313     //},
49314     // private
49315     /**
49316      * @cfg {Object} fck options - see fck manual for details.
49317      */
49318     fckconfig : false,
49319     
49320     /**
49321      * @cfg {Object} fck toolbar set (Basic or Default)
49322      */
49323     toolbarSet : 'Basic',
49324     /**
49325      * @cfg {Object} fck BasePath
49326      */ 
49327     basePath : '/fckeditor/',
49328     
49329     
49330     frame : false,
49331     
49332     value : '',
49333     
49334    
49335     onRender : function(ct, position)
49336     {
49337         if(!this.el){
49338             this.defaultAutoCreate = {
49339                 tag: "textarea",
49340                 style:"width:300px;height:60px;",
49341                 autocomplete: "new-password"
49342             };
49343         }
49344         Roo.form.FCKeditor.superclass.onRender.call(this, ct, position);
49345         /*
49346         if(this.grow){
49347             this.textSizeEl = Roo.DomHelper.append(document.body, {tag: "pre", cls: "x-form-grow-sizer"});
49348             if(this.preventScrollbars){
49349                 this.el.setStyle("overflow", "hidden");
49350             }
49351             this.el.setHeight(this.growMin);
49352         }
49353         */
49354         //console.log('onrender' + this.getId() );
49355         Roo.form.FCKeditor.editors[this.getId()] = this;
49356          
49357
49358         this.replaceTextarea() ;
49359         
49360     },
49361     
49362     getEditor : function() {
49363         return this.fckEditor;
49364     },
49365     /**
49366      * Sets a data value into the field and validates it.  To set the value directly without validation see {@link #setRawValue}.
49367      * @param {Mixed} value The value to set
49368      */
49369     
49370     
49371     setValue : function(value)
49372     {
49373         //console.log('setValue: ' + value);
49374         
49375         if(typeof(value) == 'undefined') { // not sure why this is happending...
49376             return;
49377         }
49378         Roo.form.FCKeditor.superclass.setValue.apply(this,[value]);
49379         
49380         //if(!this.el || !this.getEditor()) {
49381         //    this.value = value;
49382             //this.setValue.defer(100,this,[value]);    
49383         //    return;
49384         //} 
49385         
49386         if(!this.getEditor()) {
49387             return;
49388         }
49389         
49390         this.getEditor().SetData(value);
49391         
49392         //
49393
49394     },
49395
49396     /**
49397      * Returns the normalized data value (undefined or emptyText will be returned as '').  To return the raw value see {@link #getRawValue}.
49398      * @return {Mixed} value The field value
49399      */
49400     getValue : function()
49401     {
49402         
49403         if (this.frame && this.frame.dom.style.display == 'none') {
49404             return Roo.form.FCKeditor.superclass.getValue.call(this);
49405         }
49406         
49407         if(!this.el || !this.getEditor()) {
49408            
49409            // this.getValue.defer(100,this); 
49410             return this.value;
49411         }
49412        
49413         
49414         var value=this.getEditor().GetData();
49415         Roo.form.FCKeditor.superclass.setValue.apply(this,[value]);
49416         return Roo.form.FCKeditor.superclass.getValue.call(this);
49417         
49418
49419     },
49420
49421     /**
49422      * Returns the raw data value which may or may not be a valid, defined value.  To return a normalized value see {@link #getValue}.
49423      * @return {Mixed} value The field value
49424      */
49425     getRawValue : function()
49426     {
49427         if (this.frame && this.frame.dom.style.display == 'none') {
49428             return Roo.form.FCKeditor.superclass.getRawValue.call(this);
49429         }
49430         
49431         if(!this.el || !this.getEditor()) {
49432             //this.getRawValue.defer(100,this); 
49433             return this.value;
49434             return;
49435         }
49436         
49437         
49438         
49439         var value=this.getEditor().GetData();
49440         Roo.form.FCKeditor.superclass.setRawValue.apply(this,[value]);
49441         return Roo.form.FCKeditor.superclass.getRawValue.call(this);
49442          
49443     },
49444     
49445     setSize : function(w,h) {
49446         
49447         
49448         
49449         //if (this.frame && this.frame.dom.style.display == 'none') {
49450         //    Roo.form.FCKeditor.superclass.setSize.apply(this, [w, h]);
49451         //    return;
49452         //}
49453         //if(!this.el || !this.getEditor()) {
49454         //    this.setSize.defer(100,this, [w,h]); 
49455         //    return;
49456         //}
49457         
49458         
49459         
49460         Roo.form.FCKeditor.superclass.setSize.apply(this, [w, h]);
49461         
49462         this.frame.dom.setAttribute('width', w);
49463         this.frame.dom.setAttribute('height', h);
49464         this.frame.setSize(w,h);
49465         
49466     },
49467     
49468     toggleSourceEdit : function(value) {
49469         
49470       
49471          
49472         this.el.dom.style.display = value ? '' : 'none';
49473         this.frame.dom.style.display = value ?  'none' : '';
49474         
49475     },
49476     
49477     
49478     focus: function(tag)
49479     {
49480         if (this.frame.dom.style.display == 'none') {
49481             return Roo.form.FCKeditor.superclass.focus.call(this);
49482         }
49483         if(!this.el || !this.getEditor()) {
49484             this.focus.defer(100,this, [tag]); 
49485             return;
49486         }
49487         
49488         
49489         
49490         
49491         var tgs = this.getEditor().EditorDocument.getElementsByTagName(tag);
49492         this.getEditor().Focus();
49493         if (tgs.length) {
49494             if (!this.getEditor().Selection.GetSelection()) {
49495                 this.focus.defer(100,this, [tag]); 
49496                 return;
49497             }
49498             
49499             
49500             var r = this.getEditor().EditorDocument.createRange();
49501             r.setStart(tgs[0],0);
49502             r.setEnd(tgs[0],0);
49503             this.getEditor().Selection.GetSelection().removeAllRanges();
49504             this.getEditor().Selection.GetSelection().addRange(r);
49505             this.getEditor().Focus();
49506         }
49507         
49508     },
49509     
49510     
49511     
49512     replaceTextarea : function()
49513     {
49514         if ( document.getElementById( this.getId() + '___Frame' ) ) {
49515             return ;
49516         }
49517         //if ( !this.checkBrowser || this._isCompatibleBrowser() )
49518         //{
49519             // We must check the elements firstly using the Id and then the name.
49520         var oTextarea = document.getElementById( this.getId() );
49521         
49522         var colElementsByName = document.getElementsByName( this.getId() ) ;
49523          
49524         oTextarea.style.display = 'none' ;
49525
49526         if ( oTextarea.tabIndex ) {            
49527             this.TabIndex = oTextarea.tabIndex ;
49528         }
49529         
49530         this._insertHtmlBefore( this._getConfigHtml(), oTextarea ) ;
49531         this._insertHtmlBefore( this._getIFrameHtml(), oTextarea ) ;
49532         this.frame = Roo.get(this.getId() + '___Frame')
49533     },
49534     
49535     _getConfigHtml : function()
49536     {
49537         var sConfig = '' ;
49538
49539         for ( var o in this.fckconfig ) {
49540             sConfig += sConfig.length > 0  ? '&amp;' : '';
49541             sConfig += encodeURIComponent( o ) + '=' + encodeURIComponent( this.fckconfig[o] ) ;
49542         }
49543
49544         return '<input type="hidden" id="' + this.getId() + '___Config" value="' + sConfig + '" style="display:none" />' ;
49545     },
49546     
49547     
49548     _getIFrameHtml : function()
49549     {
49550         var sFile = 'fckeditor.html' ;
49551         /* no idea what this is about..
49552         try
49553         {
49554             if ( (/fcksource=true/i).test( window.top.location.search ) )
49555                 sFile = 'fckeditor.original.html' ;
49556         }
49557         catch (e) { 
49558         */
49559
49560         var sLink = this.basePath + 'editor/' + sFile + '?InstanceName=' + encodeURIComponent( this.getId() ) ;
49561         sLink += this.toolbarSet ? ( '&amp;Toolbar=' + this.toolbarSet)  : '';
49562         
49563         
49564         var html = '<iframe id="' + this.getId() +
49565             '___Frame" src="' + sLink +
49566             '" width="' + this.width +
49567             '" height="' + this.height + '"' +
49568             (this.tabIndex ?  ' tabindex="' + this.tabIndex + '"' :'' ) +
49569             ' frameborder="0" scrolling="no"></iframe>' ;
49570
49571         return html ;
49572     },
49573     
49574     _insertHtmlBefore : function( html, element )
49575     {
49576         if ( element.insertAdjacentHTML )       {
49577             // IE
49578             element.insertAdjacentHTML( 'beforeBegin', html ) ;
49579         } else { // Gecko
49580             var oRange = document.createRange() ;
49581             oRange.setStartBefore( element ) ;
49582             var oFragment = oRange.createContextualFragment( html );
49583             element.parentNode.insertBefore( oFragment, element ) ;
49584         }
49585     }
49586     
49587     
49588   
49589     
49590     
49591     
49592     
49593
49594 });
49595
49596 //Roo.reg('fckeditor', Roo.form.FCKeditor);
49597
49598 function FCKeditor_OnComplete(editorInstance){
49599     var f = Roo.form.FCKeditor.editors[editorInstance.Name];
49600     f.fckEditor = editorInstance;
49601     //console.log("loaded");
49602     f.fireEvent('editorinit', f, editorInstance);
49603
49604   
49605
49606  
49607
49608
49609
49610
49611
49612
49613
49614
49615
49616
49617
49618
49619
49620
49621
49622 //<script type="text/javascript">
49623 /**
49624  * @class Roo.form.GridField
49625  * @extends Roo.form.Field
49626  * Embed a grid (or editable grid into a form)
49627  * STATUS ALPHA
49628  * 
49629  * This embeds a grid in a form, the value of the field should be the json encoded array of rows
49630  * it needs 
49631  * xgrid.store = Roo.data.Store
49632  * xgrid.store.proxy = Roo.data.MemoryProxy (data = [] )
49633  * xgrid.store.reader = Roo.data.JsonReader 
49634  * 
49635  * 
49636  * @constructor
49637  * Creates a new GridField
49638  * @param {Object} config Configuration options
49639  */
49640 Roo.form.GridField = function(config){
49641     Roo.form.GridField.superclass.constructor.call(this, config);
49642      
49643 };
49644
49645 Roo.extend(Roo.form.GridField, Roo.form.Field,  {
49646     /**
49647      * @cfg {Number} width  - used to restrict width of grid..
49648      */
49649     width : 100,
49650     /**
49651      * @cfg {Number} height - used to restrict height of grid..
49652      */
49653     height : 50,
49654      /**
49655      * @cfg {Object} xgrid (xtype'd description of grid) { xtype : 'Grid', dataSource: .... }
49656          * 
49657          *}
49658      */
49659     xgrid : false, 
49660     /**
49661      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49662      * {tag: "input", type: "checkbox", autocomplete: "off"})
49663      */
49664    // defaultAutoCreate : { tag: 'div' },
49665     defaultAutoCreate : { tag: 'input', type: 'hidden', autocomplete: 'new-password'},
49666     /**
49667      * @cfg {String} addTitle Text to include for adding a title.
49668      */
49669     addTitle : false,
49670     //
49671     onResize : function(){
49672         Roo.form.Field.superclass.onResize.apply(this, arguments);
49673     },
49674
49675     initEvents : function(){
49676         // Roo.form.Checkbox.superclass.initEvents.call(this);
49677         // has no events...
49678        
49679     },
49680
49681
49682     getResizeEl : function(){
49683         return this.wrap;
49684     },
49685
49686     getPositionEl : function(){
49687         return this.wrap;
49688     },
49689
49690     // private
49691     onRender : function(ct, position){
49692         
49693         this.style = this.style || 'overflow: hidden; border:1px solid #c3daf9;';
49694         var style = this.style;
49695         delete this.style;
49696         
49697         Roo.form.GridField.superclass.onRender.call(this, ct, position);
49698         this.wrap = this.el.wrap({cls: ''}); // not sure why ive done thsi...
49699         this.viewEl = this.wrap.createChild({ tag: 'div' });
49700         if (style) {
49701             this.viewEl.applyStyles(style);
49702         }
49703         if (this.width) {
49704             this.viewEl.setWidth(this.width);
49705         }
49706         if (this.height) {
49707             this.viewEl.setHeight(this.height);
49708         }
49709         //if(this.inputValue !== undefined){
49710         //this.setValue(this.value);
49711         
49712         
49713         this.grid = new Roo.grid[this.xgrid.xtype](this.viewEl, this.xgrid);
49714         
49715         
49716         this.grid.render();
49717         this.grid.getDataSource().on('remove', this.refreshValue, this);
49718         this.grid.getDataSource().on('update', this.refreshValue, this);
49719         this.grid.on('afteredit', this.refreshValue, this);
49720  
49721     },
49722      
49723     
49724     /**
49725      * Sets the value of the item. 
49726      * @param {String} either an object  or a string..
49727      */
49728     setValue : function(v){
49729         //this.value = v;
49730         v = v || []; // empty set..
49731         // this does not seem smart - it really only affects memoryproxy grids..
49732         if (this.grid && this.grid.getDataSource() && typeof(v) != 'undefined') {
49733             var ds = this.grid.getDataSource();
49734             // assumes a json reader..
49735             var data = {}
49736             data[ds.reader.meta.root ] =  typeof(v) == 'string' ? Roo.decode(v) : v;
49737             ds.loadData( data);
49738         }
49739         // clear selection so it does not get stale.
49740         if (this.grid.sm) { 
49741             this.grid.sm.clearSelections();
49742         }
49743         
49744         Roo.form.GridField.superclass.setValue.call(this, v);
49745         this.refreshValue();
49746         // should load data in the grid really....
49747     },
49748     
49749     // private
49750     refreshValue: function() {
49751          var val = [];
49752         this.grid.getDataSource().each(function(r) {
49753             val.push(r.data);
49754         });
49755         this.el.dom.value = Roo.encode(val);
49756     }
49757     
49758      
49759     
49760     
49761 });/*
49762  * Based on:
49763  * Ext JS Library 1.1.1
49764  * Copyright(c) 2006-2007, Ext JS, LLC.
49765  *
49766  * Originally Released Under LGPL - original licence link has changed is not relivant.
49767  *
49768  * Fork - LGPL
49769  * <script type="text/javascript">
49770  */
49771 /**
49772  * @class Roo.form.DisplayField
49773  * @extends Roo.form.Field
49774  * A generic Field to display non-editable data.
49775  * @cfg {Boolean} closable (true|false) default false
49776  * @constructor
49777  * Creates a new Display Field item.
49778  * @param {Object} config Configuration options
49779  */
49780 Roo.form.DisplayField = function(config){
49781     Roo.form.DisplayField.superclass.constructor.call(this, config);
49782     
49783     this.addEvents({
49784         /**
49785          * @event close
49786          * Fires after the click the close btn
49787              * @param {Roo.form.DisplayField} this
49788              */
49789         close : true
49790     });
49791 };
49792
49793 Roo.extend(Roo.form.DisplayField, Roo.form.TextField,  {
49794     inputType:      'hidden',
49795     allowBlank:     true,
49796     readOnly:         true,
49797     
49798  
49799     /**
49800      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
49801      */
49802     focusClass : undefined,
49803     /**
49804      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
49805      */
49806     fieldClass: 'x-form-field',
49807     
49808      /**
49809      * @cfg {Function} valueRenderer The renderer for the field (so you can reformat output). should return raw HTML
49810      */
49811     valueRenderer: undefined,
49812     
49813     width: 100,
49814     /**
49815      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49816      * {tag: "input", type: "checkbox", autocomplete: "off"})
49817      */
49818      
49819  //   defaultAutoCreate : { tag: 'input', type: 'hidden', autocomplete: 'off'},
49820  
49821     closable : false,
49822     
49823     onResize : function(){
49824         Roo.form.DisplayField.superclass.onResize.apply(this, arguments);
49825         
49826     },
49827
49828     initEvents : function(){
49829         // Roo.form.Checkbox.superclass.initEvents.call(this);
49830         // has no events...
49831         
49832         if(this.closable){
49833             this.closeEl.on('click', this.onClose, this);
49834         }
49835        
49836     },
49837
49838
49839     getResizeEl : function(){
49840         return this.wrap;
49841     },
49842
49843     getPositionEl : function(){
49844         return this.wrap;
49845     },
49846
49847     // private
49848     onRender : function(ct, position){
49849         
49850         Roo.form.DisplayField.superclass.onRender.call(this, ct, position);
49851         //if(this.inputValue !== undefined){
49852         this.wrap = this.el.wrap();
49853         
49854         this.viewEl = this.wrap.createChild({ tag: 'div', cls: 'x-form-displayfield'});
49855         
49856         if(this.closable){
49857             this.closeEl = this.wrap.createChild({ tag: 'div', cls: 'x-dlg-close'});
49858         }
49859         
49860         if (this.bodyStyle) {
49861             this.viewEl.applyStyles(this.bodyStyle);
49862         }
49863         //this.viewEl.setStyle('padding', '2px');
49864         
49865         this.setValue(this.value);
49866         
49867     },
49868 /*
49869     // private
49870     initValue : Roo.emptyFn,
49871
49872   */
49873
49874         // private
49875     onClick : function(){
49876         
49877     },
49878
49879     /**
49880      * Sets the checked state of the checkbox.
49881      * @param {Boolean/String} checked True, 'true', '1', or 'on' to check the checkbox, any other value will uncheck it.
49882      */
49883     setValue : function(v){
49884         this.value = v;
49885         var html = this.valueRenderer ?  this.valueRenderer(v) : String.format('{0}', v);
49886         // this might be called before we have a dom element..
49887         if (!this.viewEl) {
49888             return;
49889         }
49890         this.viewEl.dom.innerHTML = html;
49891         Roo.form.DisplayField.superclass.setValue.call(this, v);
49892
49893     },
49894     
49895     onClose : function(e)
49896     {
49897         e.preventDefault();
49898         
49899         this.fireEvent('close', this);
49900     }
49901 });/*
49902  * 
49903  * Licence- LGPL
49904  * 
49905  */
49906
49907 /**
49908  * @class Roo.form.DayPicker
49909  * @extends Roo.form.Field
49910  * A Day picker show [M] [T] [W] ....
49911  * @constructor
49912  * Creates a new Day Picker
49913  * @param {Object} config Configuration options
49914  */
49915 Roo.form.DayPicker= function(config){
49916     Roo.form.DayPicker.superclass.constructor.call(this, config);
49917      
49918 };
49919
49920 Roo.extend(Roo.form.DayPicker, Roo.form.Field,  {
49921     /**
49922      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
49923      */
49924     focusClass : undefined,
49925     /**
49926      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
49927      */
49928     fieldClass: "x-form-field",
49929    
49930     /**
49931      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49932      * {tag: "input", type: "checkbox", autocomplete: "off"})
49933      */
49934     defaultAutoCreate : { tag: "input", type: 'hidden', autocomplete: "new-password"},
49935     
49936    
49937     actionMode : 'viewEl', 
49938     //
49939     // private
49940  
49941     inputType : 'hidden',
49942     
49943      
49944     inputElement: false, // real input element?
49945     basedOn: false, // ????
49946     
49947     isFormField: true, // not sure where this is needed!!!!
49948
49949     onResize : function(){
49950         Roo.form.Checkbox.superclass.onResize.apply(this, arguments);
49951         if(!this.boxLabel){
49952             this.el.alignTo(this.wrap, 'c-c');
49953         }
49954     },
49955
49956     initEvents : function(){
49957         Roo.form.Checkbox.superclass.initEvents.call(this);
49958         this.el.on("click", this.onClick,  this);
49959         this.el.on("change", this.onClick,  this);
49960     },
49961
49962
49963     getResizeEl : function(){
49964         return this.wrap;
49965     },
49966
49967     getPositionEl : function(){
49968         return this.wrap;
49969     },
49970
49971     
49972     // private
49973     onRender : function(ct, position){
49974         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
49975        
49976         this.wrap = this.el.wrap({cls: 'x-form-daypick-item '});
49977         
49978         var r1 = '<table><tr>';
49979         var r2 = '<tr class="x-form-daypick-icons">';
49980         for (var i=0; i < 7; i++) {
49981             r1+= '<td><div>' + Date.dayNames[i].substring(0,3) + '</div></td>';
49982             r2+= '<td><img class="x-menu-item-icon" src="' + Roo.BLANK_IMAGE_URL  +'"></td>';
49983         }
49984         
49985         var viewEl = this.wrap.createChild( r1 + '</tr>' + r2 + '</tr></table>');
49986         viewEl.select('img').on('click', this.onClick, this);
49987         this.viewEl = viewEl;   
49988         
49989         
49990         // this will not work on Chrome!!!
49991         this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
49992         this.el.on('propertychange', this.setFromHidden,  this);  //ie
49993         
49994         
49995           
49996
49997     },
49998
49999     // private
50000     initValue : Roo.emptyFn,
50001
50002     /**
50003      * Returns the checked state of the checkbox.
50004      * @return {Boolean} True if checked, else false
50005      */
50006     getValue : function(){
50007         return this.el.dom.value;
50008         
50009     },
50010
50011         // private
50012     onClick : function(e){ 
50013         //this.setChecked(!this.checked);
50014         Roo.get(e.target).toggleClass('x-menu-item-checked');
50015         this.refreshValue();
50016         //if(this.el.dom.checked != this.checked){
50017         //    this.setValue(this.el.dom.checked);
50018        // }
50019     },
50020     
50021     // private
50022     refreshValue : function()
50023     {
50024         var val = '';
50025         this.viewEl.select('img',true).each(function(e,i,n)  {
50026             val += e.is(".x-menu-item-checked") ? String(n) : '';
50027         });
50028         this.setValue(val, true);
50029     },
50030
50031     /**
50032      * Sets the checked state of the checkbox.
50033      * On is always based on a string comparison between inputValue and the param.
50034      * @param {Boolean/String} value - the value to set 
50035      * @param {Boolean/String} suppressEvent - whether to suppress the checkchange event.
50036      */
50037     setValue : function(v,suppressEvent){
50038         if (!this.el.dom) {
50039             return;
50040         }
50041         var old = this.el.dom.value ;
50042         this.el.dom.value = v;
50043         if (suppressEvent) {
50044             return ;
50045         }
50046          
50047         // update display..
50048         this.viewEl.select('img',true).each(function(e,i,n)  {
50049             
50050             var on = e.is(".x-menu-item-checked");
50051             var newv = v.indexOf(String(n)) > -1;
50052             if (on != newv) {
50053                 e.toggleClass('x-menu-item-checked');
50054             }
50055             
50056         });
50057         
50058         
50059         this.fireEvent('change', this, v, old);
50060         
50061         
50062     },
50063    
50064     // handle setting of hidden value by some other method!!?!?
50065     setFromHidden: function()
50066     {
50067         if(!this.el){
50068             return;
50069         }
50070         //console.log("SET FROM HIDDEN");
50071         //alert('setFrom hidden');
50072         this.setValue(this.el.dom.value);
50073     },
50074     
50075     onDestroy : function()
50076     {
50077         if(this.viewEl){
50078             Roo.get(this.viewEl).remove();
50079         }
50080          
50081         Roo.form.DayPicker.superclass.onDestroy.call(this);
50082     }
50083
50084 });/*
50085  * RooJS Library 1.1.1
50086  * Copyright(c) 2008-2011  Alan Knowles
50087  *
50088  * License - LGPL
50089  */
50090  
50091
50092 /**
50093  * @class Roo.form.ComboCheck
50094  * @extends Roo.form.ComboBox
50095  * A combobox for multiple select items.
50096  *
50097  * FIXME - could do with a reset button..
50098  * 
50099  * @constructor
50100  * Create a new ComboCheck
50101  * @param {Object} config Configuration options
50102  */
50103 Roo.form.ComboCheck = function(config){
50104     Roo.form.ComboCheck.superclass.constructor.call(this, config);
50105     // should verify some data...
50106     // like
50107     // hiddenName = required..
50108     // displayField = required
50109     // valudField == required
50110     var req= [ 'hiddenName', 'displayField', 'valueField' ];
50111     var _t = this;
50112     Roo.each(req, function(e) {
50113         if ((typeof(_t[e]) == 'undefined' ) || !_t[e].length) {
50114             throw "Roo.form.ComboCheck : missing value for: " + e;
50115         }
50116     });
50117     
50118     
50119 };
50120
50121 Roo.extend(Roo.form.ComboCheck, Roo.form.ComboBox, {
50122      
50123      
50124     editable : false,
50125      
50126     selectedClass: 'x-menu-item-checked', 
50127     
50128     // private
50129     onRender : function(ct, position){
50130         var _t = this;
50131         
50132         
50133         
50134         if(!this.tpl){
50135             var cls = 'x-combo-list';
50136
50137             
50138             this.tpl =  new Roo.Template({
50139                 html :  '<div class="'+cls+'-item x-menu-check-item">' +
50140                    '<img class="x-menu-item-icon" style="margin: 0px;" src="' + Roo.BLANK_IMAGE_URL + '">' + 
50141                    '<span>{' + this.displayField + '}</span>' +
50142                     '</div>' 
50143                 
50144             });
50145         }
50146  
50147         
50148         Roo.form.ComboCheck.superclass.onRender.call(this, ct, position);
50149         this.view.singleSelect = false;
50150         this.view.multiSelect = true;
50151         this.view.toggleSelect = true;
50152         this.pageTb.add(new Roo.Toolbar.Fill(), {
50153             
50154             text: 'Done',
50155             handler: function()
50156             {
50157                 _t.collapse();
50158             }
50159         });
50160     },
50161     
50162     onViewOver : function(e, t){
50163         // do nothing...
50164         return;
50165         
50166     },
50167     
50168     onViewClick : function(doFocus,index){
50169         return;
50170         
50171     },
50172     select: function () {
50173         //Roo.log("SELECT CALLED");
50174     },
50175      
50176     selectByValue : function(xv, scrollIntoView){
50177         var ar = this.getValueArray();
50178         var sels = [];
50179         
50180         Roo.each(ar, function(v) {
50181             if(v === undefined || v === null){
50182                 return;
50183             }
50184             var r = this.findRecord(this.valueField, v);
50185             if(r){
50186                 sels.push(this.store.indexOf(r))
50187                 
50188             }
50189         },this);
50190         this.view.select(sels);
50191         return false;
50192     },
50193     
50194     
50195     
50196     onSelect : function(record, index){
50197        // Roo.log("onselect Called");
50198        // this is only called by the clear button now..
50199         this.view.clearSelections();
50200         this.setValue('[]');
50201         if (this.value != this.valueBefore) {
50202             this.fireEvent('change', this, this.value, this.valueBefore);
50203             this.valueBefore = this.value;
50204         }
50205     },
50206     getValueArray : function()
50207     {
50208         var ar = [] ;
50209         
50210         try {
50211             //Roo.log(this.value);
50212             if (typeof(this.value) == 'undefined') {
50213                 return [];
50214             }
50215             var ar = Roo.decode(this.value);
50216             return  ar instanceof Array ? ar : []; //?? valid?
50217             
50218         } catch(e) {
50219             Roo.log(e + "\nRoo.form.ComboCheck:getValueArray  invalid data:" + this.getValue());
50220             return [];
50221         }
50222          
50223     },
50224     expand : function ()
50225     {
50226         
50227         Roo.form.ComboCheck.superclass.expand.call(this);
50228         this.valueBefore = typeof(this.value) == 'undefined' ? '' : this.value;
50229         //this.valueBefore = typeof(this.valueBefore) == 'undefined' ? '' : this.valueBefore;
50230         
50231
50232     },
50233     
50234     collapse : function(){
50235         Roo.form.ComboCheck.superclass.collapse.call(this);
50236         var sl = this.view.getSelectedIndexes();
50237         var st = this.store;
50238         var nv = [];
50239         var tv = [];
50240         var r;
50241         Roo.each(sl, function(i) {
50242             r = st.getAt(i);
50243             nv.push(r.get(this.valueField));
50244         },this);
50245         this.setValue(Roo.encode(nv));
50246         if (this.value != this.valueBefore) {
50247
50248             this.fireEvent('change', this, this.value, this.valueBefore);
50249             this.valueBefore = this.value;
50250         }
50251         
50252     },
50253     
50254     setValue : function(v){
50255         // Roo.log(v);
50256         this.value = v;
50257         
50258         var vals = this.getValueArray();
50259         var tv = [];
50260         Roo.each(vals, function(k) {
50261             var r = this.findRecord(this.valueField, k);
50262             if(r){
50263                 tv.push(r.data[this.displayField]);
50264             }else if(this.valueNotFoundText !== undefined){
50265                 tv.push( this.valueNotFoundText );
50266             }
50267         },this);
50268        // Roo.log(tv);
50269         
50270         Roo.form.ComboBox.superclass.setValue.call(this, tv.join(', '));
50271         this.hiddenField.value = v;
50272         this.value = v;
50273     }
50274     
50275 });/*
50276  * Based on:
50277  * Ext JS Library 1.1.1
50278  * Copyright(c) 2006-2007, Ext JS, LLC.
50279  *
50280  * Originally Released Under LGPL - original licence link has changed is not relivant.
50281  *
50282  * Fork - LGPL
50283  * <script type="text/javascript">
50284  */
50285  
50286 /**
50287  * @class Roo.form.Signature
50288  * @extends Roo.form.Field
50289  * Signature field.  
50290  * @constructor
50291  * 
50292  * @param {Object} config Configuration options
50293  */
50294
50295 Roo.form.Signature = function(config){
50296     Roo.form.Signature.superclass.constructor.call(this, config);
50297     
50298     this.addEvents({// not in used??
50299          /**
50300          * @event confirm
50301          * Fires when the 'confirm' icon is pressed (add a listener to enable add button)
50302              * @param {Roo.form.Signature} combo This combo box
50303              */
50304         'confirm' : true,
50305         /**
50306          * @event reset
50307          * Fires when the 'edit' icon is pressed (add a listener to enable add button)
50308              * @param {Roo.form.ComboBox} combo This combo box
50309              * @param {Roo.data.Record|false} record The data record returned from the underlying store (or false on nothing selected)
50310              */
50311         'reset' : true
50312     });
50313 };
50314
50315 Roo.extend(Roo.form.Signature, Roo.form.Field,  {
50316     /**
50317      * @cfg {Object} labels Label to use when rendering a form.
50318      * defaults to 
50319      * labels : { 
50320      *      clear : "Clear",
50321      *      confirm : "Confirm"
50322      *  }
50323      */
50324     labels : { 
50325         clear : "Clear",
50326         confirm : "Confirm"
50327     },
50328     /**
50329      * @cfg {Number} width The signature panel width (defaults to 300)
50330      */
50331     width: 300,
50332     /**
50333      * @cfg {Number} height The signature panel height (defaults to 100)
50334      */
50335     height : 100,
50336     /**
50337      * @cfg {Boolean} allowBlank False to validate that the value length > 0 (defaults to false)
50338      */
50339     allowBlank : false,
50340     
50341     //private
50342     // {Object} signPanel The signature SVG panel element (defaults to {})
50343     signPanel : {},
50344     // {Boolean} isMouseDown False to validate that the mouse down event (defaults to false)
50345     isMouseDown : false,
50346     // {Boolean} isConfirmed validate the signature is confirmed or not for submitting form (defaults to false)
50347     isConfirmed : false,
50348     // {String} signatureTmp SVG mapping string (defaults to empty string)
50349     signatureTmp : '',
50350     
50351     
50352     defaultAutoCreate : { // modified by initCompnoent..
50353         tag: "input",
50354         type:"hidden"
50355     },
50356
50357     // private
50358     onRender : function(ct, position){
50359         
50360         Roo.form.Signature.superclass.onRender.call(this, ct, position);
50361         
50362         this.wrap = this.el.wrap({
50363             cls:'x-form-signature-wrap', style : 'width: ' + this.width + 'px', cn:{cls:'x-form-signature'}
50364         });
50365         
50366         this.createToolbar(this);
50367         this.signPanel = this.wrap.createChild({
50368                 tag: 'div',
50369                 style: 'width: ' + this.width + 'px; height: ' + this.height + 'px; border: 0;'
50370             }, this.el
50371         );
50372             
50373         this.svgID = Roo.id();
50374         this.svgEl = this.signPanel.createChild({
50375               xmlns : 'http://www.w3.org/2000/svg',
50376               tag : 'svg',
50377               id : this.svgID + "-svg",
50378               width: this.width,
50379               height: this.height,
50380               viewBox: '0 0 '+this.width+' '+this.height,
50381               cn : [
50382                 {
50383                     tag: "rect",
50384                     id: this.svgID + "-svg-r",
50385                     width: this.width,
50386                     height: this.height,
50387                     fill: "#ffa"
50388                 },
50389                 {
50390                     tag: "line",
50391                     id: this.svgID + "-svg-l",
50392                     x1: "0", // start
50393                     y1: (this.height*0.8), // start set the line in 80% of height
50394                     x2: this.width, // end
50395                     y2: (this.height*0.8), // end set the line in 80% of height
50396                     'stroke': "#666",
50397                     'stroke-width': "1",
50398                     'stroke-dasharray': "3",
50399                     'shape-rendering': "crispEdges",
50400                     'pointer-events': "none"
50401                 },
50402                 {
50403                     tag: "path",
50404                     id: this.svgID + "-svg-p",
50405                     'stroke': "navy",
50406                     'stroke-width': "3",
50407                     'fill': "none",
50408                     'pointer-events': 'none'
50409                 }
50410               ]
50411         });
50412         this.createSVG();
50413         this.svgBox = this.svgEl.dom.getScreenCTM();
50414     },
50415     createSVG : function(){ 
50416         var svg = this.signPanel;
50417         var r = svg.select('#'+ this.svgID + '-svg-r', true).first().dom;
50418         var t = this;
50419
50420         r.addEventListener('mousedown', function(e) { return t.down(e); }, false);
50421         r.addEventListener('mousemove', function(e) { return t.move(e); }, false);
50422         r.addEventListener('mouseup', function(e) { return t.up(e); }, false);
50423         r.addEventListener('mouseout', function(e) { return t.up(e); }, false);
50424         r.addEventListener('touchstart', function(e) { return t.down(e); }, false);
50425         r.addEventListener('touchmove', function(e) { return t.move(e); }, false);
50426         r.addEventListener('touchend', function(e) { return t.up(e); }, false);
50427         
50428     },
50429     isTouchEvent : function(e){
50430         return e.type.match(/^touch/);
50431     },
50432     getCoords : function (e) {
50433         var pt    = this.svgEl.dom.createSVGPoint();
50434         pt.x = e.clientX; 
50435         pt.y = e.clientY;
50436         if (this.isTouchEvent(e)) {
50437             pt.x =  e.targetTouches[0].clientX;
50438             pt.y = e.targetTouches[0].clientY;
50439         }
50440         var a = this.svgEl.dom.getScreenCTM();
50441         var b = a.inverse();
50442         var mx = pt.matrixTransform(b);
50443         return mx.x + ',' + mx.y;
50444     },
50445     //mouse event headler 
50446     down : function (e) {
50447         this.signatureTmp += 'M' + this.getCoords(e) + ' ';
50448         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr('d', this.signatureTmp);
50449         
50450         this.isMouseDown = true;
50451         
50452         e.preventDefault();
50453     },
50454     move : function (e) {
50455         if (this.isMouseDown) {
50456             this.signatureTmp += 'L' + this.getCoords(e) + ' ';
50457             this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', this.signatureTmp);
50458         }
50459         
50460         e.preventDefault();
50461     },
50462     up : function (e) {
50463         this.isMouseDown = false;
50464         var sp = this.signatureTmp.split(' ');
50465         
50466         if(sp.length > 1){
50467             if(!sp[sp.length-2].match(/^L/)){
50468                 sp.pop();
50469                 sp.pop();
50470                 sp.push("");
50471                 this.signatureTmp = sp.join(" ");
50472             }
50473         }
50474         if(this.getValue() != this.signatureTmp){
50475             this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50476             this.isConfirmed = false;
50477         }
50478         e.preventDefault();
50479     },
50480     
50481     /**
50482      * Protected method that will not generally be called directly. It
50483      * is called when the editor creates its toolbar. Override this method if you need to
50484      * add custom toolbar buttons.
50485      * @param {HtmlEditor} editor
50486      */
50487     createToolbar : function(editor){
50488          function btn(id, toggle, handler){
50489             var xid = fid + '-'+ id ;
50490             return {
50491                 id : xid,
50492                 cmd : id,
50493                 cls : 'x-btn-icon x-edit-'+id,
50494                 enableToggle:toggle !== false,
50495                 scope: editor, // was editor...
50496                 handler:handler||editor.relayBtnCmd,
50497                 clickEvent:'mousedown',
50498                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
50499                 tabIndex:-1
50500             };
50501         }
50502         
50503         
50504         var tb = new Roo.Toolbar(editor.wrap.dom.firstChild);
50505         this.tb = tb;
50506         this.tb.add(
50507            {
50508                 cls : ' x-signature-btn x-signature-'+id,
50509                 scope: editor, // was editor...
50510                 handler: this.reset,
50511                 clickEvent:'mousedown',
50512                 text: this.labels.clear
50513             },
50514             {
50515                  xtype : 'Fill',
50516                  xns: Roo.Toolbar
50517             }, 
50518             {
50519                 cls : '  x-signature-btn x-signature-'+id,
50520                 scope: editor, // was editor...
50521                 handler: this.confirmHandler,
50522                 clickEvent:'mousedown',
50523                 text: this.labels.confirm
50524             }
50525         );
50526     
50527     },
50528     //public
50529     /**
50530      * when user is clicked confirm then show this image.....
50531      * 
50532      * @return {String} Image Data URI
50533      */
50534     getImageDataURI : function(){
50535         var svg = this.svgEl.dom.parentNode.innerHTML;
50536         var src = 'data:image/svg+xml;base64,'+window.btoa(svg);
50537         return src; 
50538     },
50539     /**
50540      * 
50541      * @return {Boolean} this.isConfirmed
50542      */
50543     getConfirmed : function(){
50544         return this.isConfirmed;
50545     },
50546     /**
50547      * 
50548      * @return {Number} this.width
50549      */
50550     getWidth : function(){
50551         return this.width;
50552     },
50553     /**
50554      * 
50555      * @return {Number} this.height
50556      */
50557     getHeight : function(){
50558         return this.height;
50559     },
50560     // private
50561     getSignature : function(){
50562         return this.signatureTmp;
50563     },
50564     // private
50565     reset : function(){
50566         this.signatureTmp = '';
50567         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50568         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', '');
50569         this.isConfirmed = false;
50570         Roo.form.Signature.superclass.reset.call(this);
50571     },
50572     setSignature : function(s){
50573         this.signatureTmp = s;
50574         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50575         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', s);
50576         this.setValue(s);
50577         this.isConfirmed = false;
50578         Roo.form.Signature.superclass.reset.call(this);
50579     }, 
50580     test : function(){
50581 //        Roo.log(this.signPanel.dom.contentWindow.up())
50582     },
50583     //private
50584     setConfirmed : function(){
50585         
50586         
50587         
50588 //        Roo.log(Roo.get(this.signPanel.dom.contentWindow.r).attr('fill', '#cfc'));
50589     },
50590     // private
50591     confirmHandler : function(){
50592         if(!this.getSignature()){
50593             return;
50594         }
50595         
50596         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#cfc');
50597         this.setValue(this.getSignature());
50598         this.isConfirmed = true;
50599         
50600         this.fireEvent('confirm', this);
50601     },
50602     // private
50603     // Subclasses should provide the validation implementation by overriding this
50604     validateValue : function(value){
50605         if(this.allowBlank){
50606             return true;
50607         }
50608         
50609         if(this.isConfirmed){
50610             return true;
50611         }
50612         return false;
50613     }
50614 });/*
50615  * Based on:
50616  * Ext JS Library 1.1.1
50617  * Copyright(c) 2006-2007, Ext JS, LLC.
50618  *
50619  * Originally Released Under LGPL - original licence link has changed is not relivant.
50620  *
50621  * Fork - LGPL
50622  * <script type="text/javascript">
50623  */
50624  
50625
50626 /**
50627  * @class Roo.form.ComboBox
50628  * @extends Roo.form.TriggerField
50629  * A combobox control with support for autocomplete, remote-loading, paging and many other features.
50630  * @constructor
50631  * Create a new ComboBox.
50632  * @param {Object} config Configuration options
50633  */
50634 Roo.form.Select = function(config){
50635     Roo.form.Select.superclass.constructor.call(this, config);
50636      
50637 };
50638
50639 Roo.extend(Roo.form.Select , Roo.form.ComboBox, {
50640     /**
50641      * @cfg {String/HTMLElement/Element} transform The id, DOM node or element of an existing select to convert to a ComboBox
50642      */
50643     /**
50644      * @cfg {Boolean} lazyRender True to prevent the ComboBox from rendering until requested (should always be used when
50645      * rendering into an Roo.Editor, defaults to false)
50646      */
50647     /**
50648      * @cfg {Boolean/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to:
50649      * {tag: "input", type: "text", size: "24", autocomplete: "off"})
50650      */
50651     /**
50652      * @cfg {Roo.data.Store} store The data store to which this combo is bound (defaults to undefined)
50653      */
50654     /**
50655      * @cfg {String} title If supplied, a header element is created containing this text and added into the top of
50656      * the dropdown list (defaults to undefined, with no header element)
50657      */
50658
50659      /**
50660      * @cfg {String/Roo.Template} tpl The template to use to render the output
50661      */
50662      
50663     // private
50664     defaultAutoCreate : {tag: "select"  },
50665     /**
50666      * @cfg {Number} listWidth The width in pixels of the dropdown list (defaults to the width of the ComboBox field)
50667      */
50668     listWidth: undefined,
50669     /**
50670      * @cfg {String} displayField The underlying data field name to bind to this CombBox (defaults to undefined if
50671      * mode = 'remote' or 'text' if mode = 'local')
50672      */
50673     displayField: undefined,
50674     /**
50675      * @cfg {String} valueField The underlying data value name to bind to this CombBox (defaults to undefined if
50676      * mode = 'remote' or 'value' if mode = 'local'). 
50677      * Note: use of a valueField requires the user make a selection
50678      * in order for a value to be mapped.
50679      */
50680     valueField: undefined,
50681     
50682     
50683     /**
50684      * @cfg {String} hiddenName If specified, a hidden form field with this name is dynamically generated to store the
50685      * field's data value (defaults to the underlying DOM element's name)
50686      */
50687     hiddenName: undefined,
50688     /**
50689      * @cfg {String} listClass CSS class to apply to the dropdown list element (defaults to '')
50690      */
50691     listClass: '',
50692     /**
50693      * @cfg {String} selectedClass CSS class to apply to the selected item in the dropdown list (defaults to 'x-combo-selected')
50694      */
50695     selectedClass: 'x-combo-selected',
50696     /**
50697      * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
50698      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-arrow-trigger'
50699      * which displays a downward arrow icon).
50700      */
50701     triggerClass : 'x-form-arrow-trigger',
50702     /**
50703      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
50704      */
50705     shadow:'sides',
50706     /**
50707      * @cfg {String} listAlign A valid anchor position value. See {@link Roo.Element#alignTo} for details on supported
50708      * anchor positions (defaults to 'tl-bl')
50709      */
50710     listAlign: 'tl-bl?',
50711     /**
50712      * @cfg {Number} maxHeight The maximum height in pixels of the dropdown list before scrollbars are shown (defaults to 300)
50713      */
50714     maxHeight: 300,
50715     /**
50716      * @cfg {String} triggerAction The action to execute when the trigger field is activated.  Use 'all' to run the
50717      * query specified by the allQuery config option (defaults to 'query')
50718      */
50719     triggerAction: 'query',
50720     /**
50721      * @cfg {Number} minChars The minimum number of characters the user must type before autocomplete and typeahead activate
50722      * (defaults to 4, does not apply if editable = false)
50723      */
50724     minChars : 4,
50725     /**
50726      * @cfg {Boolean} typeAhead True to populate and autoselect the remainder of the text being typed after a configurable
50727      * delay (typeAheadDelay) if it matches a known value (defaults to false)
50728      */
50729     typeAhead: false,
50730     /**
50731      * @cfg {Number} queryDelay The length of time in milliseconds to delay between the start of typing and sending the
50732      * query to filter the dropdown list (defaults to 500 if mode = 'remote' or 10 if mode = 'local')
50733      */
50734     queryDelay: 500,
50735     /**
50736      * @cfg {Number} pageSize If greater than 0, a paging toolbar is displayed in the footer of the dropdown list and the
50737      * filter queries will execute with page start and limit parameters.  Only applies when mode = 'remote' (defaults to 0)
50738      */
50739     pageSize: 0,
50740     /**
50741      * @cfg {Boolean} selectOnFocus True to select any existing text in the field immediately on focus.  Only applies
50742      * when editable = true (defaults to false)
50743      */
50744     selectOnFocus:false,
50745     /**
50746      * @cfg {String} queryParam Name of the query as it will be passed on the querystring (defaults to 'query')
50747      */
50748     queryParam: 'query',
50749     /**
50750      * @cfg {String} loadingText The text to display in the dropdown list while data is loading.  Only applies
50751      * when mode = 'remote' (defaults to 'Loading...')
50752      */
50753     loadingText: 'Loading...',
50754     /**
50755      * @cfg {Boolean} resizable True to add a resize handle to the bottom of the dropdown list (defaults to false)
50756      */
50757     resizable: false,
50758     /**
50759      * @cfg {Number} handleHeight The height in pixels of the dropdown list resize handle if resizable = true (defaults to 8)
50760      */
50761     handleHeight : 8,
50762     /**
50763      * @cfg {Boolean} editable False to prevent the user from typing text directly into the field, just like a
50764      * traditional select (defaults to true)
50765      */
50766     editable: true,
50767     /**
50768      * @cfg {String} allQuery The text query to send to the server to return all records for the list with no filtering (defaults to '')
50769      */
50770     allQuery: '',
50771     /**
50772      * @cfg {String} mode Set to 'local' if the ComboBox loads local data (defaults to 'remote' which loads from the server)
50773      */
50774     mode: 'remote',
50775     /**
50776      * @cfg {Number} minListWidth The minimum width of the dropdown list in pixels (defaults to 70, will be ignored if
50777      * listWidth has a higher value)
50778      */
50779     minListWidth : 70,
50780     /**
50781      * @cfg {Boolean} forceSelection True to restrict the selected value to one of the values in the list, false to
50782      * allow the user to set arbitrary text into the field (defaults to false)
50783      */
50784     forceSelection:false,
50785     /**
50786      * @cfg {Number} typeAheadDelay The length of time in milliseconds to wait until the typeahead text is displayed
50787      * if typeAhead = true (defaults to 250)
50788      */
50789     typeAheadDelay : 250,
50790     /**
50791      * @cfg {String} valueNotFoundText When using a name/value combo, if the value passed to setValue is not found in
50792      * the store, valueNotFoundText will be displayed as the field text if defined (defaults to undefined)
50793      */
50794     valueNotFoundText : undefined,
50795     
50796     /**
50797      * @cfg {String} defaultValue The value displayed after loading the store.
50798      */
50799     defaultValue: '',
50800     
50801     /**
50802      * @cfg {Boolean} blockFocus Prevents all focus calls, so it can work with things like HTML edtor bar
50803      */
50804     blockFocus : false,
50805     
50806     /**
50807      * @cfg {Boolean} disableClear Disable showing of clear button.
50808      */
50809     disableClear : false,
50810     /**
50811      * @cfg {Boolean} alwaysQuery  Disable caching of results, and always send query
50812      */
50813     alwaysQuery : false,
50814     
50815     //private
50816     addicon : false,
50817     editicon: false,
50818     
50819     // element that contains real text value.. (when hidden is used..)
50820      
50821     // private
50822     onRender : function(ct, position){
50823         Roo.form.Field.prototype.onRender.call(this, ct, position);
50824         
50825         if(this.store){
50826             this.store.on('beforeload', this.onBeforeLoad, this);
50827             this.store.on('load', this.onLoad, this);
50828             this.store.on('loadexception', this.onLoadException, this);
50829             this.store.load({});
50830         }
50831         
50832         
50833         
50834     },
50835
50836     // private
50837     initEvents : function(){
50838         //Roo.form.ComboBox.superclass.initEvents.call(this);
50839  
50840     },
50841
50842     onDestroy : function(){
50843        
50844         if(this.store){
50845             this.store.un('beforeload', this.onBeforeLoad, this);
50846             this.store.un('load', this.onLoad, this);
50847             this.store.un('loadexception', this.onLoadException, this);
50848         }
50849         //Roo.form.ComboBox.superclass.onDestroy.call(this);
50850     },
50851
50852     // private
50853     fireKey : function(e){
50854         if(e.isNavKeyPress() && !this.list.isVisible()){
50855             this.fireEvent("specialkey", this, e);
50856         }
50857     },
50858
50859     // private
50860     onResize: function(w, h){
50861         
50862         return; 
50863     
50864         
50865     },
50866
50867     /**
50868      * Allow or prevent the user from directly editing the field text.  If false is passed,
50869      * the user will only be able to select from the items defined in the dropdown list.  This method
50870      * is the runtime equivalent of setting the 'editable' config option at config time.
50871      * @param {Boolean} value True to allow the user to directly edit the field text
50872      */
50873     setEditable : function(value){
50874          
50875     },
50876
50877     // private
50878     onBeforeLoad : function(){
50879         
50880         Roo.log("Select before load");
50881         return;
50882     
50883         this.innerList.update(this.loadingText ?
50884                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
50885         //this.restrictHeight();
50886         this.selectedIndex = -1;
50887     },
50888
50889     // private
50890     onLoad : function(){
50891
50892     
50893         var dom = this.el.dom;
50894         dom.innerHTML = '';
50895          var od = dom.ownerDocument;
50896          
50897         if (this.emptyText) {
50898             var op = od.createElement('option');
50899             op.setAttribute('value', '');
50900             op.innerHTML = String.format('{0}', this.emptyText);
50901             dom.appendChild(op);
50902         }
50903         if(this.store.getCount() > 0){
50904            
50905             var vf = this.valueField;
50906             var df = this.displayField;
50907             this.store.data.each(function(r) {
50908                 // which colmsn to use... testing - cdoe / title..
50909                 var op = od.createElement('option');
50910                 op.setAttribute('value', r.data[vf]);
50911                 op.innerHTML = String.format('{0}', r.data[df]);
50912                 dom.appendChild(op);
50913             });
50914             if (typeof(this.defaultValue != 'undefined')) {
50915                 this.setValue(this.defaultValue);
50916             }
50917             
50918              
50919         }else{
50920             //this.onEmptyResults();
50921         }
50922         //this.el.focus();
50923     },
50924     // private
50925     onLoadException : function()
50926     {
50927         dom.innerHTML = '';
50928             
50929         Roo.log("Select on load exception");
50930         return;
50931     
50932         this.collapse();
50933         Roo.log(this.store.reader.jsonData);
50934         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
50935             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
50936         }
50937         
50938         
50939     },
50940     // private
50941     onTypeAhead : function(){
50942          
50943     },
50944
50945     // private
50946     onSelect : function(record, index){
50947         Roo.log('on select?');
50948         return;
50949         if(this.fireEvent('beforeselect', this, record, index) !== false){
50950             this.setFromData(index > -1 ? record.data : false);
50951             this.collapse();
50952             this.fireEvent('select', this, record, index);
50953         }
50954     },
50955
50956     /**
50957      * Returns the currently selected field value or empty string if no value is set.
50958      * @return {String} value The selected value
50959      */
50960     getValue : function(){
50961         var dom = this.el.dom;
50962         this.value = dom.options[dom.selectedIndex].value;
50963         return this.value;
50964         
50965     },
50966
50967     /**
50968      * Clears any text/value currently set in the field
50969      */
50970     clearValue : function(){
50971         this.value = '';
50972         this.el.dom.selectedIndex = this.emptyText ? 0 : -1;
50973         
50974     },
50975
50976     /**
50977      * Sets the specified value into the field.  If the value finds a match, the corresponding record text
50978      * will be displayed in the field.  If the value does not match the data value of an existing item,
50979      * and the valueNotFoundText config option is defined, it will be displayed as the default field text.
50980      * Otherwise the field will be blank (although the value will still be set).
50981      * @param {String} value The value to match
50982      */
50983     setValue : function(v){
50984         var d = this.el.dom;
50985         for (var i =0; i < d.options.length;i++) {
50986             if (v == d.options[i].value) {
50987                 d.selectedIndex = i;
50988                 this.value = v;
50989                 return;
50990             }
50991         }
50992         this.clearValue();
50993     },
50994     /**
50995      * @property {Object} the last set data for the element
50996      */
50997     
50998     lastData : false,
50999     /**
51000      * Sets the value of the field based on a object which is related to the record format for the store.
51001      * @param {Object} value the value to set as. or false on reset?
51002      */
51003     setFromData : function(o){
51004         Roo.log('setfrom data?');
51005          
51006         
51007         
51008     },
51009     // private
51010     reset : function(){
51011         this.clearValue();
51012     },
51013     // private
51014     findRecord : function(prop, value){
51015         
51016         return false;
51017     
51018         var record;
51019         if(this.store.getCount() > 0){
51020             this.store.each(function(r){
51021                 if(r.data[prop] == value){
51022                     record = r;
51023                     return false;
51024                 }
51025                 return true;
51026             });
51027         }
51028         return record;
51029     },
51030     
51031     getName: function()
51032     {
51033         // returns hidden if it's set..
51034         if (!this.rendered) {return ''};
51035         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
51036         
51037     },
51038      
51039
51040     
51041
51042     // private
51043     onEmptyResults : function(){
51044         Roo.log('empty results');
51045         //this.collapse();
51046     },
51047
51048     /**
51049      * Returns true if the dropdown list is expanded, else false.
51050      */
51051     isExpanded : function(){
51052         return false;
51053     },
51054
51055     /**
51056      * Select an item in the dropdown list by its data value. This function does NOT cause the select event to fire.
51057      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
51058      * @param {String} value The data value of the item to select
51059      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
51060      * selected item if it is not currently in view (defaults to true)
51061      * @return {Boolean} True if the value matched an item in the list, else false
51062      */
51063     selectByValue : function(v, scrollIntoView){
51064         Roo.log('select By Value');
51065         return false;
51066     
51067         if(v !== undefined && v !== null){
51068             var r = this.findRecord(this.valueField || this.displayField, v);
51069             if(r){
51070                 this.select(this.store.indexOf(r), scrollIntoView);
51071                 return true;
51072             }
51073         }
51074         return false;
51075     },
51076
51077     /**
51078      * Select an item in the dropdown list by its numeric index in the list. This function does NOT cause the select event to fire.
51079      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
51080      * @param {Number} index The zero-based index of the list item to select
51081      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
51082      * selected item if it is not currently in view (defaults to true)
51083      */
51084     select : function(index, scrollIntoView){
51085         Roo.log('select ');
51086         return  ;
51087         
51088         this.selectedIndex = index;
51089         this.view.select(index);
51090         if(scrollIntoView !== false){
51091             var el = this.view.getNode(index);
51092             if(el){
51093                 this.innerList.scrollChildIntoView(el, false);
51094             }
51095         }
51096     },
51097
51098       
51099
51100     // private
51101     validateBlur : function(){
51102         
51103         return;
51104         
51105     },
51106
51107     // private
51108     initQuery : function(){
51109         this.doQuery(this.getRawValue());
51110     },
51111
51112     // private
51113     doForce : function(){
51114         if(this.el.dom.value.length > 0){
51115             this.el.dom.value =
51116                 this.lastSelectionText === undefined ? '' : this.lastSelectionText;
51117              
51118         }
51119     },
51120
51121     /**
51122      * Execute a query to filter the dropdown list.  Fires the beforequery event prior to performing the
51123      * query allowing the query action to be canceled if needed.
51124      * @param {String} query The SQL query to execute
51125      * @param {Boolean} forceAll True to force the query to execute even if there are currently fewer characters
51126      * in the field than the minimum specified by the minChars config option.  It also clears any filter previously
51127      * saved in the current store (defaults to false)
51128      */
51129     doQuery : function(q, forceAll){
51130         
51131         Roo.log('doQuery?');
51132         if(q === undefined || q === null){
51133             q = '';
51134         }
51135         var qe = {
51136             query: q,
51137             forceAll: forceAll,
51138             combo: this,
51139             cancel:false
51140         };
51141         if(this.fireEvent('beforequery', qe)===false || qe.cancel){
51142             return false;
51143         }
51144         q = qe.query;
51145         forceAll = qe.forceAll;
51146         if(forceAll === true || (q.length >= this.minChars)){
51147             if(this.lastQuery != q || this.alwaysQuery){
51148                 this.lastQuery = q;
51149                 if(this.mode == 'local'){
51150                     this.selectedIndex = -1;
51151                     if(forceAll){
51152                         this.store.clearFilter();
51153                     }else{
51154                         this.store.filter(this.displayField, q);
51155                     }
51156                     this.onLoad();
51157                 }else{
51158                     this.store.baseParams[this.queryParam] = q;
51159                     this.store.load({
51160                         params: this.getParams(q)
51161                     });
51162                     this.expand();
51163                 }
51164             }else{
51165                 this.selectedIndex = -1;
51166                 this.onLoad();   
51167             }
51168         }
51169     },
51170
51171     // private
51172     getParams : function(q){
51173         var p = {};
51174         //p[this.queryParam] = q;
51175         if(this.pageSize){
51176             p.start = 0;
51177             p.limit = this.pageSize;
51178         }
51179         return p;
51180     },
51181
51182     /**
51183      * Hides the dropdown list if it is currently expanded. Fires the 'collapse' event on completion.
51184      */
51185     collapse : function(){
51186         
51187     },
51188
51189     // private
51190     collapseIf : function(e){
51191         
51192     },
51193
51194     /**
51195      * Expands the dropdown list if it is currently hidden. Fires the 'expand' event on completion.
51196      */
51197     expand : function(){
51198         
51199     } ,
51200
51201     // private
51202      
51203
51204     /** 
51205     * @cfg {Boolean} grow 
51206     * @hide 
51207     */
51208     /** 
51209     * @cfg {Number} growMin 
51210     * @hide 
51211     */
51212     /** 
51213     * @cfg {Number} growMax 
51214     * @hide 
51215     */
51216     /**
51217      * @hide
51218      * @method autoSize
51219      */
51220     
51221     setWidth : function()
51222     {
51223         
51224     },
51225     getResizeEl : function(){
51226         return this.el;
51227     }
51228 });//<script type="text/javasscript">
51229  
51230
51231 /**
51232  * @class Roo.DDView
51233  * A DnD enabled version of Roo.View.
51234  * @param {Element/String} container The Element in which to create the View.
51235  * @param {String} tpl The template string used to create the markup for each element of the View
51236  * @param {Object} config The configuration properties. These include all the config options of
51237  * {@link Roo.View} plus some specific to this class.<br>
51238  * <p>
51239  * Drag/drop is implemented by adding {@link Roo.data.Record}s to the target DDView. If copying is
51240  * not being performed, the original {@link Roo.data.Record} is removed from the source DDView.<br>
51241  * <p>
51242  * The following extra CSS rules are needed to provide insertion point highlighting:<pre><code>
51243 .x-view-drag-insert-above {
51244         border-top:1px dotted #3366cc;
51245 }
51246 .x-view-drag-insert-below {
51247         border-bottom:1px dotted #3366cc;
51248 }
51249 </code></pre>
51250  * 
51251  */
51252  
51253 Roo.DDView = function(container, tpl, config) {
51254     Roo.DDView.superclass.constructor.apply(this, arguments);
51255     this.getEl().setStyle("outline", "0px none");
51256     this.getEl().unselectable();
51257     if (this.dragGroup) {
51258                 this.setDraggable(this.dragGroup.split(","));
51259     }
51260     if (this.dropGroup) {
51261                 this.setDroppable(this.dropGroup.split(","));
51262     }
51263     if (this.deletable) {
51264         this.setDeletable();
51265     }
51266     this.isDirtyFlag = false;
51267         this.addEvents({
51268                 "drop" : true
51269         });
51270 };
51271
51272 Roo.extend(Roo.DDView, Roo.View, {
51273 /**     @cfg {String/Array} dragGroup The ddgroup name(s) for the View's DragZone. */
51274 /**     @cfg {String/Array} dropGroup The ddgroup name(s) for the View's DropZone. */
51275 /**     @cfg {Boolean} copy Causes drag operations to copy nodes rather than move. */
51276 /**     @cfg {Boolean} allowCopy Causes ctrl/drag operations to copy nodes rather than move. */
51277
51278         isFormField: true,
51279
51280         reset: Roo.emptyFn,
51281         
51282         clearInvalid: Roo.form.Field.prototype.clearInvalid,
51283
51284         validate: function() {
51285                 return true;
51286         },
51287         
51288         destroy: function() {
51289                 this.purgeListeners();
51290                 this.getEl.removeAllListeners();
51291                 this.getEl().remove();
51292                 if (this.dragZone) {
51293                         if (this.dragZone.destroy) {
51294                                 this.dragZone.destroy();
51295                         }
51296                 }
51297                 if (this.dropZone) {
51298                         if (this.dropZone.destroy) {
51299                                 this.dropZone.destroy();
51300                         }
51301                 }
51302         },
51303
51304 /**     Allows this class to be an Roo.form.Field so it can be found using {@link Roo.form.BasicForm#findField}. */
51305         getName: function() {
51306                 return this.name;
51307         },
51308
51309 /**     Loads the View from a JSON string representing the Records to put into the Store. */
51310         setValue: function(v) {
51311                 if (!this.store) {
51312                         throw "DDView.setValue(). DDView must be constructed with a valid Store";
51313                 }
51314                 var data = {};
51315                 data[this.store.reader.meta.root] = v ? [].concat(v) : [];
51316                 this.store.proxy = new Roo.data.MemoryProxy(data);
51317                 this.store.load();
51318         },
51319
51320 /**     @return {String} a parenthesised list of the ids of the Records in the View. */
51321         getValue: function() {
51322                 var result = '(';
51323                 this.store.each(function(rec) {
51324                         result += rec.id + ',';
51325                 });
51326                 return result.substr(0, result.length - 1) + ')';
51327         },
51328         
51329         getIds: function() {
51330                 var i = 0, result = new Array(this.store.getCount());
51331                 this.store.each(function(rec) {
51332                         result[i++] = rec.id;
51333                 });
51334                 return result;
51335         },
51336         
51337         isDirty: function() {
51338                 return this.isDirtyFlag;
51339         },
51340
51341 /**
51342  *      Part of the Roo.dd.DropZone interface. If no target node is found, the
51343  *      whole Element becomes the target, and this causes the drop gesture to append.
51344  */
51345     getTargetFromEvent : function(e) {
51346                 var target = e.getTarget();
51347                 while ((target !== null) && (target.parentNode != this.el.dom)) {
51348                 target = target.parentNode;
51349                 }
51350                 if (!target) {
51351                         target = this.el.dom.lastChild || this.el.dom;
51352                 }
51353                 return target;
51354     },
51355
51356 /**
51357  *      Create the drag data which consists of an object which has the property "ddel" as
51358  *      the drag proxy element. 
51359  */
51360     getDragData : function(e) {
51361         var target = this.findItemFromChild(e.getTarget());
51362                 if(target) {
51363                         this.handleSelection(e);
51364                         var selNodes = this.getSelectedNodes();
51365             var dragData = {
51366                 source: this,
51367                 copy: this.copy || (this.allowCopy && e.ctrlKey),
51368                 nodes: selNodes,
51369                 records: []
51370                         };
51371                         var selectedIndices = this.getSelectedIndexes();
51372                         for (var i = 0; i < selectedIndices.length; i++) {
51373                                 dragData.records.push(this.store.getAt(selectedIndices[i]));
51374                         }
51375                         if (selNodes.length == 1) {
51376                                 dragData.ddel = target.cloneNode(true); // the div element
51377                         } else {
51378                                 var div = document.createElement('div'); // create the multi element drag "ghost"
51379                                 div.className = 'multi-proxy';
51380                                 for (var i = 0, len = selNodes.length; i < len; i++) {
51381                                         div.appendChild(selNodes[i].cloneNode(true));
51382                                 }
51383                                 dragData.ddel = div;
51384                         }
51385             //console.log(dragData)
51386             //console.log(dragData.ddel.innerHTML)
51387                         return dragData;
51388                 }
51389         //console.log('nodragData')
51390                 return false;
51391     },
51392     
51393 /**     Specify to which ddGroup items in this DDView may be dragged. */
51394     setDraggable: function(ddGroup) {
51395         if (ddGroup instanceof Array) {
51396                 Roo.each(ddGroup, this.setDraggable, this);
51397                 return;
51398         }
51399         if (this.dragZone) {
51400                 this.dragZone.addToGroup(ddGroup);
51401         } else {
51402                         this.dragZone = new Roo.dd.DragZone(this.getEl(), {
51403                                 containerScroll: true,
51404                                 ddGroup: ddGroup 
51405
51406                         });
51407 //                      Draggability implies selection. DragZone's mousedown selects the element.
51408                         if (!this.multiSelect) { this.singleSelect = true; }
51409
51410 //                      Wire the DragZone's handlers up to methods in *this*
51411                         this.dragZone.getDragData = this.getDragData.createDelegate(this);
51412                 }
51413     },
51414
51415 /**     Specify from which ddGroup this DDView accepts drops. */
51416     setDroppable: function(ddGroup) {
51417         if (ddGroup instanceof Array) {
51418                 Roo.each(ddGroup, this.setDroppable, this);
51419                 return;
51420         }
51421         if (this.dropZone) {
51422                 this.dropZone.addToGroup(ddGroup);
51423         } else {
51424                         this.dropZone = new Roo.dd.DropZone(this.getEl(), {
51425                                 containerScroll: true,
51426                                 ddGroup: ddGroup
51427                         });
51428
51429 //                      Wire the DropZone's handlers up to methods in *this*
51430                         this.dropZone.getTargetFromEvent = this.getTargetFromEvent.createDelegate(this);
51431                         this.dropZone.onNodeEnter = this.onNodeEnter.createDelegate(this);
51432                         this.dropZone.onNodeOver = this.onNodeOver.createDelegate(this);
51433                         this.dropZone.onNodeOut = this.onNodeOut.createDelegate(this);
51434                         this.dropZone.onNodeDrop = this.onNodeDrop.createDelegate(this);
51435                 }
51436     },
51437
51438 /**     Decide whether to drop above or below a View node. */
51439     getDropPoint : function(e, n, dd){
51440         if (n == this.el.dom) { return "above"; }
51441                 var t = Roo.lib.Dom.getY(n), b = t + n.offsetHeight;
51442                 var c = t + (b - t) / 2;
51443                 var y = Roo.lib.Event.getPageY(e);
51444                 if(y <= c) {
51445                         return "above";
51446                 }else{
51447                         return "below";
51448                 }
51449     },
51450
51451     onNodeEnter : function(n, dd, e, data){
51452                 return false;
51453     },
51454     
51455     onNodeOver : function(n, dd, e, data){
51456                 var pt = this.getDropPoint(e, n, dd);
51457                 // set the insert point style on the target node
51458                 var dragElClass = this.dropNotAllowed;
51459                 if (pt) {
51460                         var targetElClass;
51461                         if (pt == "above"){
51462                                 dragElClass = n.previousSibling ? "x-tree-drop-ok-between" : "x-tree-drop-ok-above";
51463                                 targetElClass = "x-view-drag-insert-above";
51464                         } else {
51465                                 dragElClass = n.nextSibling ? "x-tree-drop-ok-between" : "x-tree-drop-ok-below";
51466                                 targetElClass = "x-view-drag-insert-below";
51467                         }
51468                         if (this.lastInsertClass != targetElClass){
51469                                 Roo.fly(n).replaceClass(this.lastInsertClass, targetElClass);
51470                                 this.lastInsertClass = targetElClass;
51471                         }
51472                 }
51473                 return dragElClass;
51474         },
51475
51476     onNodeOut : function(n, dd, e, data){
51477                 this.removeDropIndicators(n);
51478     },
51479
51480     onNodeDrop : function(n, dd, e, data){
51481         if (this.fireEvent("drop", this, n, dd, e, data) === false) {
51482                 return false;
51483         }
51484         var pt = this.getDropPoint(e, n, dd);
51485                 var insertAt = (n == this.el.dom) ? this.nodes.length : n.nodeIndex;
51486                 if (pt == "below") { insertAt++; }
51487                 for (var i = 0; i < data.records.length; i++) {
51488                         var r = data.records[i];
51489                         var dup = this.store.getById(r.id);
51490                         if (dup && (dd != this.dragZone)) {
51491                                 Roo.fly(this.getNode(this.store.indexOf(dup))).frame("red", 1);
51492                         } else {
51493                                 if (data.copy) {
51494                                         this.store.insert(insertAt++, r.copy());
51495                                 } else {
51496                                         data.source.isDirtyFlag = true;
51497                                         r.store.remove(r);
51498                                         this.store.insert(insertAt++, r);
51499                                 }
51500                                 this.isDirtyFlag = true;
51501                         }
51502                 }
51503                 this.dragZone.cachedTarget = null;
51504                 return true;
51505     },
51506
51507     removeDropIndicators : function(n){
51508                 if(n){
51509                         Roo.fly(n).removeClass([
51510                                 "x-view-drag-insert-above",
51511                                 "x-view-drag-insert-below"]);
51512                         this.lastInsertClass = "_noclass";
51513                 }
51514     },
51515
51516 /**
51517  *      Utility method. Add a delete option to the DDView's context menu.
51518  *      @param {String} imageUrl The URL of the "delete" icon image.
51519  */
51520         setDeletable: function(imageUrl) {
51521                 if (!this.singleSelect && !this.multiSelect) {
51522                         this.singleSelect = true;
51523                 }
51524                 var c = this.getContextMenu();
51525                 this.contextMenu.on("itemclick", function(item) {
51526                         switch (item.id) {
51527                                 case "delete":
51528                                         this.remove(this.getSelectedIndexes());
51529                                         break;
51530                         }
51531                 }, this);
51532                 this.contextMenu.add({
51533                         icon: imageUrl,
51534                         id: "delete",
51535                         text: 'Delete'
51536                 });
51537         },
51538         
51539 /**     Return the context menu for this DDView. */
51540         getContextMenu: function() {
51541                 if (!this.contextMenu) {
51542 //                      Create the View's context menu
51543                         this.contextMenu = new Roo.menu.Menu({
51544                                 id: this.id + "-contextmenu"
51545                         });
51546                         this.el.on("contextmenu", this.showContextMenu, this);
51547                 }
51548                 return this.contextMenu;
51549         },
51550         
51551         disableContextMenu: function() {
51552                 if (this.contextMenu) {
51553                         this.el.un("contextmenu", this.showContextMenu, this);
51554                 }
51555         },
51556
51557         showContextMenu: function(e, item) {
51558         item = this.findItemFromChild(e.getTarget());
51559                 if (item) {
51560                         e.stopEvent();
51561                         this.select(this.getNode(item), this.multiSelect && e.ctrlKey, true);
51562                         this.contextMenu.showAt(e.getXY());
51563             }
51564     },
51565
51566 /**
51567  *      Remove {@link Roo.data.Record}s at the specified indices.
51568  *      @param {Array/Number} selectedIndices The index (or Array of indices) of Records to remove.
51569  */
51570     remove: function(selectedIndices) {
51571                 selectedIndices = [].concat(selectedIndices);
51572                 for (var i = 0; i < selectedIndices.length; i++) {
51573                         var rec = this.store.getAt(selectedIndices[i]);
51574                         this.store.remove(rec);
51575                 }
51576     },
51577
51578 /**
51579  *      Double click fires the event, but also, if this is draggable, and there is only one other
51580  *      related DropZone, it transfers the selected node.
51581  */
51582     onDblClick : function(e){
51583         var item = this.findItemFromChild(e.getTarget());
51584         if(item){
51585             if (this.fireEvent("dblclick", this, this.indexOf(item), item, e) === false) {
51586                 return false;
51587             }
51588             if (this.dragGroup) {
51589                     var targets = Roo.dd.DragDropMgr.getRelated(this.dragZone, true);
51590                     while (targets.indexOf(this.dropZone) > -1) {
51591                             targets.remove(this.dropZone);
51592                                 }
51593                     if (targets.length == 1) {
51594                                         this.dragZone.cachedTarget = null;
51595                         var el = Roo.get(targets[0].getEl());
51596                         var box = el.getBox(true);
51597                         targets[0].onNodeDrop(el.dom, {
51598                                 target: el.dom,
51599                                 xy: [box.x, box.y + box.height - 1]
51600                         }, null, this.getDragData(e));
51601                     }
51602                 }
51603         }
51604     },
51605     
51606     handleSelection: function(e) {
51607                 this.dragZone.cachedTarget = null;
51608         var item = this.findItemFromChild(e.getTarget());
51609         if (!item) {
51610                 this.clearSelections(true);
51611                 return;
51612         }
51613                 if (item && (this.multiSelect || this.singleSelect)){
51614                         if(this.multiSelect && e.shiftKey && (!e.ctrlKey) && this.lastSelection){
51615                                 this.select(this.getNodes(this.indexOf(this.lastSelection), item.nodeIndex), false);
51616                         }else if (this.isSelected(this.getNode(item)) && e.ctrlKey){
51617                                 this.unselect(item);
51618                         } else {
51619                                 this.select(item, this.multiSelect && e.ctrlKey);
51620                                 this.lastSelection = item;
51621                         }
51622                 }
51623     },
51624
51625     onItemClick : function(item, index, e){
51626                 if(this.fireEvent("beforeclick", this, index, item, e) === false){
51627                         return false;
51628                 }
51629                 return true;
51630     },
51631
51632     unselect : function(nodeInfo, suppressEvent){
51633                 var node = this.getNode(nodeInfo);
51634                 if(node && this.isSelected(node)){
51635                         if(this.fireEvent("beforeselect", this, node, this.selections) !== false){
51636                                 Roo.fly(node).removeClass(this.selectedClass);
51637                                 this.selections.remove(node);
51638                                 if(!suppressEvent){
51639                                         this.fireEvent("selectionchange", this, this.selections);
51640                                 }
51641                         }
51642                 }
51643     }
51644 });
51645 /*
51646  * Based on:
51647  * Ext JS Library 1.1.1
51648  * Copyright(c) 2006-2007, Ext JS, LLC.
51649  *
51650  * Originally Released Under LGPL - original licence link has changed is not relivant.
51651  *
51652  * Fork - LGPL
51653  * <script type="text/javascript">
51654  */
51655  
51656 /**
51657  * @class Roo.LayoutManager
51658  * @extends Roo.util.Observable
51659  * Base class for layout managers.
51660  */
51661 Roo.LayoutManager = function(container, config){
51662     Roo.LayoutManager.superclass.constructor.call(this);
51663     this.el = Roo.get(container);
51664     // ie scrollbar fix
51665     if(this.el.dom == document.body && Roo.isIE && !config.allowScroll){
51666         document.body.scroll = "no";
51667     }else if(this.el.dom != document.body && this.el.getStyle('position') == 'static'){
51668         this.el.position('relative');
51669     }
51670     this.id = this.el.id;
51671     this.el.addClass("x-layout-container");
51672     /** false to disable window resize monitoring @type Boolean */
51673     this.monitorWindowResize = true;
51674     this.regions = {};
51675     this.addEvents({
51676         /**
51677          * @event layout
51678          * Fires when a layout is performed. 
51679          * @param {Roo.LayoutManager} this
51680          */
51681         "layout" : true,
51682         /**
51683          * @event regionresized
51684          * Fires when the user resizes a region. 
51685          * @param {Roo.LayoutRegion} region The resized region
51686          * @param {Number} newSize The new size (width for east/west, height for north/south)
51687          */
51688         "regionresized" : true,
51689         /**
51690          * @event regioncollapsed
51691          * Fires when a region is collapsed. 
51692          * @param {Roo.LayoutRegion} region The collapsed region
51693          */
51694         "regioncollapsed" : true,
51695         /**
51696          * @event regionexpanded
51697          * Fires when a region is expanded.  
51698          * @param {Roo.LayoutRegion} region The expanded region
51699          */
51700         "regionexpanded" : true
51701     });
51702     this.updating = false;
51703     Roo.EventManager.onWindowResize(this.onWindowResize, this, true);
51704 };
51705
51706 Roo.extend(Roo.LayoutManager, Roo.util.Observable, {
51707     /**
51708      * Returns true if this layout is currently being updated
51709      * @return {Boolean}
51710      */
51711     isUpdating : function(){
51712         return this.updating; 
51713     },
51714     
51715     /**
51716      * Suspend the LayoutManager from doing auto-layouts while
51717      * making multiple add or remove calls
51718      */
51719     beginUpdate : function(){
51720         this.updating = true;    
51721     },
51722     
51723     /**
51724      * Restore auto-layouts and optionally disable the manager from performing a layout
51725      * @param {Boolean} noLayout true to disable a layout update 
51726      */
51727     endUpdate : function(noLayout){
51728         this.updating = false;
51729         if(!noLayout){
51730             this.layout();
51731         }    
51732     },
51733     
51734     layout: function(){
51735         
51736     },
51737     
51738     onRegionResized : function(region, newSize){
51739         this.fireEvent("regionresized", region, newSize);
51740         this.layout();
51741     },
51742     
51743     onRegionCollapsed : function(region){
51744         this.fireEvent("regioncollapsed", region);
51745     },
51746     
51747     onRegionExpanded : function(region){
51748         this.fireEvent("regionexpanded", region);
51749     },
51750         
51751     /**
51752      * Returns the size of the current view. This method normalizes document.body and element embedded layouts and
51753      * performs box-model adjustments.
51754      * @return {Object} The size as an object {width: (the width), height: (the height)}
51755      */
51756     getViewSize : function(){
51757         var size;
51758         if(this.el.dom != document.body){
51759             size = this.el.getSize();
51760         }else{
51761             size = {width: Roo.lib.Dom.getViewWidth(), height: Roo.lib.Dom.getViewHeight()};
51762         }
51763         size.width -= this.el.getBorderWidth("lr")-this.el.getPadding("lr");
51764         size.height -= this.el.getBorderWidth("tb")-this.el.getPadding("tb");
51765         return size;
51766     },
51767     
51768     /**
51769      * Returns the Element this layout is bound to.
51770      * @return {Roo.Element}
51771      */
51772     getEl : function(){
51773         return this.el;
51774     },
51775     
51776     /**
51777      * Returns the specified region.
51778      * @param {String} target The region key ('center', 'north', 'south', 'east' or 'west')
51779      * @return {Roo.LayoutRegion}
51780      */
51781     getRegion : function(target){
51782         return this.regions[target.toLowerCase()];
51783     },
51784     
51785     onWindowResize : function(){
51786         if(this.monitorWindowResize){
51787             this.layout();
51788         }
51789     }
51790 });/*
51791  * Based on:
51792  * Ext JS Library 1.1.1
51793  * Copyright(c) 2006-2007, Ext JS, LLC.
51794  *
51795  * Originally Released Under LGPL - original licence link has changed is not relivant.
51796  *
51797  * Fork - LGPL
51798  * <script type="text/javascript">
51799  */
51800 /**
51801  * @class Roo.BorderLayout
51802  * @extends Roo.LayoutManager
51803  * This class represents a common layout manager used in desktop applications. For screenshots and more details,
51804  * please see: <br><br>
51805  * <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>
51806  * <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>
51807  * Example:
51808  <pre><code>
51809  var layout = new Roo.BorderLayout(document.body, {
51810     north: {
51811         initialSize: 25,
51812         titlebar: false
51813     },
51814     west: {
51815         split:true,
51816         initialSize: 200,
51817         minSize: 175,
51818         maxSize: 400,
51819         titlebar: true,
51820         collapsible: true
51821     },
51822     east: {
51823         split:true,
51824         initialSize: 202,
51825         minSize: 175,
51826         maxSize: 400,
51827         titlebar: true,
51828         collapsible: true
51829     },
51830     south: {
51831         split:true,
51832         initialSize: 100,
51833         minSize: 100,
51834         maxSize: 200,
51835         titlebar: true,
51836         collapsible: true
51837     },
51838     center: {
51839         titlebar: true,
51840         autoScroll:true,
51841         resizeTabs: true,
51842         minTabWidth: 50,
51843         preferredTabWidth: 150
51844     }
51845 });
51846
51847 // shorthand
51848 var CP = Roo.ContentPanel;
51849
51850 layout.beginUpdate();
51851 layout.add("north", new CP("north", "North"));
51852 layout.add("south", new CP("south", {title: "South", closable: true}));
51853 layout.add("west", new CP("west", {title: "West"}));
51854 layout.add("east", new CP("autoTabs", {title: "Auto Tabs", closable: true}));
51855 layout.add("center", new CP("center1", {title: "Close Me", closable: true}));
51856 layout.add("center", new CP("center2", {title: "Center Panel", closable: false}));
51857 layout.getRegion("center").showPanel("center1");
51858 layout.endUpdate();
51859 </code></pre>
51860
51861 <b>The container the layout is rendered into can be either the body element or any other element.
51862 If it is not the body element, the container needs to either be an absolute positioned element,
51863 or you will need to add "position:relative" to the css of the container.  You will also need to specify
51864 the container size if it is not the body element.</b>
51865
51866 * @constructor
51867 * Create a new BorderLayout
51868 * @param {String/HTMLElement/Element} container The container this layout is bound to
51869 * @param {Object} config Configuration options
51870  */
51871 Roo.BorderLayout = function(container, config){
51872     config = config || {};
51873     Roo.BorderLayout.superclass.constructor.call(this, container, config);
51874     this.factory = config.factory || Roo.BorderLayout.RegionFactory;
51875     for(var i = 0, len = this.factory.validRegions.length; i < len; i++) {
51876         var target = this.factory.validRegions[i];
51877         if(config[target]){
51878             this.addRegion(target, config[target]);
51879         }
51880     }
51881 };
51882
51883 Roo.extend(Roo.BorderLayout, Roo.LayoutManager, {
51884     /**
51885      * Creates and adds a new region if it doesn't already exist.
51886      * @param {String} target The target region key (north, south, east, west or center).
51887      * @param {Object} config The regions config object
51888      * @return {BorderLayoutRegion} The new region
51889      */
51890     addRegion : function(target, config){
51891         if(!this.regions[target]){
51892             var r = this.factory.create(target, this, config);
51893             this.bindRegion(target, r);
51894         }
51895         return this.regions[target];
51896     },
51897
51898     // private (kinda)
51899     bindRegion : function(name, r){
51900         this.regions[name] = r;
51901         r.on("visibilitychange", this.layout, this);
51902         r.on("paneladded", this.layout, this);
51903         r.on("panelremoved", this.layout, this);
51904         r.on("invalidated", this.layout, this);
51905         r.on("resized", this.onRegionResized, this);
51906         r.on("collapsed", this.onRegionCollapsed, this);
51907         r.on("expanded", this.onRegionExpanded, this);
51908     },
51909
51910     /**
51911      * Performs a layout update.
51912      */
51913     layout : function(){
51914         if(this.updating) {
51915             return;
51916         }
51917         var size = this.getViewSize();
51918         var w = size.width;
51919         var h = size.height;
51920         var centerW = w;
51921         var centerH = h;
51922         var centerY = 0;
51923         var centerX = 0;
51924         //var x = 0, y = 0;
51925
51926         var rs = this.regions;
51927         var north = rs["north"];
51928         var south = rs["south"]; 
51929         var west = rs["west"];
51930         var east = rs["east"];
51931         var center = rs["center"];
51932         //if(this.hideOnLayout){ // not supported anymore
51933             //c.el.setStyle("display", "none");
51934         //}
51935         if(north && north.isVisible()){
51936             var b = north.getBox();
51937             var m = north.getMargins();
51938             b.width = w - (m.left+m.right);
51939             b.x = m.left;
51940             b.y = m.top;
51941             centerY = b.height + b.y + m.bottom;
51942             centerH -= centerY;
51943             north.updateBox(this.safeBox(b));
51944         }
51945         if(south && south.isVisible()){
51946             var b = south.getBox();
51947             var m = south.getMargins();
51948             b.width = w - (m.left+m.right);
51949             b.x = m.left;
51950             var totalHeight = (b.height + m.top + m.bottom);
51951             b.y = h - totalHeight + m.top;
51952             centerH -= totalHeight;
51953             south.updateBox(this.safeBox(b));
51954         }
51955         if(west && west.isVisible()){
51956             var b = west.getBox();
51957             var m = west.getMargins();
51958             b.height = centerH - (m.top+m.bottom);
51959             b.x = m.left;
51960             b.y = centerY + m.top;
51961             var totalWidth = (b.width + m.left + m.right);
51962             centerX += totalWidth;
51963             centerW -= totalWidth;
51964             west.updateBox(this.safeBox(b));
51965         }
51966         if(east && east.isVisible()){
51967             var b = east.getBox();
51968             var m = east.getMargins();
51969             b.height = centerH - (m.top+m.bottom);
51970             var totalWidth = (b.width + m.left + m.right);
51971             b.x = w - totalWidth + m.left;
51972             b.y = centerY + m.top;
51973             centerW -= totalWidth;
51974             east.updateBox(this.safeBox(b));
51975         }
51976         if(center){
51977             var m = center.getMargins();
51978             var centerBox = {
51979                 x: centerX + m.left,
51980                 y: centerY + m.top,
51981                 width: centerW - (m.left+m.right),
51982                 height: centerH - (m.top+m.bottom)
51983             };
51984             //if(this.hideOnLayout){
51985                 //center.el.setStyle("display", "block");
51986             //}
51987             center.updateBox(this.safeBox(centerBox));
51988         }
51989         this.el.repaint();
51990         this.fireEvent("layout", this);
51991     },
51992
51993     // private
51994     safeBox : function(box){
51995         box.width = Math.max(0, box.width);
51996         box.height = Math.max(0, box.height);
51997         return box;
51998     },
51999
52000     /**
52001      * Adds a ContentPanel (or subclass) to this layout.
52002      * @param {String} target The target region key (north, south, east, west or center).
52003      * @param {Roo.ContentPanel} panel The panel to add
52004      * @return {Roo.ContentPanel} The added panel
52005      */
52006     add : function(target, panel){
52007          
52008         target = target.toLowerCase();
52009         return this.regions[target].add(panel);
52010     },
52011
52012     /**
52013      * Remove a ContentPanel (or subclass) to this layout.
52014      * @param {String} target The target region key (north, south, east, west or center).
52015      * @param {Number/String/Roo.ContentPanel} panel The index, id or panel to remove
52016      * @return {Roo.ContentPanel} The removed panel
52017      */
52018     remove : function(target, panel){
52019         target = target.toLowerCase();
52020         return this.regions[target].remove(panel);
52021     },
52022
52023     /**
52024      * Searches all regions for a panel with the specified id
52025      * @param {String} panelId
52026      * @return {Roo.ContentPanel} The panel or null if it wasn't found
52027      */
52028     findPanel : function(panelId){
52029         var rs = this.regions;
52030         for(var target in rs){
52031             if(typeof rs[target] != "function"){
52032                 var p = rs[target].getPanel(panelId);
52033                 if(p){
52034                     return p;
52035                 }
52036             }
52037         }
52038         return null;
52039     },
52040
52041     /**
52042      * Searches all regions for a panel with the specified id and activates (shows) it.
52043      * @param {String/ContentPanel} panelId The panels id or the panel itself
52044      * @return {Roo.ContentPanel} The shown panel or null
52045      */
52046     showPanel : function(panelId) {
52047       var rs = this.regions;
52048       for(var target in rs){
52049          var r = rs[target];
52050          if(typeof r != "function"){
52051             if(r.hasPanel(panelId)){
52052                return r.showPanel(panelId);
52053             }
52054          }
52055       }
52056       return null;
52057    },
52058
52059    /**
52060      * Restores this layout's state using Roo.state.Manager or the state provided by the passed provider.
52061      * @param {Roo.state.Provider} provider (optional) An alternate state provider
52062      */
52063     restoreState : function(provider){
52064         if(!provider){
52065             provider = Roo.state.Manager;
52066         }
52067         var sm = new Roo.LayoutStateManager();
52068         sm.init(this, provider);
52069     },
52070
52071     /**
52072      * Adds a batch of multiple ContentPanels dynamically by passing a special regions config object.  This config
52073      * object should contain properties for each region to add ContentPanels to, and each property's value should be
52074      * a valid ContentPanel config object.  Example:
52075      * <pre><code>
52076 // Create the main layout
52077 var layout = new Roo.BorderLayout('main-ct', {
52078     west: {
52079         split:true,
52080         minSize: 175,
52081         titlebar: true
52082     },
52083     center: {
52084         title:'Components'
52085     }
52086 }, 'main-ct');
52087
52088 // Create and add multiple ContentPanels at once via configs
52089 layout.batchAdd({
52090    west: {
52091        id: 'source-files',
52092        autoCreate:true,
52093        title:'Ext Source Files',
52094        autoScroll:true,
52095        fitToFrame:true
52096    },
52097    center : {
52098        el: cview,
52099        autoScroll:true,
52100        fitToFrame:true,
52101        toolbar: tb,
52102        resizeEl:'cbody'
52103    }
52104 });
52105 </code></pre>
52106      * @param {Object} regions An object containing ContentPanel configs by region name
52107      */
52108     batchAdd : function(regions){
52109         this.beginUpdate();
52110         for(var rname in regions){
52111             var lr = this.regions[rname];
52112             if(lr){
52113                 this.addTypedPanels(lr, regions[rname]);
52114             }
52115         }
52116         this.endUpdate();
52117     },
52118
52119     // private
52120     addTypedPanels : function(lr, ps){
52121         if(typeof ps == 'string'){
52122             lr.add(new Roo.ContentPanel(ps));
52123         }
52124         else if(ps instanceof Array){
52125             for(var i =0, len = ps.length; i < len; i++){
52126                 this.addTypedPanels(lr, ps[i]);
52127             }
52128         }
52129         else if(!ps.events){ // raw config?
52130             var el = ps.el;
52131             delete ps.el; // prevent conflict
52132             lr.add(new Roo.ContentPanel(el || Roo.id(), ps));
52133         }
52134         else {  // panel object assumed!
52135             lr.add(ps);
52136         }
52137     },
52138     /**
52139      * Adds a xtype elements to the layout.
52140      * <pre><code>
52141
52142 layout.addxtype({
52143        xtype : 'ContentPanel',
52144        region: 'west',
52145        items: [ .... ]
52146    }
52147 );
52148
52149 layout.addxtype({
52150         xtype : 'NestedLayoutPanel',
52151         region: 'west',
52152         layout: {
52153            center: { },
52154            west: { }   
52155         },
52156         items : [ ... list of content panels or nested layout panels.. ]
52157    }
52158 );
52159 </code></pre>
52160      * @param {Object} cfg Xtype definition of item to add.
52161      */
52162     addxtype : function(cfg)
52163     {
52164         // basically accepts a pannel...
52165         // can accept a layout region..!?!?
52166         //Roo.log('Roo.BorderLayout add ' + cfg.xtype)
52167         
52168         if (!cfg.xtype.match(/Panel$/)) {
52169             return false;
52170         }
52171         var ret = false;
52172         
52173         if (typeof(cfg.region) == 'undefined') {
52174             Roo.log("Failed to add Panel, region was not set");
52175             Roo.log(cfg);
52176             return false;
52177         }
52178         var region = cfg.region;
52179         delete cfg.region;
52180         
52181           
52182         var xitems = [];
52183         if (cfg.items) {
52184             xitems = cfg.items;
52185             delete cfg.items;
52186         }
52187         var nb = false;
52188         
52189         switch(cfg.xtype) 
52190         {
52191             case 'ContentPanel':  // ContentPanel (el, cfg)
52192             case 'ScrollPanel':  // ContentPanel (el, cfg)
52193             case 'ViewPanel': 
52194                 if(cfg.autoCreate) {
52195                     ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52196                 } else {
52197                     var el = this.el.createChild();
52198                     ret = new Roo[cfg.xtype](el, cfg); // new panel!!!!!
52199                 }
52200                 
52201                 this.add(region, ret);
52202                 break;
52203             
52204             
52205             case 'TreePanel': // our new panel!
52206                 cfg.el = this.el.createChild();
52207                 ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52208                 this.add(region, ret);
52209                 break;
52210             
52211             case 'NestedLayoutPanel': 
52212                 // create a new Layout (which is  a Border Layout...
52213                 var el = this.el.createChild();
52214                 var clayout = cfg.layout;
52215                 delete cfg.layout;
52216                 clayout.items   = clayout.items  || [];
52217                 // replace this exitems with the clayout ones..
52218                 xitems = clayout.items;
52219                  
52220                 
52221                 if (region == 'center' && this.active && this.getRegion('center').panels.length < 1) {
52222                     cfg.background = false;
52223                 }
52224                 var layout = new Roo.BorderLayout(el, clayout);
52225                 
52226                 ret = new Roo[cfg.xtype](layout, cfg); // new panel!!!!!
52227                 //console.log('adding nested layout panel '  + cfg.toSource());
52228                 this.add(region, ret);
52229                 nb = {}; /// find first...
52230                 break;
52231                 
52232             case 'GridPanel': 
52233             
52234                 // needs grid and region
52235                 
52236                 //var el = this.getRegion(region).el.createChild();
52237                 var el = this.el.createChild();
52238                 // create the grid first...
52239                 
52240                 var grid = new Roo.grid[cfg.grid.xtype](el, cfg.grid);
52241                 delete cfg.grid;
52242                 if (region == 'center' && this.active ) {
52243                     cfg.background = false;
52244                 }
52245                 ret = new Roo[cfg.xtype](grid, cfg); // new panel!!!!!
52246                 
52247                 this.add(region, ret);
52248                 if (cfg.background) {
52249                     ret.on('activate', function(gp) {
52250                         if (!gp.grid.rendered) {
52251                             gp.grid.render();
52252                         }
52253                     });
52254                 } else {
52255                     grid.render();
52256                 }
52257                 break;
52258            
52259            
52260            
52261                 
52262                 
52263                 
52264             default:
52265                 if (typeof(Roo[cfg.xtype]) != 'undefined') {
52266                     
52267                     ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52268                     this.add(region, ret);
52269                 } else {
52270                 
52271                     alert("Can not add '" + cfg.xtype + "' to BorderLayout");
52272                     return null;
52273                 }
52274                 
52275              // GridPanel (grid, cfg)
52276             
52277         }
52278         this.beginUpdate();
52279         // add children..
52280         var region = '';
52281         var abn = {};
52282         Roo.each(xitems, function(i)  {
52283             region = nb && i.region ? i.region : false;
52284             
52285             var add = ret.addxtype(i);
52286            
52287             if (region) {
52288                 nb[region] = nb[region] == undefined ? 0 : nb[region]+1;
52289                 if (!i.background) {
52290                     abn[region] = nb[region] ;
52291                 }
52292             }
52293             
52294         });
52295         this.endUpdate();
52296
52297         // make the last non-background panel active..
52298         //if (nb) { Roo.log(abn); }
52299         if (nb) {
52300             
52301             for(var r in abn) {
52302                 region = this.getRegion(r);
52303                 if (region) {
52304                     // tried using nb[r], but it does not work..
52305                      
52306                     region.showPanel(abn[r]);
52307                    
52308                 }
52309             }
52310         }
52311         return ret;
52312         
52313     }
52314 });
52315
52316 /**
52317  * Shortcut for creating a new BorderLayout object and adding one or more ContentPanels to it in a single step, handling
52318  * the beginUpdate and endUpdate calls internally.  The key to this method is the <b>panels</b> property that can be
52319  * provided with each region config, which allows you to add ContentPanel configs in addition to the region configs
52320  * during creation.  The following code is equivalent to the constructor-based example at the beginning of this class:
52321  * <pre><code>
52322 // shorthand
52323 var CP = Roo.ContentPanel;
52324
52325 var layout = Roo.BorderLayout.create({
52326     north: {
52327         initialSize: 25,
52328         titlebar: false,
52329         panels: [new CP("north", "North")]
52330     },
52331     west: {
52332         split:true,
52333         initialSize: 200,
52334         minSize: 175,
52335         maxSize: 400,
52336         titlebar: true,
52337         collapsible: true,
52338         panels: [new CP("west", {title: "West"})]
52339     },
52340     east: {
52341         split:true,
52342         initialSize: 202,
52343         minSize: 175,
52344         maxSize: 400,
52345         titlebar: true,
52346         collapsible: true,
52347         panels: [new CP("autoTabs", {title: "Auto Tabs", closable: true})]
52348     },
52349     south: {
52350         split:true,
52351         initialSize: 100,
52352         minSize: 100,
52353         maxSize: 200,
52354         titlebar: true,
52355         collapsible: true,
52356         panels: [new CP("south", {title: "South", closable: true})]
52357     },
52358     center: {
52359         titlebar: true,
52360         autoScroll:true,
52361         resizeTabs: true,
52362         minTabWidth: 50,
52363         preferredTabWidth: 150,
52364         panels: [
52365             new CP("center1", {title: "Close Me", closable: true}),
52366             new CP("center2", {title: "Center Panel", closable: false})
52367         ]
52368     }
52369 }, document.body);
52370
52371 layout.getRegion("center").showPanel("center1");
52372 </code></pre>
52373  * @param config
52374  * @param targetEl
52375  */
52376 Roo.BorderLayout.create = function(config, targetEl){
52377     var layout = new Roo.BorderLayout(targetEl || document.body, config);
52378     layout.beginUpdate();
52379     var regions = Roo.BorderLayout.RegionFactory.validRegions;
52380     for(var j = 0, jlen = regions.length; j < jlen; j++){
52381         var lr = regions[j];
52382         if(layout.regions[lr] && config[lr].panels){
52383             var r = layout.regions[lr];
52384             var ps = config[lr].panels;
52385             layout.addTypedPanels(r, ps);
52386         }
52387     }
52388     layout.endUpdate();
52389     return layout;
52390 };
52391
52392 // private
52393 Roo.BorderLayout.RegionFactory = {
52394     // private
52395     validRegions : ["north","south","east","west","center"],
52396
52397     // private
52398     create : function(target, mgr, config){
52399         target = target.toLowerCase();
52400         if(config.lightweight || config.basic){
52401             return new Roo.BasicLayoutRegion(mgr, config, target);
52402         }
52403         switch(target){
52404             case "north":
52405                 return new Roo.NorthLayoutRegion(mgr, config);
52406             case "south":
52407                 return new Roo.SouthLayoutRegion(mgr, config);
52408             case "east":
52409                 return new Roo.EastLayoutRegion(mgr, config);
52410             case "west":
52411                 return new Roo.WestLayoutRegion(mgr, config);
52412             case "center":
52413                 return new Roo.CenterLayoutRegion(mgr, config);
52414         }
52415         throw 'Layout region "'+target+'" not supported.';
52416     }
52417 };/*
52418  * Based on:
52419  * Ext JS Library 1.1.1
52420  * Copyright(c) 2006-2007, Ext JS, LLC.
52421  *
52422  * Originally Released Under LGPL - original licence link has changed is not relivant.
52423  *
52424  * Fork - LGPL
52425  * <script type="text/javascript">
52426  */
52427  
52428 /**
52429  * @class Roo.BasicLayoutRegion
52430  * @extends Roo.util.Observable
52431  * This class represents a lightweight region in a layout manager. This region does not move dom nodes
52432  * and does not have a titlebar, tabs or any other features. All it does is size and position 
52433  * panels. To create a BasicLayoutRegion, add lightweight:true or basic:true to your regions config.
52434  */
52435 Roo.BasicLayoutRegion = function(mgr, config, pos, skipConfig){
52436     this.mgr = mgr;
52437     this.position  = pos;
52438     this.events = {
52439         /**
52440          * @scope Roo.BasicLayoutRegion
52441          */
52442         
52443         /**
52444          * @event beforeremove
52445          * Fires before a panel is removed (or closed). To cancel the removal set "e.cancel = true" on the event argument.
52446          * @param {Roo.LayoutRegion} this
52447          * @param {Roo.ContentPanel} panel The panel
52448          * @param {Object} e The cancel event object
52449          */
52450         "beforeremove" : true,
52451         /**
52452          * @event invalidated
52453          * Fires when the layout for this region is changed.
52454          * @param {Roo.LayoutRegion} this
52455          */
52456         "invalidated" : true,
52457         /**
52458          * @event visibilitychange
52459          * Fires when this region is shown or hidden 
52460          * @param {Roo.LayoutRegion} this
52461          * @param {Boolean} visibility true or false
52462          */
52463         "visibilitychange" : true,
52464         /**
52465          * @event paneladded
52466          * Fires when a panel is added. 
52467          * @param {Roo.LayoutRegion} this
52468          * @param {Roo.ContentPanel} panel The panel
52469          */
52470         "paneladded" : true,
52471         /**
52472          * @event panelremoved
52473          * Fires when a panel is removed. 
52474          * @param {Roo.LayoutRegion} this
52475          * @param {Roo.ContentPanel} panel The panel
52476          */
52477         "panelremoved" : true,
52478         /**
52479          * @event beforecollapse
52480          * Fires when this region before collapse.
52481          * @param {Roo.LayoutRegion} this
52482          */
52483         "beforecollapse" : true,
52484         /**
52485          * @event collapsed
52486          * Fires when this region is collapsed.
52487          * @param {Roo.LayoutRegion} this
52488          */
52489         "collapsed" : true,
52490         /**
52491          * @event expanded
52492          * Fires when this region is expanded.
52493          * @param {Roo.LayoutRegion} this
52494          */
52495         "expanded" : true,
52496         /**
52497          * @event slideshow
52498          * Fires when this region is slid into view.
52499          * @param {Roo.LayoutRegion} this
52500          */
52501         "slideshow" : true,
52502         /**
52503          * @event slidehide
52504          * Fires when this region slides out of view. 
52505          * @param {Roo.LayoutRegion} this
52506          */
52507         "slidehide" : true,
52508         /**
52509          * @event panelactivated
52510          * Fires when a panel is activated. 
52511          * @param {Roo.LayoutRegion} this
52512          * @param {Roo.ContentPanel} panel The activated panel
52513          */
52514         "panelactivated" : true,
52515         /**
52516          * @event resized
52517          * Fires when the user resizes this region. 
52518          * @param {Roo.LayoutRegion} this
52519          * @param {Number} newSize The new size (width for east/west, height for north/south)
52520          */
52521         "resized" : true
52522     };
52523     /** A collection of panels in this region. @type Roo.util.MixedCollection */
52524     this.panels = new Roo.util.MixedCollection();
52525     this.panels.getKey = this.getPanelId.createDelegate(this);
52526     this.box = null;
52527     this.activePanel = null;
52528     // ensure listeners are added...
52529     
52530     if (config.listeners || config.events) {
52531         Roo.BasicLayoutRegion.superclass.constructor.call(this, {
52532             listeners : config.listeners || {},
52533             events : config.events || {}
52534         });
52535     }
52536     
52537     if(skipConfig !== true){
52538         this.applyConfig(config);
52539     }
52540 };
52541
52542 Roo.extend(Roo.BasicLayoutRegion, Roo.util.Observable, {
52543     getPanelId : function(p){
52544         return p.getId();
52545     },
52546     
52547     applyConfig : function(config){
52548         this.margins = config.margins || this.margins || {top: 0, left: 0, right:0, bottom: 0};
52549         this.config = config;
52550         
52551     },
52552     
52553     /**
52554      * Resizes the region to the specified size. For vertical regions (west, east) this adjusts 
52555      * the width, for horizontal (north, south) the height.
52556      * @param {Number} newSize The new width or height
52557      */
52558     resizeTo : function(newSize){
52559         var el = this.el ? this.el :
52560                  (this.activePanel ? this.activePanel.getEl() : null);
52561         if(el){
52562             switch(this.position){
52563                 case "east":
52564                 case "west":
52565                     el.setWidth(newSize);
52566                     this.fireEvent("resized", this, newSize);
52567                 break;
52568                 case "north":
52569                 case "south":
52570                     el.setHeight(newSize);
52571                     this.fireEvent("resized", this, newSize);
52572                 break;                
52573             }
52574         }
52575     },
52576     
52577     getBox : function(){
52578         return this.activePanel ? this.activePanel.getEl().getBox(false, true) : null;
52579     },
52580     
52581     getMargins : function(){
52582         return this.margins;
52583     },
52584     
52585     updateBox : function(box){
52586         this.box = box;
52587         var el = this.activePanel.getEl();
52588         el.dom.style.left = box.x + "px";
52589         el.dom.style.top = box.y + "px";
52590         this.activePanel.setSize(box.width, box.height);
52591     },
52592     
52593     /**
52594      * Returns the container element for this region.
52595      * @return {Roo.Element}
52596      */
52597     getEl : function(){
52598         return this.activePanel;
52599     },
52600     
52601     /**
52602      * Returns true if this region is currently visible.
52603      * @return {Boolean}
52604      */
52605     isVisible : function(){
52606         return this.activePanel ? true : false;
52607     },
52608     
52609     setActivePanel : function(panel){
52610         panel = this.getPanel(panel);
52611         if(this.activePanel && this.activePanel != panel){
52612             this.activePanel.setActiveState(false);
52613             this.activePanel.getEl().setLeftTop(-10000,-10000);
52614         }
52615         this.activePanel = panel;
52616         panel.setActiveState(true);
52617         if(this.box){
52618             panel.setSize(this.box.width, this.box.height);
52619         }
52620         this.fireEvent("panelactivated", this, panel);
52621         this.fireEvent("invalidated");
52622     },
52623     
52624     /**
52625      * Show the specified panel.
52626      * @param {Number/String/ContentPanel} panelId The panels index, id or the panel itself
52627      * @return {Roo.ContentPanel} The shown panel or null
52628      */
52629     showPanel : function(panel){
52630         if(panel = this.getPanel(panel)){
52631             this.setActivePanel(panel);
52632         }
52633         return panel;
52634     },
52635     
52636     /**
52637      * Get the active panel for this region.
52638      * @return {Roo.ContentPanel} The active panel or null
52639      */
52640     getActivePanel : function(){
52641         return this.activePanel;
52642     },
52643     
52644     /**
52645      * Add the passed ContentPanel(s)
52646      * @param {ContentPanel...} panel The ContentPanel(s) to add (you can pass more than one)
52647      * @return {Roo.ContentPanel} The panel added (if only one was added)
52648      */
52649     add : function(panel){
52650         if(arguments.length > 1){
52651             for(var i = 0, len = arguments.length; i < len; i++) {
52652                 this.add(arguments[i]);
52653             }
52654             return null;
52655         }
52656         if(this.hasPanel(panel)){
52657             this.showPanel(panel);
52658             return panel;
52659         }
52660         var el = panel.getEl();
52661         if(el.dom.parentNode != this.mgr.el.dom){
52662             this.mgr.el.dom.appendChild(el.dom);
52663         }
52664         if(panel.setRegion){
52665             panel.setRegion(this);
52666         }
52667         this.panels.add(panel);
52668         el.setStyle("position", "absolute");
52669         if(!panel.background){
52670             this.setActivePanel(panel);
52671             if(this.config.initialSize && this.panels.getCount()==1){
52672                 this.resizeTo(this.config.initialSize);
52673             }
52674         }
52675         this.fireEvent("paneladded", this, panel);
52676         return panel;
52677     },
52678     
52679     /**
52680      * Returns true if the panel is in this region.
52681      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52682      * @return {Boolean}
52683      */
52684     hasPanel : function(panel){
52685         if(typeof panel == "object"){ // must be panel obj
52686             panel = panel.getId();
52687         }
52688         return this.getPanel(panel) ? true : false;
52689     },
52690     
52691     /**
52692      * Removes the specified panel. If preservePanel is not true (either here or in the config), the panel is destroyed.
52693      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52694      * @param {Boolean} preservePanel Overrides the config preservePanel option
52695      * @return {Roo.ContentPanel} The panel that was removed
52696      */
52697     remove : function(panel, preservePanel){
52698         panel = this.getPanel(panel);
52699         if(!panel){
52700             return null;
52701         }
52702         var e = {};
52703         this.fireEvent("beforeremove", this, panel, e);
52704         if(e.cancel === true){
52705             return null;
52706         }
52707         var panelId = panel.getId();
52708         this.panels.removeKey(panelId);
52709         return panel;
52710     },
52711     
52712     /**
52713      * Returns the panel specified or null if it's not in this region.
52714      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52715      * @return {Roo.ContentPanel}
52716      */
52717     getPanel : function(id){
52718         if(typeof id == "object"){ // must be panel obj
52719             return id;
52720         }
52721         return this.panels.get(id);
52722     },
52723     
52724     /**
52725      * Returns this regions position (north/south/east/west/center).
52726      * @return {String} 
52727      */
52728     getPosition: function(){
52729         return this.position;    
52730     }
52731 });/*
52732  * Based on:
52733  * Ext JS Library 1.1.1
52734  * Copyright(c) 2006-2007, Ext JS, LLC.
52735  *
52736  * Originally Released Under LGPL - original licence link has changed is not relivant.
52737  *
52738  * Fork - LGPL
52739  * <script type="text/javascript">
52740  */
52741  
52742 /**
52743  * @class Roo.LayoutRegion
52744  * @extends Roo.BasicLayoutRegion
52745  * This class represents a region in a layout manager.
52746  * @cfg {Boolean}   collapsible     False to disable collapsing (defaults to true)
52747  * @cfg {Boolean}   collapsed       True to set the initial display to collapsed (defaults to false)
52748  * @cfg {Boolean}   floatable       False to disable floating (defaults to true)
52749  * @cfg {Object}    margins         Margins for the element (defaults to {top: 0, left: 0, right:0, bottom: 0})
52750  * @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})
52751  * @cfg {String}    tabPosition     (top|bottom) "top" or "bottom" (defaults to "bottom")
52752  * @cfg {String}    collapsedTitle  Optional string message to display in the collapsed block of a north or south region
52753  * @cfg {Boolean}   alwaysShowTabs  True to always display tabs even when there is only 1 panel (defaults to false)
52754  * @cfg {Boolean}   autoScroll      True to enable overflow scrolling (defaults to false)
52755  * @cfg {Boolean}   titlebar        True to display a title bar (defaults to true)
52756  * @cfg {String}    title           The title for the region (overrides panel titles)
52757  * @cfg {Boolean}   animate         True to animate expand/collapse (defaults to false)
52758  * @cfg {Boolean}   autoHide        False to disable auto hiding when the mouse leaves the "floated" region (defaults to true)
52759  * @cfg {Boolean}   preservePanels  True to preserve removed panels so they can be readded later (defaults to false)
52760  * @cfg {Boolean}   closeOnTab      True to place the close icon on the tabs instead of the region titlebar (defaults to false)
52761  * @cfg {Boolean}   hideTabs        True to hide the tab strip (defaults to false)
52762  * @cfg {Boolean}   resizeTabs      True to enable automatic tab resizing. This will resize the tabs so they are all the same size and fit within
52763  *                      the space available, similar to FireFox 1.5 tabs (defaults to false)
52764  * @cfg {Number}    minTabWidth     The minimum tab width (defaults to 40)
52765  * @cfg {Number}    preferredTabWidth The preferred tab width (defaults to 150)
52766  * @cfg {Boolean}   showPin         True to show a pin button
52767  * @cfg {Boolean}   hidden          True to start the region hidden (defaults to false)
52768  * @cfg {Boolean}   hideWhenEmpty   True to hide the region when it has no panels
52769  * @cfg {Boolean}   disableTabTips  True to disable tab tooltips
52770  * @cfg {Number}    width           For East/West panels
52771  * @cfg {Number}    height          For North/South panels
52772  * @cfg {Boolean}   split           To show the splitter
52773  * @cfg {Boolean}   toolbar         xtype configuration for a toolbar - shows on right of tabbar
52774  */
52775 Roo.LayoutRegion = function(mgr, config, pos){
52776     Roo.LayoutRegion.superclass.constructor.call(this, mgr, config, pos, true);
52777     var dh = Roo.DomHelper;
52778     /** This region's container element 
52779     * @type Roo.Element */
52780     this.el = dh.append(mgr.el.dom, {tag: "div", cls: "x-layout-panel x-layout-panel-" + this.position}, true);
52781     /** This region's title element 
52782     * @type Roo.Element */
52783
52784     this.titleEl = dh.append(this.el.dom, {tag: "div", unselectable: "on", cls: "x-unselectable x-layout-panel-hd x-layout-title-"+this.position, children:[
52785         {tag: "span", cls: "x-unselectable x-layout-panel-hd-text", unselectable: "on", html: "&#160;"},
52786         {tag: "div", cls: "x-unselectable x-layout-panel-hd-tools", unselectable: "on"}
52787     ]}, true);
52788     this.titleEl.enableDisplayMode();
52789     /** This region's title text element 
52790     * @type HTMLElement */
52791     this.titleTextEl = this.titleEl.dom.firstChild;
52792     this.tools = Roo.get(this.titleEl.dom.childNodes[1], true);
52793     this.closeBtn = this.createTool(this.tools.dom, "x-layout-close");
52794     this.closeBtn.enableDisplayMode();
52795     this.closeBtn.on("click", this.closeClicked, this);
52796     this.closeBtn.hide();
52797
52798     this.createBody(config);
52799     this.visible = true;
52800     this.collapsed = false;
52801
52802     if(config.hideWhenEmpty){
52803         this.hide();
52804         this.on("paneladded", this.validateVisibility, this);
52805         this.on("panelremoved", this.validateVisibility, this);
52806     }
52807     this.applyConfig(config);
52808 };
52809
52810 Roo.extend(Roo.LayoutRegion, Roo.BasicLayoutRegion, {
52811
52812     createBody : function(){
52813         /** This region's body element 
52814         * @type Roo.Element */
52815         this.bodyEl = this.el.createChild({tag: "div", cls: "x-layout-panel-body"});
52816     },
52817
52818     applyConfig : function(c){
52819         if(c.collapsible && this.position != "center" && !this.collapsedEl){
52820             var dh = Roo.DomHelper;
52821             if(c.titlebar !== false){
52822                 this.collapseBtn = this.createTool(this.tools.dom, "x-layout-collapse-"+this.position);
52823                 this.collapseBtn.on("click", this.collapse, this);
52824                 this.collapseBtn.enableDisplayMode();
52825
52826                 if(c.showPin === true || this.showPin){
52827                     this.stickBtn = this.createTool(this.tools.dom, "x-layout-stick");
52828                     this.stickBtn.enableDisplayMode();
52829                     this.stickBtn.on("click", this.expand, this);
52830                     this.stickBtn.hide();
52831                 }
52832             }
52833             /** This region's collapsed element
52834             * @type Roo.Element */
52835             this.collapsedEl = dh.append(this.mgr.el.dom, {cls: "x-layout-collapsed x-layout-collapsed-"+this.position, children:[
52836                 {cls: "x-layout-collapsed-tools", children:[{cls: "x-layout-ctools-inner"}]}
52837             ]}, true);
52838             if(c.floatable !== false){
52839                this.collapsedEl.addClassOnOver("x-layout-collapsed-over");
52840                this.collapsedEl.on("click", this.collapseClick, this);
52841             }
52842
52843             if(c.collapsedTitle && (this.position == "north" || this.position== "south")) {
52844                 this.collapsedTitleTextEl = dh.append(this.collapsedEl.dom, {tag: "div", cls: "x-unselectable x-layout-panel-hd-text",
52845                    id: "message", unselectable: "on", style:{"float":"left"}});
52846                this.collapsedTitleTextEl.innerHTML = c.collapsedTitle;
52847              }
52848             this.expandBtn = this.createTool(this.collapsedEl.dom.firstChild.firstChild, "x-layout-expand-"+this.position);
52849             this.expandBtn.on("click", this.expand, this);
52850         }
52851         if(this.collapseBtn){
52852             this.collapseBtn.setVisible(c.collapsible == true);
52853         }
52854         this.cmargins = c.cmargins || this.cmargins ||
52855                          (this.position == "west" || this.position == "east" ?
52856                              {top: 0, left: 2, right:2, bottom: 0} :
52857                              {top: 2, left: 0, right:0, bottom: 2});
52858         this.margins = c.margins || this.margins || {top: 0, left: 0, right:0, bottom: 0};
52859         this.bottomTabs = c.tabPosition != "top";
52860         this.autoScroll = c.autoScroll || false;
52861         if(this.autoScroll){
52862             this.bodyEl.setStyle("overflow", "auto");
52863         }else{
52864             this.bodyEl.setStyle("overflow", "hidden");
52865         }
52866         //if(c.titlebar !== false){
52867             if((!c.titlebar && !c.title) || c.titlebar === false){
52868                 this.titleEl.hide();
52869             }else{
52870                 this.titleEl.show();
52871                 if(c.title){
52872                     this.titleTextEl.innerHTML = c.title;
52873                 }
52874             }
52875         //}
52876         this.duration = c.duration || .30;
52877         this.slideDuration = c.slideDuration || .45;
52878         this.config = c;
52879         if(c.collapsed){
52880             this.collapse(true);
52881         }
52882         if(c.hidden){
52883             this.hide();
52884         }
52885     },
52886     /**
52887      * Returns true if this region is currently visible.
52888      * @return {Boolean}
52889      */
52890     isVisible : function(){
52891         return this.visible;
52892     },
52893
52894     /**
52895      * Updates the title for collapsed north/south regions (used with {@link #collapsedTitle} config option)
52896      * @param {String} title (optional) The title text (accepts HTML markup, defaults to the numeric character reference for a non-breaking space, "&amp;#160;")
52897      */
52898     setCollapsedTitle : function(title){
52899         title = title || "&#160;";
52900         if(this.collapsedTitleTextEl){
52901             this.collapsedTitleTextEl.innerHTML = title;
52902         }
52903     },
52904
52905     getBox : function(){
52906         var b;
52907         if(!this.collapsed){
52908             b = this.el.getBox(false, true);
52909         }else{
52910             b = this.collapsedEl.getBox(false, true);
52911         }
52912         return b;
52913     },
52914
52915     getMargins : function(){
52916         return this.collapsed ? this.cmargins : this.margins;
52917     },
52918
52919     highlight : function(){
52920         this.el.addClass("x-layout-panel-dragover");
52921     },
52922
52923     unhighlight : function(){
52924         this.el.removeClass("x-layout-panel-dragover");
52925     },
52926
52927     updateBox : function(box){
52928         this.box = box;
52929         if(!this.collapsed){
52930             this.el.dom.style.left = box.x + "px";
52931             this.el.dom.style.top = box.y + "px";
52932             this.updateBody(box.width, box.height);
52933         }else{
52934             this.collapsedEl.dom.style.left = box.x + "px";
52935             this.collapsedEl.dom.style.top = box.y + "px";
52936             this.collapsedEl.setSize(box.width, box.height);
52937         }
52938         if(this.tabs){
52939             this.tabs.autoSizeTabs();
52940         }
52941     },
52942
52943     updateBody : function(w, h){
52944         if(w !== null){
52945             this.el.setWidth(w);
52946             w -= this.el.getBorderWidth("rl");
52947             if(this.config.adjustments){
52948                 w += this.config.adjustments[0];
52949             }
52950         }
52951         if(h !== null){
52952             this.el.setHeight(h);
52953             h = this.titleEl && this.titleEl.isDisplayed() ? h - (this.titleEl.getHeight()||0) : h;
52954             h -= this.el.getBorderWidth("tb");
52955             if(this.config.adjustments){
52956                 h += this.config.adjustments[1];
52957             }
52958             this.bodyEl.setHeight(h);
52959             if(this.tabs){
52960                 h = this.tabs.syncHeight(h);
52961             }
52962         }
52963         if(this.panelSize){
52964             w = w !== null ? w : this.panelSize.width;
52965             h = h !== null ? h : this.panelSize.height;
52966         }
52967         if(this.activePanel){
52968             var el = this.activePanel.getEl();
52969             w = w !== null ? w : el.getWidth();
52970             h = h !== null ? h : el.getHeight();
52971             this.panelSize = {width: w, height: h};
52972             this.activePanel.setSize(w, h);
52973         }
52974         if(Roo.isIE && this.tabs){
52975             this.tabs.el.repaint();
52976         }
52977     },
52978
52979     /**
52980      * Returns the container element for this region.
52981      * @return {Roo.Element}
52982      */
52983     getEl : function(){
52984         return this.el;
52985     },
52986
52987     /**
52988      * Hides this region.
52989      */
52990     hide : function(){
52991         if(!this.collapsed){
52992             this.el.dom.style.left = "-2000px";
52993             this.el.hide();
52994         }else{
52995             this.collapsedEl.dom.style.left = "-2000px";
52996             this.collapsedEl.hide();
52997         }
52998         this.visible = false;
52999         this.fireEvent("visibilitychange", this, false);
53000     },
53001
53002     /**
53003      * Shows this region if it was previously hidden.
53004      */
53005     show : function(){
53006         if(!this.collapsed){
53007             this.el.show();
53008         }else{
53009             this.collapsedEl.show();
53010         }
53011         this.visible = true;
53012         this.fireEvent("visibilitychange", this, true);
53013     },
53014
53015     closeClicked : function(){
53016         if(this.activePanel){
53017             this.remove(this.activePanel);
53018         }
53019     },
53020
53021     collapseClick : function(e){
53022         if(this.isSlid){
53023            e.stopPropagation();
53024            this.slideIn();
53025         }else{
53026            e.stopPropagation();
53027            this.slideOut();
53028         }
53029     },
53030
53031     /**
53032      * Collapses this region.
53033      * @param {Boolean} skipAnim (optional) true to collapse the element without animation (if animate is true)
53034      */
53035     collapse : function(skipAnim, skipCheck){
53036         if(this.collapsed) {
53037             return;
53038         }
53039         
53040         if(skipCheck || this.fireEvent("beforecollapse", this) != false){
53041             
53042             this.collapsed = true;
53043             if(this.split){
53044                 this.split.el.hide();
53045             }
53046             if(this.config.animate && skipAnim !== true){
53047                 this.fireEvent("invalidated", this);
53048                 this.animateCollapse();
53049             }else{
53050                 this.el.setLocation(-20000,-20000);
53051                 this.el.hide();
53052                 this.collapsedEl.show();
53053                 this.fireEvent("collapsed", this);
53054                 this.fireEvent("invalidated", this);
53055             }
53056         }
53057         
53058     },
53059
53060     animateCollapse : function(){
53061         // overridden
53062     },
53063
53064     /**
53065      * Expands this region if it was previously collapsed.
53066      * @param {Roo.EventObject} e The event that triggered the expand (or null if calling manually)
53067      * @param {Boolean} skipAnim (optional) true to expand the element without animation (if animate is true)
53068      */
53069     expand : function(e, skipAnim){
53070         if(e) {
53071             e.stopPropagation();
53072         }
53073         if(!this.collapsed || this.el.hasActiveFx()) {
53074             return;
53075         }
53076         if(this.isSlid){
53077             this.afterSlideIn();
53078             skipAnim = true;
53079         }
53080         this.collapsed = false;
53081         if(this.config.animate && skipAnim !== true){
53082             this.animateExpand();
53083         }else{
53084             this.el.show();
53085             if(this.split){
53086                 this.split.el.show();
53087             }
53088             this.collapsedEl.setLocation(-2000,-2000);
53089             this.collapsedEl.hide();
53090             this.fireEvent("invalidated", this);
53091             this.fireEvent("expanded", this);
53092         }
53093     },
53094
53095     animateExpand : function(){
53096         // overridden
53097     },
53098
53099     initTabs : function()
53100     {
53101         this.bodyEl.setStyle("overflow", "hidden");
53102         var ts = new Roo.TabPanel(
53103                 this.bodyEl.dom,
53104                 {
53105                     tabPosition: this.bottomTabs ? 'bottom' : 'top',
53106                     disableTooltips: this.config.disableTabTips,
53107                     toolbar : this.config.toolbar
53108                 }
53109         );
53110         if(this.config.hideTabs){
53111             ts.stripWrap.setDisplayed(false);
53112         }
53113         this.tabs = ts;
53114         ts.resizeTabs = this.config.resizeTabs === true;
53115         ts.minTabWidth = this.config.minTabWidth || 40;
53116         ts.maxTabWidth = this.config.maxTabWidth || 250;
53117         ts.preferredTabWidth = this.config.preferredTabWidth || 150;
53118         ts.monitorResize = false;
53119         ts.bodyEl.setStyle("overflow", this.config.autoScroll ? "auto" : "hidden");
53120         ts.bodyEl.addClass('x-layout-tabs-body');
53121         this.panels.each(this.initPanelAsTab, this);
53122     },
53123
53124     initPanelAsTab : function(panel){
53125         var ti = this.tabs.addTab(panel.getEl().id, panel.getTitle(), null,
53126                     this.config.closeOnTab && panel.isClosable());
53127         if(panel.tabTip !== undefined){
53128             ti.setTooltip(panel.tabTip);
53129         }
53130         ti.on("activate", function(){
53131               this.setActivePanel(panel);
53132         }, this);
53133         if(this.config.closeOnTab){
53134             ti.on("beforeclose", function(t, e){
53135                 e.cancel = true;
53136                 this.remove(panel);
53137             }, this);
53138         }
53139         return ti;
53140     },
53141
53142     updatePanelTitle : function(panel, title){
53143         if(this.activePanel == panel){
53144             this.updateTitle(title);
53145         }
53146         if(this.tabs){
53147             var ti = this.tabs.getTab(panel.getEl().id);
53148             ti.setText(title);
53149             if(panel.tabTip !== undefined){
53150                 ti.setTooltip(panel.tabTip);
53151             }
53152         }
53153     },
53154
53155     updateTitle : function(title){
53156         if(this.titleTextEl && !this.config.title){
53157             this.titleTextEl.innerHTML = (typeof title != "undefined" && title.length > 0 ? title : "&#160;");
53158         }
53159     },
53160
53161     setActivePanel : function(panel){
53162         panel = this.getPanel(panel);
53163         if(this.activePanel && this.activePanel != panel){
53164             this.activePanel.setActiveState(false);
53165         }
53166         this.activePanel = panel;
53167         panel.setActiveState(true);
53168         if(this.panelSize){
53169             panel.setSize(this.panelSize.width, this.panelSize.height);
53170         }
53171         if(this.closeBtn){
53172             this.closeBtn.setVisible(!this.config.closeOnTab && !this.isSlid && panel.isClosable());
53173         }
53174         this.updateTitle(panel.getTitle());
53175         if(this.tabs){
53176             this.fireEvent("invalidated", this);
53177         }
53178         this.fireEvent("panelactivated", this, panel);
53179     },
53180
53181     /**
53182      * Shows the specified panel.
53183      * @param {Number/String/ContentPanel} panelId The panel's index, id or the panel itself
53184      * @return {Roo.ContentPanel} The shown panel, or null if a panel could not be found from panelId
53185      */
53186     showPanel : function(panel)
53187     {
53188         panel = this.getPanel(panel);
53189         if(panel){
53190             if(this.tabs){
53191                 var tab = this.tabs.getTab(panel.getEl().id);
53192                 if(tab.isHidden()){
53193                     this.tabs.unhideTab(tab.id);
53194                 }
53195                 tab.activate();
53196             }else{
53197                 this.setActivePanel(panel);
53198             }
53199         }
53200         return panel;
53201     },
53202
53203     /**
53204      * Get the active panel for this region.
53205      * @return {Roo.ContentPanel} The active panel or null
53206      */
53207     getActivePanel : function(){
53208         return this.activePanel;
53209     },
53210
53211     validateVisibility : function(){
53212         if(this.panels.getCount() < 1){
53213             this.updateTitle("&#160;");
53214             this.closeBtn.hide();
53215             this.hide();
53216         }else{
53217             if(!this.isVisible()){
53218                 this.show();
53219             }
53220         }
53221     },
53222
53223     /**
53224      * Adds the passed ContentPanel(s) to this region.
53225      * @param {ContentPanel...} panel The ContentPanel(s) to add (you can pass more than one)
53226      * @return {Roo.ContentPanel} The panel added (if only one was added; null otherwise)
53227      */
53228     add : function(panel){
53229         if(arguments.length > 1){
53230             for(var i = 0, len = arguments.length; i < len; i++) {
53231                 this.add(arguments[i]);
53232             }
53233             return null;
53234         }
53235         if(this.hasPanel(panel)){
53236             this.showPanel(panel);
53237             return panel;
53238         }
53239         panel.setRegion(this);
53240         this.panels.add(panel);
53241         if(this.panels.getCount() == 1 && !this.config.alwaysShowTabs){
53242             this.bodyEl.dom.appendChild(panel.getEl().dom);
53243             if(panel.background !== true){
53244                 this.setActivePanel(panel);
53245             }
53246             this.fireEvent("paneladded", this, panel);
53247             return panel;
53248         }
53249         if(!this.tabs){
53250             this.initTabs();
53251         }else{
53252             this.initPanelAsTab(panel);
53253         }
53254         if(panel.background !== true){
53255             this.tabs.activate(panel.getEl().id);
53256         }
53257         this.fireEvent("paneladded", this, panel);
53258         return panel;
53259     },
53260
53261     /**
53262      * Hides the tab for the specified panel.
53263      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53264      */
53265     hidePanel : function(panel){
53266         if(this.tabs && (panel = this.getPanel(panel))){
53267             this.tabs.hideTab(panel.getEl().id);
53268         }
53269     },
53270
53271     /**
53272      * Unhides the tab for a previously hidden panel.
53273      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53274      */
53275     unhidePanel : function(panel){
53276         if(this.tabs && (panel = this.getPanel(panel))){
53277             this.tabs.unhideTab(panel.getEl().id);
53278         }
53279     },
53280
53281     clearPanels : function(){
53282         while(this.panels.getCount() > 0){
53283              this.remove(this.panels.first());
53284         }
53285     },
53286
53287     /**
53288      * Removes the specified panel. If preservePanel is not true (either here or in the config), the panel is destroyed.
53289      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53290      * @param {Boolean} preservePanel Overrides the config preservePanel option
53291      * @return {Roo.ContentPanel} The panel that was removed
53292      */
53293     remove : function(panel, preservePanel){
53294         panel = this.getPanel(panel);
53295         if(!panel){
53296             return null;
53297         }
53298         var e = {};
53299         this.fireEvent("beforeremove", this, panel, e);
53300         if(e.cancel === true){
53301             return null;
53302         }
53303         preservePanel = (typeof preservePanel != "undefined" ? preservePanel : (this.config.preservePanels === true || panel.preserve === true));
53304         var panelId = panel.getId();
53305         this.panels.removeKey(panelId);
53306         if(preservePanel){
53307             document.body.appendChild(panel.getEl().dom);
53308         }
53309         if(this.tabs){
53310             this.tabs.removeTab(panel.getEl().id);
53311         }else if (!preservePanel){
53312             this.bodyEl.dom.removeChild(panel.getEl().dom);
53313         }
53314         if(this.panels.getCount() == 1 && this.tabs && !this.config.alwaysShowTabs){
53315             var p = this.panels.first();
53316             var tempEl = document.createElement("div"); // temp holder to keep IE from deleting the node
53317             tempEl.appendChild(p.getEl().dom);
53318             this.bodyEl.update("");
53319             this.bodyEl.dom.appendChild(p.getEl().dom);
53320             tempEl = null;
53321             this.updateTitle(p.getTitle());
53322             this.tabs = null;
53323             this.bodyEl.setStyle("overflow", this.config.autoScroll ? "auto" : "hidden");
53324             this.setActivePanel(p);
53325         }
53326         panel.setRegion(null);
53327         if(this.activePanel == panel){
53328             this.activePanel = null;
53329         }
53330         if(this.config.autoDestroy !== false && preservePanel !== true){
53331             try{panel.destroy();}catch(e){}
53332         }
53333         this.fireEvent("panelremoved", this, panel);
53334         return panel;
53335     },
53336
53337     /**
53338      * Returns the TabPanel component used by this region
53339      * @return {Roo.TabPanel}
53340      */
53341     getTabs : function(){
53342         return this.tabs;
53343     },
53344
53345     createTool : function(parentEl, className){
53346         var btn = Roo.DomHelper.append(parentEl, {tag: "div", cls: "x-layout-tools-button",
53347             children: [{tag: "div", cls: "x-layout-tools-button-inner " + className, html: "&#160;"}]}, true);
53348         btn.addClassOnOver("x-layout-tools-button-over");
53349         return btn;
53350     }
53351 });/*
53352  * Based on:
53353  * Ext JS Library 1.1.1
53354  * Copyright(c) 2006-2007, Ext JS, LLC.
53355  *
53356  * Originally Released Under LGPL - original licence link has changed is not relivant.
53357  *
53358  * Fork - LGPL
53359  * <script type="text/javascript">
53360  */
53361  
53362
53363
53364 /**
53365  * @class Roo.SplitLayoutRegion
53366  * @extends Roo.LayoutRegion
53367  * Adds a splitbar and other (private) useful functionality to a {@link Roo.LayoutRegion}.
53368  */
53369 Roo.SplitLayoutRegion = function(mgr, config, pos, cursor){
53370     this.cursor = cursor;
53371     Roo.SplitLayoutRegion.superclass.constructor.call(this, mgr, config, pos);
53372 };
53373
53374 Roo.extend(Roo.SplitLayoutRegion, Roo.LayoutRegion, {
53375     splitTip : "Drag to resize.",
53376     collapsibleSplitTip : "Drag to resize. Double click to hide.",
53377     useSplitTips : false,
53378
53379     applyConfig : function(config){
53380         Roo.SplitLayoutRegion.superclass.applyConfig.call(this, config);
53381         if(config.split){
53382             if(!this.split){
53383                 var splitEl = Roo.DomHelper.append(this.mgr.el.dom, 
53384                         {tag: "div", id: this.el.id + "-split", cls: "x-layout-split x-layout-split-"+this.position, html: "&#160;"});
53385                 /** The SplitBar for this region 
53386                 * @type Roo.SplitBar */
53387                 this.split = new Roo.SplitBar(splitEl, this.el, this.orientation);
53388                 this.split.on("moved", this.onSplitMove, this);
53389                 this.split.useShim = config.useShim === true;
53390                 this.split.getMaximumSize = this[this.position == 'north' || this.position == 'south' ? 'getVMaxSize' : 'getHMaxSize'].createDelegate(this);
53391                 if(this.useSplitTips){
53392                     this.split.el.dom.title = config.collapsible ? this.collapsibleSplitTip : this.splitTip;
53393                 }
53394                 if(config.collapsible){
53395                     this.split.el.on("dblclick", this.collapse,  this);
53396                 }
53397             }
53398             if(typeof config.minSize != "undefined"){
53399                 this.split.minSize = config.minSize;
53400             }
53401             if(typeof config.maxSize != "undefined"){
53402                 this.split.maxSize = config.maxSize;
53403             }
53404             if(config.hideWhenEmpty || config.hidden || config.collapsed){
53405                 this.hideSplitter();
53406             }
53407         }
53408     },
53409
53410     getHMaxSize : function(){
53411          var cmax = this.config.maxSize || 10000;
53412          var center = this.mgr.getRegion("center");
53413          return Math.min(cmax, (this.el.getWidth()+center.getEl().getWidth())-center.getMinWidth());
53414     },
53415
53416     getVMaxSize : function(){
53417          var cmax = this.config.maxSize || 10000;
53418          var center = this.mgr.getRegion("center");
53419          return Math.min(cmax, (this.el.getHeight()+center.getEl().getHeight())-center.getMinHeight());
53420     },
53421
53422     onSplitMove : function(split, newSize){
53423         this.fireEvent("resized", this, newSize);
53424     },
53425     
53426     /** 
53427      * Returns the {@link Roo.SplitBar} for this region.
53428      * @return {Roo.SplitBar}
53429      */
53430     getSplitBar : function(){
53431         return this.split;
53432     },
53433     
53434     hide : function(){
53435         this.hideSplitter();
53436         Roo.SplitLayoutRegion.superclass.hide.call(this);
53437     },
53438
53439     hideSplitter : function(){
53440         if(this.split){
53441             this.split.el.setLocation(-2000,-2000);
53442             this.split.el.hide();
53443         }
53444     },
53445
53446     show : function(){
53447         if(this.split){
53448             this.split.el.show();
53449         }
53450         Roo.SplitLayoutRegion.superclass.show.call(this);
53451     },
53452     
53453     beforeSlide: function(){
53454         if(Roo.isGecko){// firefox overflow auto bug workaround
53455             this.bodyEl.clip();
53456             if(this.tabs) {
53457                 this.tabs.bodyEl.clip();
53458             }
53459             if(this.activePanel){
53460                 this.activePanel.getEl().clip();
53461                 
53462                 if(this.activePanel.beforeSlide){
53463                     this.activePanel.beforeSlide();
53464                 }
53465             }
53466         }
53467     },
53468     
53469     afterSlide : function(){
53470         if(Roo.isGecko){// firefox overflow auto bug workaround
53471             this.bodyEl.unclip();
53472             if(this.tabs) {
53473                 this.tabs.bodyEl.unclip();
53474             }
53475             if(this.activePanel){
53476                 this.activePanel.getEl().unclip();
53477                 if(this.activePanel.afterSlide){
53478                     this.activePanel.afterSlide();
53479                 }
53480             }
53481         }
53482     },
53483
53484     initAutoHide : function(){
53485         if(this.autoHide !== false){
53486             if(!this.autoHideHd){
53487                 var st = new Roo.util.DelayedTask(this.slideIn, this);
53488                 this.autoHideHd = {
53489                     "mouseout": function(e){
53490                         if(!e.within(this.el, true)){
53491                             st.delay(500);
53492                         }
53493                     },
53494                     "mouseover" : function(e){
53495                         st.cancel();
53496                     },
53497                     scope : this
53498                 };
53499             }
53500             this.el.on(this.autoHideHd);
53501         }
53502     },
53503
53504     clearAutoHide : function(){
53505         if(this.autoHide !== false){
53506             this.el.un("mouseout", this.autoHideHd.mouseout);
53507             this.el.un("mouseover", this.autoHideHd.mouseover);
53508         }
53509     },
53510
53511     clearMonitor : function(){
53512         Roo.get(document).un("click", this.slideInIf, this);
53513     },
53514
53515     // these names are backwards but not changed for compat
53516     slideOut : function(){
53517         if(this.isSlid || this.el.hasActiveFx()){
53518             return;
53519         }
53520         this.isSlid = true;
53521         if(this.collapseBtn){
53522             this.collapseBtn.hide();
53523         }
53524         this.closeBtnState = this.closeBtn.getStyle('display');
53525         this.closeBtn.hide();
53526         if(this.stickBtn){
53527             this.stickBtn.show();
53528         }
53529         this.el.show();
53530         this.el.alignTo(this.collapsedEl, this.getCollapseAnchor());
53531         this.beforeSlide();
53532         this.el.setStyle("z-index", 10001);
53533         this.el.slideIn(this.getSlideAnchor(), {
53534             callback: function(){
53535                 this.afterSlide();
53536                 this.initAutoHide();
53537                 Roo.get(document).on("click", this.slideInIf, this);
53538                 this.fireEvent("slideshow", this);
53539             },
53540             scope: this,
53541             block: true
53542         });
53543     },
53544
53545     afterSlideIn : function(){
53546         this.clearAutoHide();
53547         this.isSlid = false;
53548         this.clearMonitor();
53549         this.el.setStyle("z-index", "");
53550         if(this.collapseBtn){
53551             this.collapseBtn.show();
53552         }
53553         this.closeBtn.setStyle('display', this.closeBtnState);
53554         if(this.stickBtn){
53555             this.stickBtn.hide();
53556         }
53557         this.fireEvent("slidehide", this);
53558     },
53559
53560     slideIn : function(cb){
53561         if(!this.isSlid || this.el.hasActiveFx()){
53562             Roo.callback(cb);
53563             return;
53564         }
53565         this.isSlid = false;
53566         this.beforeSlide();
53567         this.el.slideOut(this.getSlideAnchor(), {
53568             callback: function(){
53569                 this.el.setLeftTop(-10000, -10000);
53570                 this.afterSlide();
53571                 this.afterSlideIn();
53572                 Roo.callback(cb);
53573             },
53574             scope: this,
53575             block: true
53576         });
53577     },
53578     
53579     slideInIf : function(e){
53580         if(!e.within(this.el)){
53581             this.slideIn();
53582         }
53583     },
53584
53585     animateCollapse : function(){
53586         this.beforeSlide();
53587         this.el.setStyle("z-index", 20000);
53588         var anchor = this.getSlideAnchor();
53589         this.el.slideOut(anchor, {
53590             callback : function(){
53591                 this.el.setStyle("z-index", "");
53592                 this.collapsedEl.slideIn(anchor, {duration:.3});
53593                 this.afterSlide();
53594                 this.el.setLocation(-10000,-10000);
53595                 this.el.hide();
53596                 this.fireEvent("collapsed", this);
53597             },
53598             scope: this,
53599             block: true
53600         });
53601     },
53602
53603     animateExpand : function(){
53604         this.beforeSlide();
53605         this.el.alignTo(this.collapsedEl, this.getCollapseAnchor(), this.getExpandAdj());
53606         this.el.setStyle("z-index", 20000);
53607         this.collapsedEl.hide({
53608             duration:.1
53609         });
53610         this.el.slideIn(this.getSlideAnchor(), {
53611             callback : function(){
53612                 this.el.setStyle("z-index", "");
53613                 this.afterSlide();
53614                 if(this.split){
53615                     this.split.el.show();
53616                 }
53617                 this.fireEvent("invalidated", this);
53618                 this.fireEvent("expanded", this);
53619             },
53620             scope: this,
53621             block: true
53622         });
53623     },
53624
53625     anchors : {
53626         "west" : "left",
53627         "east" : "right",
53628         "north" : "top",
53629         "south" : "bottom"
53630     },
53631
53632     sanchors : {
53633         "west" : "l",
53634         "east" : "r",
53635         "north" : "t",
53636         "south" : "b"
53637     },
53638
53639     canchors : {
53640         "west" : "tl-tr",
53641         "east" : "tr-tl",
53642         "north" : "tl-bl",
53643         "south" : "bl-tl"
53644     },
53645
53646     getAnchor : function(){
53647         return this.anchors[this.position];
53648     },
53649
53650     getCollapseAnchor : function(){
53651         return this.canchors[this.position];
53652     },
53653
53654     getSlideAnchor : function(){
53655         return this.sanchors[this.position];
53656     },
53657
53658     getAlignAdj : function(){
53659         var cm = this.cmargins;
53660         switch(this.position){
53661             case "west":
53662                 return [0, 0];
53663             break;
53664             case "east":
53665                 return [0, 0];
53666             break;
53667             case "north":
53668                 return [0, 0];
53669             break;
53670             case "south":
53671                 return [0, 0];
53672             break;
53673         }
53674     },
53675
53676     getExpandAdj : function(){
53677         var c = this.collapsedEl, cm = this.cmargins;
53678         switch(this.position){
53679             case "west":
53680                 return [-(cm.right+c.getWidth()+cm.left), 0];
53681             break;
53682             case "east":
53683                 return [cm.right+c.getWidth()+cm.left, 0];
53684             break;
53685             case "north":
53686                 return [0, -(cm.top+cm.bottom+c.getHeight())];
53687             break;
53688             case "south":
53689                 return [0, cm.top+cm.bottom+c.getHeight()];
53690             break;
53691         }
53692     }
53693 });/*
53694  * Based on:
53695  * Ext JS Library 1.1.1
53696  * Copyright(c) 2006-2007, Ext JS, LLC.
53697  *
53698  * Originally Released Under LGPL - original licence link has changed is not relivant.
53699  *
53700  * Fork - LGPL
53701  * <script type="text/javascript">
53702  */
53703 /*
53704  * These classes are private internal classes
53705  */
53706 Roo.CenterLayoutRegion = function(mgr, config){
53707     Roo.LayoutRegion.call(this, mgr, config, "center");
53708     this.visible = true;
53709     this.minWidth = config.minWidth || 20;
53710     this.minHeight = config.minHeight || 20;
53711 };
53712
53713 Roo.extend(Roo.CenterLayoutRegion, Roo.LayoutRegion, {
53714     hide : function(){
53715         // center panel can't be hidden
53716     },
53717     
53718     show : function(){
53719         // center panel can't be hidden
53720     },
53721     
53722     getMinWidth: function(){
53723         return this.minWidth;
53724     },
53725     
53726     getMinHeight: function(){
53727         return this.minHeight;
53728     }
53729 });
53730
53731
53732 Roo.NorthLayoutRegion = function(mgr, config){
53733     Roo.LayoutRegion.call(this, mgr, config, "north", "n-resize");
53734     if(this.split){
53735         this.split.placement = Roo.SplitBar.TOP;
53736         this.split.orientation = Roo.SplitBar.VERTICAL;
53737         this.split.el.addClass("x-layout-split-v");
53738     }
53739     var size = config.initialSize || config.height;
53740     if(typeof size != "undefined"){
53741         this.el.setHeight(size);
53742     }
53743 };
53744 Roo.extend(Roo.NorthLayoutRegion, Roo.SplitLayoutRegion, {
53745     orientation: Roo.SplitBar.VERTICAL,
53746     getBox : function(){
53747         if(this.collapsed){
53748             return this.collapsedEl.getBox();
53749         }
53750         var box = this.el.getBox();
53751         if(this.split){
53752             box.height += this.split.el.getHeight();
53753         }
53754         return box;
53755     },
53756     
53757     updateBox : function(box){
53758         if(this.split && !this.collapsed){
53759             box.height -= this.split.el.getHeight();
53760             this.split.el.setLeft(box.x);
53761             this.split.el.setTop(box.y+box.height);
53762             this.split.el.setWidth(box.width);
53763         }
53764         if(this.collapsed){
53765             this.updateBody(box.width, null);
53766         }
53767         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53768     }
53769 });
53770
53771 Roo.SouthLayoutRegion = function(mgr, config){
53772     Roo.SplitLayoutRegion.call(this, mgr, config, "south", "s-resize");
53773     if(this.split){
53774         this.split.placement = Roo.SplitBar.BOTTOM;
53775         this.split.orientation = Roo.SplitBar.VERTICAL;
53776         this.split.el.addClass("x-layout-split-v");
53777     }
53778     var size = config.initialSize || config.height;
53779     if(typeof size != "undefined"){
53780         this.el.setHeight(size);
53781     }
53782 };
53783 Roo.extend(Roo.SouthLayoutRegion, Roo.SplitLayoutRegion, {
53784     orientation: Roo.SplitBar.VERTICAL,
53785     getBox : function(){
53786         if(this.collapsed){
53787             return this.collapsedEl.getBox();
53788         }
53789         var box = this.el.getBox();
53790         if(this.split){
53791             var sh = this.split.el.getHeight();
53792             box.height += sh;
53793             box.y -= sh;
53794         }
53795         return box;
53796     },
53797     
53798     updateBox : function(box){
53799         if(this.split && !this.collapsed){
53800             var sh = this.split.el.getHeight();
53801             box.height -= sh;
53802             box.y += sh;
53803             this.split.el.setLeft(box.x);
53804             this.split.el.setTop(box.y-sh);
53805             this.split.el.setWidth(box.width);
53806         }
53807         if(this.collapsed){
53808             this.updateBody(box.width, null);
53809         }
53810         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53811     }
53812 });
53813
53814 Roo.EastLayoutRegion = function(mgr, config){
53815     Roo.SplitLayoutRegion.call(this, mgr, config, "east", "e-resize");
53816     if(this.split){
53817         this.split.placement = Roo.SplitBar.RIGHT;
53818         this.split.orientation = Roo.SplitBar.HORIZONTAL;
53819         this.split.el.addClass("x-layout-split-h");
53820     }
53821     var size = config.initialSize || config.width;
53822     if(typeof size != "undefined"){
53823         this.el.setWidth(size);
53824     }
53825 };
53826 Roo.extend(Roo.EastLayoutRegion, Roo.SplitLayoutRegion, {
53827     orientation: Roo.SplitBar.HORIZONTAL,
53828     getBox : function(){
53829         if(this.collapsed){
53830             return this.collapsedEl.getBox();
53831         }
53832         var box = this.el.getBox();
53833         if(this.split){
53834             var sw = this.split.el.getWidth();
53835             box.width += sw;
53836             box.x -= sw;
53837         }
53838         return box;
53839     },
53840
53841     updateBox : function(box){
53842         if(this.split && !this.collapsed){
53843             var sw = this.split.el.getWidth();
53844             box.width -= sw;
53845             this.split.el.setLeft(box.x);
53846             this.split.el.setTop(box.y);
53847             this.split.el.setHeight(box.height);
53848             box.x += sw;
53849         }
53850         if(this.collapsed){
53851             this.updateBody(null, box.height);
53852         }
53853         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53854     }
53855 });
53856
53857 Roo.WestLayoutRegion = function(mgr, config){
53858     Roo.SplitLayoutRegion.call(this, mgr, config, "west", "w-resize");
53859     if(this.split){
53860         this.split.placement = Roo.SplitBar.LEFT;
53861         this.split.orientation = Roo.SplitBar.HORIZONTAL;
53862         this.split.el.addClass("x-layout-split-h");
53863     }
53864     var size = config.initialSize || config.width;
53865     if(typeof size != "undefined"){
53866         this.el.setWidth(size);
53867     }
53868 };
53869 Roo.extend(Roo.WestLayoutRegion, Roo.SplitLayoutRegion, {
53870     orientation: Roo.SplitBar.HORIZONTAL,
53871     getBox : function(){
53872         if(this.collapsed){
53873             return this.collapsedEl.getBox();
53874         }
53875         var box = this.el.getBox();
53876         if(this.split){
53877             box.width += this.split.el.getWidth();
53878         }
53879         return box;
53880     },
53881     
53882     updateBox : function(box){
53883         if(this.split && !this.collapsed){
53884             var sw = this.split.el.getWidth();
53885             box.width -= sw;
53886             this.split.el.setLeft(box.x+box.width);
53887             this.split.el.setTop(box.y);
53888             this.split.el.setHeight(box.height);
53889         }
53890         if(this.collapsed){
53891             this.updateBody(null, box.height);
53892         }
53893         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53894     }
53895 });
53896 /*
53897  * Based on:
53898  * Ext JS Library 1.1.1
53899  * Copyright(c) 2006-2007, Ext JS, LLC.
53900  *
53901  * Originally Released Under LGPL - original licence link has changed is not relivant.
53902  *
53903  * Fork - LGPL
53904  * <script type="text/javascript">
53905  */
53906  
53907  
53908 /*
53909  * Private internal class for reading and applying state
53910  */
53911 Roo.LayoutStateManager = function(layout){
53912      // default empty state
53913      this.state = {
53914         north: {},
53915         south: {},
53916         east: {},
53917         west: {}       
53918     };
53919 };
53920
53921 Roo.LayoutStateManager.prototype = {
53922     init : function(layout, provider){
53923         this.provider = provider;
53924         var state = provider.get(layout.id+"-layout-state");
53925         if(state){
53926             var wasUpdating = layout.isUpdating();
53927             if(!wasUpdating){
53928                 layout.beginUpdate();
53929             }
53930             for(var key in state){
53931                 if(typeof state[key] != "function"){
53932                     var rstate = state[key];
53933                     var r = layout.getRegion(key);
53934                     if(r && rstate){
53935                         if(rstate.size){
53936                             r.resizeTo(rstate.size);
53937                         }
53938                         if(rstate.collapsed == true){
53939                             r.collapse(true);
53940                         }else{
53941                             r.expand(null, true);
53942                         }
53943                     }
53944                 }
53945             }
53946             if(!wasUpdating){
53947                 layout.endUpdate();
53948             }
53949             this.state = state; 
53950         }
53951         this.layout = layout;
53952         layout.on("regionresized", this.onRegionResized, this);
53953         layout.on("regioncollapsed", this.onRegionCollapsed, this);
53954         layout.on("regionexpanded", this.onRegionExpanded, this);
53955     },
53956     
53957     storeState : function(){
53958         this.provider.set(this.layout.id+"-layout-state", this.state);
53959     },
53960     
53961     onRegionResized : function(region, newSize){
53962         this.state[region.getPosition()].size = newSize;
53963         this.storeState();
53964     },
53965     
53966     onRegionCollapsed : function(region){
53967         this.state[region.getPosition()].collapsed = true;
53968         this.storeState();
53969     },
53970     
53971     onRegionExpanded : function(region){
53972         this.state[region.getPosition()].collapsed = false;
53973         this.storeState();
53974     }
53975 };/*
53976  * Based on:
53977  * Ext JS Library 1.1.1
53978  * Copyright(c) 2006-2007, Ext JS, LLC.
53979  *
53980  * Originally Released Under LGPL - original licence link has changed is not relivant.
53981  *
53982  * Fork - LGPL
53983  * <script type="text/javascript">
53984  */
53985 /**
53986  * @class Roo.ContentPanel
53987  * @extends Roo.util.Observable
53988  * A basic ContentPanel element.
53989  * @cfg {Boolean}   fitToFrame    True for this panel to adjust its size to fit when the region resizes  (defaults to false)
53990  * @cfg {Boolean}   fitContainer   When using {@link #fitToFrame} and {@link #resizeEl}, you can also fit the parent container  (defaults to false)
53991  * @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
53992  * @cfg {Boolean}   closable      True if the panel can be closed/removed
53993  * @cfg {Boolean}   background    True if the panel should not be activated when it is added (defaults to false)
53994  * @cfg {String/HTMLElement/Element} resizeEl An element to resize if {@link #fitToFrame} is true (instead of this panel's element)
53995  * @cfg {Toolbar}   toolbar       A toolbar for this panel
53996  * @cfg {Boolean} autoScroll    True to scroll overflow in this panel (use with {@link #fitToFrame})
53997  * @cfg {String} title          The title for this panel
53998  * @cfg {Array} adjustments     Values to <b>add</b> to the width/height when doing a {@link #fitToFrame} (default is [0, 0])
53999  * @cfg {String} url            Calls {@link #setUrl} with this value
54000  * @cfg {String} region         (center|north|south|east|west) which region to put this panel on (when used with xtype constructors)
54001  * @cfg {String/Object} params  When used with {@link #url}, calls {@link #setUrl} with this value
54002  * @cfg {Boolean} loadOnce      When used with {@link #url}, calls {@link #setUrl} with this value
54003  * @cfg {String}    content        Raw content to fill content panel with (uses setContent on construction.)
54004
54005  * @constructor
54006  * Create a new ContentPanel.
54007  * @param {String/HTMLElement/Roo.Element} el The container element for this panel
54008  * @param {String/Object} config A string to set only the title or a config object
54009  * @param {String} content (optional) Set the HTML content for this panel
54010  * @param {String} region (optional) Used by xtype constructors to add to regions. (values center,east,west,south,north)
54011  */
54012 Roo.ContentPanel = function(el, config, content){
54013     
54014      
54015     /*
54016     if(el.autoCreate || el.xtype){ // xtype is available if this is called from factory
54017         config = el;
54018         el = Roo.id();
54019     }
54020     if (config && config.parentLayout) { 
54021         el = config.parentLayout.el.createChild(); 
54022     }
54023     */
54024     if(el.autoCreate){ // xtype is available if this is called from factory
54025         config = el;
54026         el = Roo.id();
54027     }
54028     this.el = Roo.get(el);
54029     if(!this.el && config && config.autoCreate){
54030         if(typeof config.autoCreate == "object"){
54031             if(!config.autoCreate.id){
54032                 config.autoCreate.id = config.id||el;
54033             }
54034             this.el = Roo.DomHelper.append(document.body,
54035                         config.autoCreate, true);
54036         }else{
54037             this.el = Roo.DomHelper.append(document.body,
54038                         {tag: "div", cls: "x-layout-inactive-content", id: config.id||el}, true);
54039         }
54040     }
54041     this.closable = false;
54042     this.loaded = false;
54043     this.active = false;
54044     if(typeof config == "string"){
54045         this.title = config;
54046     }else{
54047         Roo.apply(this, config);
54048     }
54049     
54050     if (this.toolbar && !this.toolbar.el && this.toolbar.xtype) {
54051         this.wrapEl = this.el.wrap();
54052         this.toolbar.container = this.el.insertSibling(false, 'before');
54053         this.toolbar = new Roo.Toolbar(this.toolbar);
54054     }
54055     
54056     // xtype created footer. - not sure if will work as we normally have to render first..
54057     if (this.footer && !this.footer.el && this.footer.xtype) {
54058         if (!this.wrapEl) {
54059             this.wrapEl = this.el.wrap();
54060         }
54061     
54062         this.footer.container = this.wrapEl.createChild();
54063          
54064         this.footer = Roo.factory(this.footer, Roo);
54065         
54066     }
54067     
54068     if(this.resizeEl){
54069         this.resizeEl = Roo.get(this.resizeEl, true);
54070     }else{
54071         this.resizeEl = this.el;
54072     }
54073     // handle view.xtype
54074     
54075  
54076     
54077     
54078     this.addEvents({
54079         /**
54080          * @event activate
54081          * Fires when this panel is activated. 
54082          * @param {Roo.ContentPanel} this
54083          */
54084         "activate" : true,
54085         /**
54086          * @event deactivate
54087          * Fires when this panel is activated. 
54088          * @param {Roo.ContentPanel} this
54089          */
54090         "deactivate" : true,
54091
54092         /**
54093          * @event resize
54094          * Fires when this panel is resized if fitToFrame is true.
54095          * @param {Roo.ContentPanel} this
54096          * @param {Number} width The width after any component adjustments
54097          * @param {Number} height The height after any component adjustments
54098          */
54099         "resize" : true,
54100         
54101          /**
54102          * @event render
54103          * Fires when this tab is created
54104          * @param {Roo.ContentPanel} this
54105          */
54106         "render" : true
54107          
54108         
54109     });
54110     
54111
54112     
54113     
54114     if(this.autoScroll){
54115         this.resizeEl.setStyle("overflow", "auto");
54116     } else {
54117         // fix randome scrolling
54118         this.el.on('scroll', function() {
54119             Roo.log('fix random scolling');
54120             this.scrollTo('top',0); 
54121         });
54122     }
54123     content = content || this.content;
54124     if(content){
54125         this.setContent(content);
54126     }
54127     if(config && config.url){
54128         this.setUrl(this.url, this.params, this.loadOnce);
54129     }
54130     
54131     
54132     
54133     Roo.ContentPanel.superclass.constructor.call(this);
54134     
54135     if (this.view && typeof(this.view.xtype) != 'undefined') {
54136         this.view.el = this.el.appendChild(document.createElement("div"));
54137         this.view = Roo.factory(this.view); 
54138         this.view.render  &&  this.view.render(false, '');  
54139     }
54140     
54141     
54142     this.fireEvent('render', this);
54143 };
54144
54145 Roo.extend(Roo.ContentPanel, Roo.util.Observable, {
54146     tabTip:'',
54147     setRegion : function(region){
54148         this.region = region;
54149         if(region){
54150            this.el.replaceClass("x-layout-inactive-content", "x-layout-active-content");
54151         }else{
54152            this.el.replaceClass("x-layout-active-content", "x-layout-inactive-content");
54153         } 
54154     },
54155     
54156     /**
54157      * Returns the toolbar for this Panel if one was configured. 
54158      * @return {Roo.Toolbar} 
54159      */
54160     getToolbar : function(){
54161         return this.toolbar;
54162     },
54163     
54164     setActiveState : function(active){
54165         this.active = active;
54166         if(!active){
54167             this.fireEvent("deactivate", this);
54168         }else{
54169             this.fireEvent("activate", this);
54170         }
54171     },
54172     /**
54173      * Updates this panel's element
54174      * @param {String} content The new content
54175      * @param {Boolean} loadScripts (optional) true to look for and process scripts
54176     */
54177     setContent : function(content, loadScripts){
54178         this.el.update(content, loadScripts);
54179     },
54180
54181     ignoreResize : function(w, h){
54182         if(this.lastSize && this.lastSize.width == w && this.lastSize.height == h){
54183             return true;
54184         }else{
54185             this.lastSize = {width: w, height: h};
54186             return false;
54187         }
54188     },
54189     /**
54190      * Get the {@link Roo.UpdateManager} for this panel. Enables you to perform Ajax updates.
54191      * @return {Roo.UpdateManager} The UpdateManager
54192      */
54193     getUpdateManager : function(){
54194         return this.el.getUpdateManager();
54195     },
54196      /**
54197      * Loads this content panel immediately with content from XHR. Note: to delay loading until the panel is activated, use {@link #setUrl}.
54198      * @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:
54199 <pre><code>
54200 panel.load({
54201     url: "your-url.php",
54202     params: {param1: "foo", param2: "bar"}, // or a URL encoded string
54203     callback: yourFunction,
54204     scope: yourObject, //(optional scope)
54205     discardUrl: false,
54206     nocache: false,
54207     text: "Loading...",
54208     timeout: 30,
54209     scripts: false
54210 });
54211 </code></pre>
54212      * The only required property is <i>url</i>. The optional properties <i>nocache</i>, <i>text</i> and <i>scripts</i>
54213      * 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.
54214      * @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}
54215      * @param {Function} callback (optional) Callback when transaction is complete -- called with signature (oElement, bSuccess, oResponse)
54216      * @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.
54217      * @return {Roo.ContentPanel} this
54218      */
54219     load : function(){
54220         var um = this.el.getUpdateManager();
54221         um.update.apply(um, arguments);
54222         return this;
54223     },
54224
54225
54226     /**
54227      * 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.
54228      * @param {String/Function} url The URL to load the content from or a function to call to get the URL
54229      * @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)
54230      * @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)
54231      * @return {Roo.UpdateManager} The UpdateManager
54232      */
54233     setUrl : function(url, params, loadOnce){
54234         if(this.refreshDelegate){
54235             this.removeListener("activate", this.refreshDelegate);
54236         }
54237         this.refreshDelegate = this._handleRefresh.createDelegate(this, [url, params, loadOnce]);
54238         this.on("activate", this.refreshDelegate);
54239         return this.el.getUpdateManager();
54240     },
54241     
54242     _handleRefresh : function(url, params, loadOnce){
54243         if(!loadOnce || !this.loaded){
54244             var updater = this.el.getUpdateManager();
54245             updater.update(url, params, this._setLoaded.createDelegate(this));
54246         }
54247     },
54248     
54249     _setLoaded : function(){
54250         this.loaded = true;
54251     }, 
54252     
54253     /**
54254      * Returns this panel's id
54255      * @return {String} 
54256      */
54257     getId : function(){
54258         return this.el.id;
54259     },
54260     
54261     /** 
54262      * Returns this panel's element - used by regiosn to add.
54263      * @return {Roo.Element} 
54264      */
54265     getEl : function(){
54266         return this.wrapEl || this.el;
54267     },
54268     
54269     adjustForComponents : function(width, height)
54270     {
54271         //Roo.log('adjustForComponents ');
54272         if(this.resizeEl != this.el){
54273             width -= this.el.getFrameWidth('lr');
54274             height -= this.el.getFrameWidth('tb');
54275         }
54276         if(this.toolbar){
54277             var te = this.toolbar.getEl();
54278             height -= te.getHeight();
54279             te.setWidth(width);
54280         }
54281         if(this.footer){
54282             var te = this.footer.getEl();
54283             //Roo.log("footer:" + te.getHeight());
54284             
54285             height -= te.getHeight();
54286             te.setWidth(width);
54287         }
54288         
54289         
54290         if(this.adjustments){
54291             width += this.adjustments[0];
54292             height += this.adjustments[1];
54293         }
54294         return {"width": width, "height": height};
54295     },
54296     
54297     setSize : function(width, height){
54298         if(this.fitToFrame && !this.ignoreResize(width, height)){
54299             if(this.fitContainer && this.resizeEl != this.el){
54300                 this.el.setSize(width, height);
54301             }
54302             var size = this.adjustForComponents(width, height);
54303             this.resizeEl.setSize(this.autoWidth ? "auto" : size.width, this.autoHeight ? "auto" : size.height);
54304             this.fireEvent('resize', this, size.width, size.height);
54305         }
54306     },
54307     
54308     /**
54309      * Returns this panel's title
54310      * @return {String} 
54311      */
54312     getTitle : function(){
54313         return this.title;
54314     },
54315     
54316     /**
54317      * Set this panel's title
54318      * @param {String} title
54319      */
54320     setTitle : function(title){
54321         this.title = title;
54322         if(this.region){
54323             this.region.updatePanelTitle(this, title);
54324         }
54325     },
54326     
54327     /**
54328      * Returns true is this panel was configured to be closable
54329      * @return {Boolean} 
54330      */
54331     isClosable : function(){
54332         return this.closable;
54333     },
54334     
54335     beforeSlide : function(){
54336         this.el.clip();
54337         this.resizeEl.clip();
54338     },
54339     
54340     afterSlide : function(){
54341         this.el.unclip();
54342         this.resizeEl.unclip();
54343     },
54344     
54345     /**
54346      *   Force a content refresh from the URL specified in the {@link #setUrl} method.
54347      *   Will fail silently if the {@link #setUrl} method has not been called.
54348      *   This does not activate the panel, just updates its content.
54349      */
54350     refresh : function(){
54351         if(this.refreshDelegate){
54352            this.loaded = false;
54353            this.refreshDelegate();
54354         }
54355     },
54356     
54357     /**
54358      * Destroys this panel
54359      */
54360     destroy : function(){
54361         this.el.removeAllListeners();
54362         var tempEl = document.createElement("span");
54363         tempEl.appendChild(this.el.dom);
54364         tempEl.innerHTML = "";
54365         this.el.remove();
54366         this.el = null;
54367     },
54368     
54369     /**
54370      * form - if the content panel contains a form - this is a reference to it.
54371      * @type {Roo.form.Form}
54372      */
54373     form : false,
54374     /**
54375      * view - if the content panel contains a view (Roo.DatePicker / Roo.View / Roo.JsonView)
54376      *    This contains a reference to it.
54377      * @type {Roo.View}
54378      */
54379     view : false,
54380     
54381       /**
54382      * Adds a xtype elements to the panel - currently only supports Forms, View, JsonView.
54383      * <pre><code>
54384
54385 layout.addxtype({
54386        xtype : 'Form',
54387        items: [ .... ]
54388    }
54389 );
54390
54391 </code></pre>
54392      * @param {Object} cfg Xtype definition of item to add.
54393      */
54394     
54395     addxtype : function(cfg) {
54396         // add form..
54397         if (cfg.xtype.match(/^Form$/)) {
54398             
54399             var el;
54400             //if (this.footer) {
54401             //    el = this.footer.container.insertSibling(false, 'before');
54402             //} else {
54403                 el = this.el.createChild();
54404             //}
54405
54406             this.form = new  Roo.form.Form(cfg);
54407             
54408             
54409             if ( this.form.allItems.length) {
54410                 this.form.render(el.dom);
54411             }
54412             return this.form;
54413         }
54414         // should only have one of theses..
54415         if ([ 'View', 'JsonView', 'DatePicker'].indexOf(cfg.xtype) > -1) {
54416             // views.. should not be just added - used named prop 'view''
54417             
54418             cfg.el = this.el.appendChild(document.createElement("div"));
54419             // factory?
54420             
54421             var ret = new Roo.factory(cfg);
54422              
54423              ret.render && ret.render(false, ''); // render blank..
54424             this.view = ret;
54425             return ret;
54426         }
54427         return false;
54428     }
54429 });
54430
54431 /**
54432  * @class Roo.GridPanel
54433  * @extends Roo.ContentPanel
54434  * @constructor
54435  * Create a new GridPanel.
54436  * @param {Roo.grid.Grid} grid The grid for this panel
54437  * @param {String/Object} config A string to set only the panel's title, or a config object
54438  */
54439 Roo.GridPanel = function(grid, config){
54440     
54441   
54442     this.wrapper = Roo.DomHelper.append(document.body, // wrapper for IE7 strict & safari scroll issue
54443         {tag: "div", cls: "x-layout-grid-wrapper x-layout-inactive-content"}, true);
54444         
54445     this.wrapper.dom.appendChild(grid.getGridEl().dom);
54446     
54447     Roo.GridPanel.superclass.constructor.call(this, this.wrapper, config);
54448     
54449     if(this.toolbar){
54450         this.toolbar.el.insertBefore(this.wrapper.dom.firstChild);
54451     }
54452     // xtype created footer. - not sure if will work as we normally have to render first..
54453     if (this.footer && !this.footer.el && this.footer.xtype) {
54454         
54455         this.footer.container = this.grid.getView().getFooterPanel(true);
54456         this.footer.dataSource = this.grid.dataSource;
54457         this.footer = Roo.factory(this.footer, Roo);
54458         
54459     }
54460     
54461     grid.monitorWindowResize = false; // turn off autosizing
54462     grid.autoHeight = false;
54463     grid.autoWidth = false;
54464     this.grid = grid;
54465     this.grid.getGridEl().replaceClass("x-layout-inactive-content", "x-layout-component-panel");
54466 };
54467
54468 Roo.extend(Roo.GridPanel, Roo.ContentPanel, {
54469     getId : function(){
54470         return this.grid.id;
54471     },
54472     
54473     /**
54474      * Returns the grid for this panel
54475      * @return {Roo.grid.Grid} 
54476      */
54477     getGrid : function(){
54478         return this.grid;    
54479     },
54480     
54481     setSize : function(width, height){
54482         if(!this.ignoreResize(width, height)){
54483             var grid = this.grid;
54484             var size = this.adjustForComponents(width, height);
54485             grid.getGridEl().setSize(size.width, size.height);
54486             grid.autoSize();
54487         }
54488     },
54489     
54490     beforeSlide : function(){
54491         this.grid.getView().scroller.clip();
54492     },
54493     
54494     afterSlide : function(){
54495         this.grid.getView().scroller.unclip();
54496     },
54497     
54498     destroy : function(){
54499         this.grid.destroy();
54500         delete this.grid;
54501         Roo.GridPanel.superclass.destroy.call(this); 
54502     }
54503 });
54504
54505
54506 /**
54507  * @class Roo.NestedLayoutPanel
54508  * @extends Roo.ContentPanel
54509  * @constructor
54510  * Create a new NestedLayoutPanel.
54511  * 
54512  * 
54513  * @param {Roo.BorderLayout} layout The layout for this panel
54514  * @param {String/Object} config A string to set only the title or a config object
54515  */
54516 Roo.NestedLayoutPanel = function(layout, config)
54517 {
54518     // construct with only one argument..
54519     /* FIXME - implement nicer consturctors
54520     if (layout.layout) {
54521         config = layout;
54522         layout = config.layout;
54523         delete config.layout;
54524     }
54525     if (layout.xtype && !layout.getEl) {
54526         // then layout needs constructing..
54527         layout = Roo.factory(layout, Roo);
54528     }
54529     */
54530     
54531     
54532     Roo.NestedLayoutPanel.superclass.constructor.call(this, layout.getEl(), config);
54533     
54534     layout.monitorWindowResize = false; // turn off autosizing
54535     this.layout = layout;
54536     this.layout.getEl().addClass("x-layout-nested-layout");
54537     
54538     
54539     
54540     
54541 };
54542
54543 Roo.extend(Roo.NestedLayoutPanel, Roo.ContentPanel, {
54544
54545     setSize : function(width, height){
54546         if(!this.ignoreResize(width, height)){
54547             var size = this.adjustForComponents(width, height);
54548             var el = this.layout.getEl();
54549             el.setSize(size.width, size.height);
54550             var touch = el.dom.offsetWidth;
54551             this.layout.layout();
54552             // ie requires a double layout on the first pass
54553             if(Roo.isIE && !this.initialized){
54554                 this.initialized = true;
54555                 this.layout.layout();
54556             }
54557         }
54558     },
54559     
54560     // activate all subpanels if not currently active..
54561     
54562     setActiveState : function(active){
54563         this.active = active;
54564         if(!active){
54565             this.fireEvent("deactivate", this);
54566             return;
54567         }
54568         
54569         this.fireEvent("activate", this);
54570         // not sure if this should happen before or after..
54571         if (!this.layout) {
54572             return; // should not happen..
54573         }
54574         var reg = false;
54575         for (var r in this.layout.regions) {
54576             reg = this.layout.getRegion(r);
54577             if (reg.getActivePanel()) {
54578                 //reg.showPanel(reg.getActivePanel()); // force it to activate.. 
54579                 reg.setActivePanel(reg.getActivePanel());
54580                 continue;
54581             }
54582             if (!reg.panels.length) {
54583                 continue;
54584             }
54585             reg.showPanel(reg.getPanel(0));
54586         }
54587         
54588         
54589         
54590         
54591     },
54592     
54593     /**
54594      * Returns the nested BorderLayout for this panel
54595      * @return {Roo.BorderLayout} 
54596      */
54597     getLayout : function(){
54598         return this.layout;
54599     },
54600     
54601      /**
54602      * Adds a xtype elements to the layout of the nested panel
54603      * <pre><code>
54604
54605 panel.addxtype({
54606        xtype : 'ContentPanel',
54607        region: 'west',
54608        items: [ .... ]
54609    }
54610 );
54611
54612 panel.addxtype({
54613         xtype : 'NestedLayoutPanel',
54614         region: 'west',
54615         layout: {
54616            center: { },
54617            west: { }   
54618         },
54619         items : [ ... list of content panels or nested layout panels.. ]
54620    }
54621 );
54622 </code></pre>
54623      * @param {Object} cfg Xtype definition of item to add.
54624      */
54625     addxtype : function(cfg) {
54626         return this.layout.addxtype(cfg);
54627     
54628     }
54629 });
54630
54631 Roo.ScrollPanel = function(el, config, content){
54632     config = config || {};
54633     config.fitToFrame = true;
54634     Roo.ScrollPanel.superclass.constructor.call(this, el, config, content);
54635     
54636     this.el.dom.style.overflow = "hidden";
54637     var wrap = this.el.wrap({cls: "x-scroller x-layout-inactive-content"});
54638     this.el.removeClass("x-layout-inactive-content");
54639     this.el.on("mousewheel", this.onWheel, this);
54640
54641     var up = wrap.createChild({cls: "x-scroller-up", html: "&#160;"}, this.el.dom);
54642     var down = wrap.createChild({cls: "x-scroller-down", html: "&#160;"});
54643     up.unselectable(); down.unselectable();
54644     up.on("click", this.scrollUp, this);
54645     down.on("click", this.scrollDown, this);
54646     up.addClassOnOver("x-scroller-btn-over");
54647     down.addClassOnOver("x-scroller-btn-over");
54648     up.addClassOnClick("x-scroller-btn-click");
54649     down.addClassOnClick("x-scroller-btn-click");
54650     this.adjustments = [0, -(up.getHeight() + down.getHeight())];
54651
54652     this.resizeEl = this.el;
54653     this.el = wrap; this.up = up; this.down = down;
54654 };
54655
54656 Roo.extend(Roo.ScrollPanel, Roo.ContentPanel, {
54657     increment : 100,
54658     wheelIncrement : 5,
54659     scrollUp : function(){
54660         this.resizeEl.scroll("up", this.increment, {callback: this.afterScroll, scope: this});
54661     },
54662
54663     scrollDown : function(){
54664         this.resizeEl.scroll("down", this.increment, {callback: this.afterScroll, scope: this});
54665     },
54666
54667     afterScroll : function(){
54668         var el = this.resizeEl;
54669         var t = el.dom.scrollTop, h = el.dom.scrollHeight, ch = el.dom.clientHeight;
54670         this.up[t == 0 ? "addClass" : "removeClass"]("x-scroller-btn-disabled");
54671         this.down[h - t <= ch ? "addClass" : "removeClass"]("x-scroller-btn-disabled");
54672     },
54673
54674     setSize : function(){
54675         Roo.ScrollPanel.superclass.setSize.apply(this, arguments);
54676         this.afterScroll();
54677     },
54678
54679     onWheel : function(e){
54680         var d = e.getWheelDelta();
54681         this.resizeEl.dom.scrollTop -= (d*this.wheelIncrement);
54682         this.afterScroll();
54683         e.stopEvent();
54684     },
54685
54686     setContent : function(content, loadScripts){
54687         this.resizeEl.update(content, loadScripts);
54688     }
54689
54690 });
54691
54692
54693
54694
54695
54696
54697
54698
54699
54700 /**
54701  * @class Roo.TreePanel
54702  * @extends Roo.ContentPanel
54703  * @constructor
54704  * Create a new TreePanel. - defaults to fit/scoll contents.
54705  * @param {String/Object} config A string to set only the panel's title, or a config object
54706  * @cfg {Roo.tree.TreePanel} tree The tree TreePanel, with config etc.
54707  */
54708 Roo.TreePanel = function(config){
54709     var el = config.el;
54710     var tree = config.tree;
54711     delete config.tree; 
54712     delete config.el; // hopefull!
54713     
54714     // wrapper for IE7 strict & safari scroll issue
54715     
54716     var treeEl = el.createChild();
54717     config.resizeEl = treeEl;
54718     
54719     
54720     
54721     Roo.TreePanel.superclass.constructor.call(this, el, config);
54722  
54723  
54724     this.tree = new Roo.tree.TreePanel(treeEl , tree);
54725     //console.log(tree);
54726     this.on('activate', function()
54727     {
54728         if (this.tree.rendered) {
54729             return;
54730         }
54731         //console.log('render tree');
54732         this.tree.render();
54733     });
54734     // this should not be needed.. - it's actually the 'el' that resizes?
54735     // actuall it breaks the containerScroll - dragging nodes auto scroll at top
54736     
54737     //this.on('resize',  function (cp, w, h) {
54738     //        this.tree.innerCt.setWidth(w);
54739     //        this.tree.innerCt.setHeight(h);
54740     //        //this.tree.innerCt.setStyle('overflow-y', 'auto');
54741     //});
54742
54743         
54744     
54745 };
54746
54747 Roo.extend(Roo.TreePanel, Roo.ContentPanel, {   
54748     fitToFrame : true,
54749     autoScroll : true
54750 });
54751
54752
54753
54754
54755
54756
54757
54758
54759
54760
54761
54762 /*
54763  * Based on:
54764  * Ext JS Library 1.1.1
54765  * Copyright(c) 2006-2007, Ext JS, LLC.
54766  *
54767  * Originally Released Under LGPL - original licence link has changed is not relivant.
54768  *
54769  * Fork - LGPL
54770  * <script type="text/javascript">
54771  */
54772  
54773
54774 /**
54775  * @class Roo.ReaderLayout
54776  * @extends Roo.BorderLayout
54777  * This is a pre-built layout that represents a classic, 5-pane application.  It consists of a header, a primary
54778  * center region containing two nested regions (a top one for a list view and one for item preview below),
54779  * and regions on either side that can be used for navigation, application commands, informational displays, etc.
54780  * The setup and configuration work exactly the same as it does for a {@link Roo.BorderLayout} - this class simply
54781  * expedites the setup of the overall layout and regions for this common application style.
54782  * Example:
54783  <pre><code>
54784 var reader = new Roo.ReaderLayout();
54785 var CP = Roo.ContentPanel;  // shortcut for adding
54786
54787 reader.beginUpdate();
54788 reader.add("north", new CP("north", "North"));
54789 reader.add("west", new CP("west", {title: "West"}));
54790 reader.add("east", new CP("east", {title: "East"}));
54791
54792 reader.regions.listView.add(new CP("listView", "List"));
54793 reader.regions.preview.add(new CP("preview", "Preview"));
54794 reader.endUpdate();
54795 </code></pre>
54796 * @constructor
54797 * Create a new ReaderLayout
54798 * @param {Object} config Configuration options
54799 * @param {String/HTMLElement/Element} container (optional) The container this layout is bound to (defaults to
54800 * document.body if omitted)
54801 */
54802 Roo.ReaderLayout = function(config, renderTo){
54803     var c = config || {size:{}};
54804     Roo.ReaderLayout.superclass.constructor.call(this, renderTo || document.body, {
54805         north: c.north !== false ? Roo.apply({
54806             split:false,
54807             initialSize: 32,
54808             titlebar: false
54809         }, c.north) : false,
54810         west: c.west !== false ? Roo.apply({
54811             split:true,
54812             initialSize: 200,
54813             minSize: 175,
54814             maxSize: 400,
54815             titlebar: true,
54816             collapsible: true,
54817             animate: true,
54818             margins:{left:5,right:0,bottom:5,top:5},
54819             cmargins:{left:5,right:5,bottom:5,top:5}
54820         }, c.west) : false,
54821         east: c.east !== false ? Roo.apply({
54822             split:true,
54823             initialSize: 200,
54824             minSize: 175,
54825             maxSize: 400,
54826             titlebar: true,
54827             collapsible: true,
54828             animate: true,
54829             margins:{left:0,right:5,bottom:5,top:5},
54830             cmargins:{left:5,right:5,bottom:5,top:5}
54831         }, c.east) : false,
54832         center: Roo.apply({
54833             tabPosition: 'top',
54834             autoScroll:false,
54835             closeOnTab: true,
54836             titlebar:false,
54837             margins:{left:c.west!==false ? 0 : 5,right:c.east!==false ? 0 : 5,bottom:5,top:2}
54838         }, c.center)
54839     });
54840
54841     this.el.addClass('x-reader');
54842
54843     this.beginUpdate();
54844
54845     var inner = new Roo.BorderLayout(Roo.get(document.body).createChild(), {
54846         south: c.preview !== false ? Roo.apply({
54847             split:true,
54848             initialSize: 200,
54849             minSize: 100,
54850             autoScroll:true,
54851             collapsible:true,
54852             titlebar: true,
54853             cmargins:{top:5,left:0, right:0, bottom:0}
54854         }, c.preview) : false,
54855         center: Roo.apply({
54856             autoScroll:false,
54857             titlebar:false,
54858             minHeight:200
54859         }, c.listView)
54860     });
54861     this.add('center', new Roo.NestedLayoutPanel(inner,
54862             Roo.apply({title: c.mainTitle || '',tabTip:''},c.innerPanelCfg)));
54863
54864     this.endUpdate();
54865
54866     this.regions.preview = inner.getRegion('south');
54867     this.regions.listView = inner.getRegion('center');
54868 };
54869
54870 Roo.extend(Roo.ReaderLayout, Roo.BorderLayout);/*
54871  * Based on:
54872  * Ext JS Library 1.1.1
54873  * Copyright(c) 2006-2007, Ext JS, LLC.
54874  *
54875  * Originally Released Under LGPL - original licence link has changed is not relivant.
54876  *
54877  * Fork - LGPL
54878  * <script type="text/javascript">
54879  */
54880  
54881 /**
54882  * @class Roo.grid.Grid
54883  * @extends Roo.util.Observable
54884  * This class represents the primary interface of a component based grid control.
54885  * <br><br>Usage:<pre><code>
54886  var grid = new Roo.grid.Grid("my-container-id", {
54887      ds: myDataStore,
54888      cm: myColModel,
54889      selModel: mySelectionModel,
54890      autoSizeColumns: true,
54891      monitorWindowResize: false,
54892      trackMouseOver: true
54893  });
54894  // set any options
54895  grid.render();
54896  * </code></pre>
54897  * <b>Common Problems:</b><br/>
54898  * - Grid does not resize properly when going smaller: Setting overflow hidden on the container
54899  * element will correct this<br/>
54900  * - If you get el.style[camel]= NaNpx or -2px or something related, be certain you have given your container element
54901  * dimensions. The grid adapts to your container's size, if your container has no size defined then the results
54902  * are unpredictable.<br/>
54903  * - Do not render the grid into an element with display:none. Try using visibility:hidden. Otherwise there is no way for the
54904  * grid to calculate dimensions/offsets.<br/>
54905   * @constructor
54906  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
54907  * The container MUST have some type of size defined for the grid to fill. The container will be
54908  * automatically set to position relative if it isn't already.
54909  * @param {Object} config A config object that sets properties on this grid.
54910  */
54911 Roo.grid.Grid = function(container, config){
54912         // initialize the container
54913         this.container = Roo.get(container);
54914         this.container.update("");
54915         this.container.setStyle("overflow", "hidden");
54916     this.container.addClass('x-grid-container');
54917
54918     this.id = this.container.id;
54919
54920     Roo.apply(this, config);
54921     // check and correct shorthanded configs
54922     if(this.ds){
54923         this.dataSource = this.ds;
54924         delete this.ds;
54925     }
54926     if(this.cm){
54927         this.colModel = this.cm;
54928         delete this.cm;
54929     }
54930     if(this.sm){
54931         this.selModel = this.sm;
54932         delete this.sm;
54933     }
54934
54935     if (this.selModel) {
54936         this.selModel = Roo.factory(this.selModel, Roo.grid);
54937         this.sm = this.selModel;
54938         this.sm.xmodule = this.xmodule || false;
54939     }
54940     if (typeof(this.colModel.config) == 'undefined') {
54941         this.colModel = new Roo.grid.ColumnModel(this.colModel);
54942         this.cm = this.colModel;
54943         this.cm.xmodule = this.xmodule || false;
54944     }
54945     if (this.dataSource) {
54946         this.dataSource= Roo.factory(this.dataSource, Roo.data);
54947         this.ds = this.dataSource;
54948         this.ds.xmodule = this.xmodule || false;
54949          
54950     }
54951     
54952     
54953     
54954     if(this.width){
54955         this.container.setWidth(this.width);
54956     }
54957
54958     if(this.height){
54959         this.container.setHeight(this.height);
54960     }
54961     /** @private */
54962         this.addEvents({
54963         // raw events
54964         /**
54965          * @event click
54966          * The raw click event for the entire grid.
54967          * @param {Roo.EventObject} e
54968          */
54969         "click" : true,
54970         /**
54971          * @event dblclick
54972          * The raw dblclick event for the entire grid.
54973          * @param {Roo.EventObject} e
54974          */
54975         "dblclick" : true,
54976         /**
54977          * @event contextmenu
54978          * The raw contextmenu event for the entire grid.
54979          * @param {Roo.EventObject} e
54980          */
54981         "contextmenu" : true,
54982         /**
54983          * @event mousedown
54984          * The raw mousedown event for the entire grid.
54985          * @param {Roo.EventObject} e
54986          */
54987         "mousedown" : true,
54988         /**
54989          * @event mouseup
54990          * The raw mouseup event for the entire grid.
54991          * @param {Roo.EventObject} e
54992          */
54993         "mouseup" : true,
54994         /**
54995          * @event mouseover
54996          * The raw mouseover event for the entire grid.
54997          * @param {Roo.EventObject} e
54998          */
54999         "mouseover" : true,
55000         /**
55001          * @event mouseout
55002          * The raw mouseout event for the entire grid.
55003          * @param {Roo.EventObject} e
55004          */
55005         "mouseout" : true,
55006         /**
55007          * @event keypress
55008          * The raw keypress event for the entire grid.
55009          * @param {Roo.EventObject} e
55010          */
55011         "keypress" : true,
55012         /**
55013          * @event keydown
55014          * The raw keydown event for the entire grid.
55015          * @param {Roo.EventObject} e
55016          */
55017         "keydown" : true,
55018
55019         // custom events
55020
55021         /**
55022          * @event cellclick
55023          * Fires when a cell is clicked
55024          * @param {Grid} this
55025          * @param {Number} rowIndex
55026          * @param {Number} columnIndex
55027          * @param {Roo.EventObject} e
55028          */
55029         "cellclick" : true,
55030         /**
55031          * @event celldblclick
55032          * Fires when a cell is double clicked
55033          * @param {Grid} this
55034          * @param {Number} rowIndex
55035          * @param {Number} columnIndex
55036          * @param {Roo.EventObject} e
55037          */
55038         "celldblclick" : true,
55039         /**
55040          * @event rowclick
55041          * Fires when a row is clicked
55042          * @param {Grid} this
55043          * @param {Number} rowIndex
55044          * @param {Roo.EventObject} e
55045          */
55046         "rowclick" : true,
55047         /**
55048          * @event rowdblclick
55049          * Fires when a row is double clicked
55050          * @param {Grid} this
55051          * @param {Number} rowIndex
55052          * @param {Roo.EventObject} e
55053          */
55054         "rowdblclick" : true,
55055         /**
55056          * @event headerclick
55057          * Fires when a header is clicked
55058          * @param {Grid} this
55059          * @param {Number} columnIndex
55060          * @param {Roo.EventObject} e
55061          */
55062         "headerclick" : true,
55063         /**
55064          * @event headerdblclick
55065          * Fires when a header cell is double clicked
55066          * @param {Grid} this
55067          * @param {Number} columnIndex
55068          * @param {Roo.EventObject} e
55069          */
55070         "headerdblclick" : true,
55071         /**
55072          * @event rowcontextmenu
55073          * Fires when a row is right clicked
55074          * @param {Grid} this
55075          * @param {Number} rowIndex
55076          * @param {Roo.EventObject} e
55077          */
55078         "rowcontextmenu" : true,
55079         /**
55080          * @event cellcontextmenu
55081          * Fires when a cell is right clicked
55082          * @param {Grid} this
55083          * @param {Number} rowIndex
55084          * @param {Number} cellIndex
55085          * @param {Roo.EventObject} e
55086          */
55087          "cellcontextmenu" : true,
55088         /**
55089          * @event headercontextmenu
55090          * Fires when a header is right clicked
55091          * @param {Grid} this
55092          * @param {Number} columnIndex
55093          * @param {Roo.EventObject} e
55094          */
55095         "headercontextmenu" : true,
55096         /**
55097          * @event bodyscroll
55098          * Fires when the body element is scrolled
55099          * @param {Number} scrollLeft
55100          * @param {Number} scrollTop
55101          */
55102         "bodyscroll" : true,
55103         /**
55104          * @event columnresize
55105          * Fires when the user resizes a column
55106          * @param {Number} columnIndex
55107          * @param {Number} newSize
55108          */
55109         "columnresize" : true,
55110         /**
55111          * @event columnmove
55112          * Fires when the user moves a column
55113          * @param {Number} oldIndex
55114          * @param {Number} newIndex
55115          */
55116         "columnmove" : true,
55117         /**
55118          * @event startdrag
55119          * Fires when row(s) start being dragged
55120          * @param {Grid} this
55121          * @param {Roo.GridDD} dd The drag drop object
55122          * @param {event} e The raw browser event
55123          */
55124         "startdrag" : true,
55125         /**
55126          * @event enddrag
55127          * Fires when a drag operation is complete
55128          * @param {Grid} this
55129          * @param {Roo.GridDD} dd The drag drop object
55130          * @param {event} e The raw browser event
55131          */
55132         "enddrag" : true,
55133         /**
55134          * @event dragdrop
55135          * Fires when dragged row(s) are dropped on a valid DD target
55136          * @param {Grid} this
55137          * @param {Roo.GridDD} dd The drag drop object
55138          * @param {String} targetId The target drag drop object
55139          * @param {event} e The raw browser event
55140          */
55141         "dragdrop" : true,
55142         /**
55143          * @event dragover
55144          * Fires while row(s) are being dragged. "targetId" is the id of the Yahoo.util.DD object the selected rows are being dragged over.
55145          * @param {Grid} this
55146          * @param {Roo.GridDD} dd The drag drop object
55147          * @param {String} targetId The target drag drop object
55148          * @param {event} e The raw browser event
55149          */
55150         "dragover" : true,
55151         /**
55152          * @event dragenter
55153          *  Fires when the dragged row(s) first cross another DD target while being dragged
55154          * @param {Grid} this
55155          * @param {Roo.GridDD} dd The drag drop object
55156          * @param {String} targetId The target drag drop object
55157          * @param {event} e The raw browser event
55158          */
55159         "dragenter" : true,
55160         /**
55161          * @event dragout
55162          * Fires when the dragged row(s) leave another DD target while being dragged
55163          * @param {Grid} this
55164          * @param {Roo.GridDD} dd The drag drop object
55165          * @param {String} targetId The target drag drop object
55166          * @param {event} e The raw browser event
55167          */
55168         "dragout" : true,
55169         /**
55170          * @event rowclass
55171          * Fires when a row is rendered, so you can change add a style to it.
55172          * @param {GridView} gridview   The grid view
55173          * @param {Object} rowcfg   contains record  rowIndex and rowClass - set rowClass to add a style.
55174          */
55175         'rowclass' : true,
55176
55177         /**
55178          * @event render
55179          * Fires when the grid is rendered
55180          * @param {Grid} grid
55181          */
55182         'render' : true
55183     });
55184
55185     Roo.grid.Grid.superclass.constructor.call(this);
55186 };
55187 Roo.extend(Roo.grid.Grid, Roo.util.Observable, {
55188     
55189     /**
55190      * @cfg {String} ddGroup - drag drop group.
55191      */
55192
55193     /**
55194      * @cfg {Number} minColumnWidth The minimum width a column can be resized to. Default is 25.
55195      */
55196     minColumnWidth : 25,
55197
55198     /**
55199      * @cfg {Boolean} autoSizeColumns True to automatically resize the columns to fit their content
55200      * <b>on initial render.</b> It is more efficient to explicitly size the columns
55201      * through the ColumnModel's {@link Roo.grid.ColumnModel#width} config option.  Default is false.
55202      */
55203     autoSizeColumns : false,
55204
55205     /**
55206      * @cfg {Boolean} autoSizeHeaders True to measure headers with column data when auto sizing columns. Default is true.
55207      */
55208     autoSizeHeaders : true,
55209
55210     /**
55211      * @cfg {Boolean} monitorWindowResize True to autoSize the grid when the window resizes. Default is true.
55212      */
55213     monitorWindowResize : true,
55214
55215     /**
55216      * @cfg {Boolean} maxRowsToMeasure If autoSizeColumns is on, maxRowsToMeasure can be used to limit the number of
55217      * rows measured to get a columns size. Default is 0 (all rows).
55218      */
55219     maxRowsToMeasure : 0,
55220
55221     /**
55222      * @cfg {Boolean} trackMouseOver True to highlight rows when the mouse is over. Default is true.
55223      */
55224     trackMouseOver : true,
55225
55226     /**
55227     * @cfg {Boolean} enableDrag  True to enable drag of rows. Default is false. (double check if this is needed?)
55228     */
55229     
55230     /**
55231     * @cfg {Boolean} enableDragDrop True to enable drag and drop of rows. Default is false.
55232     */
55233     enableDragDrop : false,
55234     
55235     /**
55236     * @cfg {Boolean} enableColumnMove True to enable drag and drop reorder of columns. Default is true.
55237     */
55238     enableColumnMove : true,
55239     
55240     /**
55241     * @cfg {Boolean} enableColumnHide True to enable hiding of columns with the header context menu. Default is true.
55242     */
55243     enableColumnHide : true,
55244     
55245     /**
55246     * @cfg {Boolean} enableRowHeightSync True to manually sync row heights across locked and not locked rows. Default is false.
55247     */
55248     enableRowHeightSync : false,
55249     
55250     /**
55251     * @cfg {Boolean} stripeRows True to stripe the rows.  Default is true.
55252     */
55253     stripeRows : true,
55254     
55255     /**
55256     * @cfg {Boolean} autoHeight True to fit the height of the grid container to the height of the data. Default is false.
55257     */
55258     autoHeight : false,
55259
55260     /**
55261      * @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.
55262      */
55263     autoExpandColumn : false,
55264
55265     /**
55266     * @cfg {Number} autoExpandMin The minimum width the autoExpandColumn can have (if enabled).
55267     * Default is 50.
55268     */
55269     autoExpandMin : 50,
55270
55271     /**
55272     * @cfg {Number} autoExpandMax The maximum width the autoExpandColumn can have (if enabled). Default is 1000.
55273     */
55274     autoExpandMax : 1000,
55275
55276     /**
55277     * @cfg {Object} view The {@link Roo.grid.GridView} used by the grid. This can be set before a call to render().
55278     */
55279     view : null,
55280
55281     /**
55282     * @cfg {Object} loadMask An {@link Roo.LoadMask} config or true to mask the grid while loading. Default is false.
55283     */
55284     loadMask : false,
55285     /**
55286     * @cfg {Roo.dd.DropTarget} dropTarget An {@link Roo.dd.DropTarget} config
55287     */
55288     dropTarget: false,
55289     
55290    
55291     
55292     // private
55293     rendered : false,
55294
55295     /**
55296     * @cfg {Boolean} autoWidth True to set the grid's width to the default total width of the grid's columns instead
55297     * of a fixed width. Default is false.
55298     */
55299     /**
55300     * @cfg {Number} maxHeight Sets the maximum height of the grid - ignored if autoHeight is not on.
55301     */
55302     /**
55303      * Called once after all setup has been completed and the grid is ready to be rendered.
55304      * @return {Roo.grid.Grid} this
55305      */
55306     render : function()
55307     {
55308         var c = this.container;
55309         // try to detect autoHeight/width mode
55310         if((!c.dom.offsetHeight || c.dom.offsetHeight < 20) || c.getStyle("height") == "auto"){
55311             this.autoHeight = true;
55312         }
55313         var view = this.getView();
55314         view.init(this);
55315
55316         c.on("click", this.onClick, this);
55317         c.on("dblclick", this.onDblClick, this);
55318         c.on("contextmenu", this.onContextMenu, this);
55319         c.on("keydown", this.onKeyDown, this);
55320         if (Roo.isTouch) {
55321             c.on("touchstart", this.onTouchStart, this);
55322         }
55323
55324         this.relayEvents(c, ["mousedown","mouseup","mouseover","mouseout","keypress"]);
55325
55326         this.getSelectionModel().init(this);
55327
55328         view.render();
55329
55330         if(this.loadMask){
55331             this.loadMask = new Roo.LoadMask(this.container,
55332                     Roo.apply({store:this.dataSource}, this.loadMask));
55333         }
55334         
55335         
55336         if (this.toolbar && this.toolbar.xtype) {
55337             this.toolbar.container = this.getView().getHeaderPanel(true);
55338             this.toolbar = new Roo.Toolbar(this.toolbar);
55339         }
55340         if (this.footer && this.footer.xtype) {
55341             this.footer.dataSource = this.getDataSource();
55342             this.footer.container = this.getView().getFooterPanel(true);
55343             this.footer = Roo.factory(this.footer, Roo);
55344         }
55345         if (this.dropTarget && this.dropTarget.xtype) {
55346             delete this.dropTarget.xtype;
55347             this.dropTarget =  new Roo.dd.DropTarget(this.getView().mainBody, this.dropTarget);
55348         }
55349         
55350         
55351         this.rendered = true;
55352         this.fireEvent('render', this);
55353         return this;
55354     },
55355
55356         /**
55357          * Reconfigures the grid to use a different Store and Column Model.
55358          * The View will be bound to the new objects and refreshed.
55359          * @param {Roo.data.Store} dataSource The new {@link Roo.data.Store} object
55360          * @param {Roo.grid.ColumnModel} The new {@link Roo.grid.ColumnModel} object
55361          */
55362     reconfigure : function(dataSource, colModel){
55363         if(this.loadMask){
55364             this.loadMask.destroy();
55365             this.loadMask = new Roo.LoadMask(this.container,
55366                     Roo.apply({store:dataSource}, this.loadMask));
55367         }
55368         this.view.bind(dataSource, colModel);
55369         this.dataSource = dataSource;
55370         this.colModel = colModel;
55371         this.view.refresh(true);
55372     },
55373
55374     // private
55375     onKeyDown : function(e){
55376         this.fireEvent("keydown", e);
55377     },
55378
55379     /**
55380      * Destroy this grid.
55381      * @param {Boolean} removeEl True to remove the element
55382      */
55383     destroy : function(removeEl, keepListeners){
55384         if(this.loadMask){
55385             this.loadMask.destroy();
55386         }
55387         var c = this.container;
55388         c.removeAllListeners();
55389         this.view.destroy();
55390         this.colModel.purgeListeners();
55391         if(!keepListeners){
55392             this.purgeListeners();
55393         }
55394         c.update("");
55395         if(removeEl === true){
55396             c.remove();
55397         }
55398     },
55399
55400     // private
55401     processEvent : function(name, e){
55402         // does this fire select???
55403         //Roo.log('grid:processEvent '  + name);
55404         
55405         if (name != 'touchstart' ) {
55406             this.fireEvent(name, e);    
55407         }
55408         
55409         var t = e.getTarget();
55410         var v = this.view;
55411         var header = v.findHeaderIndex(t);
55412         if(header !== false){
55413             var ename = name == 'touchstart' ? 'click' : name;
55414              
55415             this.fireEvent("header" + ename, this, header, e);
55416         }else{
55417             var row = v.findRowIndex(t);
55418             var cell = v.findCellIndex(t);
55419             if (name == 'touchstart') {
55420                 // first touch is always a click.
55421                 // hopefull this happens after selection is updated.?
55422                 name = false;
55423                 
55424                 if (typeof(this.selModel.getSelectedCell) != 'undefined') {
55425                     var cs = this.selModel.getSelectedCell();
55426                     if (row == cs[0] && cell == cs[1]){
55427                         name = 'dblclick';
55428                     }
55429                 }
55430                 if (typeof(this.selModel.getSelections) != 'undefined') {
55431                     var cs = this.selModel.getSelections();
55432                     var ds = this.dataSource;
55433                     if (cs.length == 1 && ds.getAt(row) == cs[0]){
55434                         name = 'dblclick';
55435                     }
55436                 }
55437                 if (!name) {
55438                     return;
55439                 }
55440             }
55441             
55442             
55443             if(row !== false){
55444                 this.fireEvent("row" + name, this, row, e);
55445                 if(cell !== false){
55446                     this.fireEvent("cell" + name, this, row, cell, e);
55447                 }
55448             }
55449         }
55450     },
55451
55452     // private
55453     onClick : function(e){
55454         this.processEvent("click", e);
55455     },
55456    // private
55457     onTouchStart : function(e){
55458         this.processEvent("touchstart", e);
55459     },
55460
55461     // private
55462     onContextMenu : function(e, t){
55463         this.processEvent("contextmenu", e);
55464     },
55465
55466     // private
55467     onDblClick : function(e){
55468         this.processEvent("dblclick", e);
55469     },
55470
55471     // private
55472     walkCells : function(row, col, step, fn, scope){
55473         var cm = this.colModel, clen = cm.getColumnCount();
55474         var ds = this.dataSource, rlen = ds.getCount(), first = true;
55475         if(step < 0){
55476             if(col < 0){
55477                 row--;
55478                 first = false;
55479             }
55480             while(row >= 0){
55481                 if(!first){
55482                     col = clen-1;
55483                 }
55484                 first = false;
55485                 while(col >= 0){
55486                     if(fn.call(scope || this, row, col, cm) === true){
55487                         return [row, col];
55488                     }
55489                     col--;
55490                 }
55491                 row--;
55492             }
55493         } else {
55494             if(col >= clen){
55495                 row++;
55496                 first = false;
55497             }
55498             while(row < rlen){
55499                 if(!first){
55500                     col = 0;
55501                 }
55502                 first = false;
55503                 while(col < clen){
55504                     if(fn.call(scope || this, row, col, cm) === true){
55505                         return [row, col];
55506                     }
55507                     col++;
55508                 }
55509                 row++;
55510             }
55511         }
55512         return null;
55513     },
55514
55515     // private
55516     getSelections : function(){
55517         return this.selModel.getSelections();
55518     },
55519
55520     /**
55521      * Causes the grid to manually recalculate its dimensions. Generally this is done automatically,
55522      * but if manual update is required this method will initiate it.
55523      */
55524     autoSize : function(){
55525         if(this.rendered){
55526             this.view.layout();
55527             if(this.view.adjustForScroll){
55528                 this.view.adjustForScroll();
55529             }
55530         }
55531     },
55532
55533     /**
55534      * Returns the grid's underlying element.
55535      * @return {Element} The element
55536      */
55537     getGridEl : function(){
55538         return this.container;
55539     },
55540
55541     // private for compatibility, overridden by editor grid
55542     stopEditing : function(){},
55543
55544     /**
55545      * Returns the grid's SelectionModel.
55546      * @return {SelectionModel}
55547      */
55548     getSelectionModel : function(){
55549         if(!this.selModel){
55550             this.selModel = new Roo.grid.RowSelectionModel();
55551         }
55552         return this.selModel;
55553     },
55554
55555     /**
55556      * Returns the grid's DataSource.
55557      * @return {DataSource}
55558      */
55559     getDataSource : function(){
55560         return this.dataSource;
55561     },
55562
55563     /**
55564      * Returns the grid's ColumnModel.
55565      * @return {ColumnModel}
55566      */
55567     getColumnModel : function(){
55568         return this.colModel;
55569     },
55570
55571     /**
55572      * Returns the grid's GridView object.
55573      * @return {GridView}
55574      */
55575     getView : function(){
55576         if(!this.view){
55577             this.view = new Roo.grid.GridView(this.viewConfig);
55578         }
55579         return this.view;
55580     },
55581     /**
55582      * Called to get grid's drag proxy text, by default returns this.ddText.
55583      * @return {String}
55584      */
55585     getDragDropText : function(){
55586         var count = this.selModel.getCount();
55587         return String.format(this.ddText, count, count == 1 ? '' : 's');
55588     }
55589 });
55590 /**
55591  * Configures the text is the drag proxy (defaults to "%0 selected row(s)").
55592  * %0 is replaced with the number of selected rows.
55593  * @type String
55594  */
55595 Roo.grid.Grid.prototype.ddText = "{0} selected row{1}";/*
55596  * Based on:
55597  * Ext JS Library 1.1.1
55598  * Copyright(c) 2006-2007, Ext JS, LLC.
55599  *
55600  * Originally Released Under LGPL - original licence link has changed is not relivant.
55601  *
55602  * Fork - LGPL
55603  * <script type="text/javascript">
55604  */
55605  
55606 Roo.grid.AbstractGridView = function(){
55607         this.grid = null;
55608         
55609         this.events = {
55610             "beforerowremoved" : true,
55611             "beforerowsinserted" : true,
55612             "beforerefresh" : true,
55613             "rowremoved" : true,
55614             "rowsinserted" : true,
55615             "rowupdated" : true,
55616             "refresh" : true
55617         };
55618     Roo.grid.AbstractGridView.superclass.constructor.call(this);
55619 };
55620
55621 Roo.extend(Roo.grid.AbstractGridView, Roo.util.Observable, {
55622     rowClass : "x-grid-row",
55623     cellClass : "x-grid-cell",
55624     tdClass : "x-grid-td",
55625     hdClass : "x-grid-hd",
55626     splitClass : "x-grid-hd-split",
55627     
55628     init: function(grid){
55629         this.grid = grid;
55630                 var cid = this.grid.getGridEl().id;
55631         this.colSelector = "#" + cid + " ." + this.cellClass + "-";
55632         this.tdSelector = "#" + cid + " ." + this.tdClass + "-";
55633         this.hdSelector = "#" + cid + " ." + this.hdClass + "-";
55634         this.splitSelector = "#" + cid + " ." + this.splitClass + "-";
55635         },
55636         
55637     getColumnRenderers : function(){
55638         var renderers = [];
55639         var cm = this.grid.colModel;
55640         var colCount = cm.getColumnCount();
55641         for(var i = 0; i < colCount; i++){
55642             renderers[i] = cm.getRenderer(i);
55643         }
55644         return renderers;
55645     },
55646     
55647     getColumnIds : function(){
55648         var ids = [];
55649         var cm = this.grid.colModel;
55650         var colCount = cm.getColumnCount();
55651         for(var i = 0; i < colCount; i++){
55652             ids[i] = cm.getColumnId(i);
55653         }
55654         return ids;
55655     },
55656     
55657     getDataIndexes : function(){
55658         if(!this.indexMap){
55659             this.indexMap = this.buildIndexMap();
55660         }
55661         return this.indexMap.colToData;
55662     },
55663     
55664     getColumnIndexByDataIndex : function(dataIndex){
55665         if(!this.indexMap){
55666             this.indexMap = this.buildIndexMap();
55667         }
55668         return this.indexMap.dataToCol[dataIndex];
55669     },
55670     
55671     /**
55672      * Set a css style for a column dynamically. 
55673      * @param {Number} colIndex The index of the column
55674      * @param {String} name The css property name
55675      * @param {String} value The css value
55676      */
55677     setCSSStyle : function(colIndex, name, value){
55678         var selector = "#" + this.grid.id + " .x-grid-col-" + colIndex;
55679         Roo.util.CSS.updateRule(selector, name, value);
55680     },
55681     
55682     generateRules : function(cm){
55683         var ruleBuf = [], rulesId = this.grid.id + '-cssrules';
55684         Roo.util.CSS.removeStyleSheet(rulesId);
55685         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
55686             var cid = cm.getColumnId(i);
55687             ruleBuf.push(this.colSelector, cid, " {\n", cm.config[i].css, "}\n",
55688                          this.tdSelector, cid, " {\n}\n",
55689                          this.hdSelector, cid, " {\n}\n",
55690                          this.splitSelector, cid, " {\n}\n");
55691         }
55692         return Roo.util.CSS.createStyleSheet(ruleBuf.join(""), rulesId);
55693     }
55694 });/*
55695  * Based on:
55696  * Ext JS Library 1.1.1
55697  * Copyright(c) 2006-2007, Ext JS, LLC.
55698  *
55699  * Originally Released Under LGPL - original licence link has changed is not relivant.
55700  *
55701  * Fork - LGPL
55702  * <script type="text/javascript">
55703  */
55704
55705 // private
55706 // This is a support class used internally by the Grid components
55707 Roo.grid.HeaderDragZone = function(grid, hd, hd2){
55708     this.grid = grid;
55709     this.view = grid.getView();
55710     this.ddGroup = "gridHeader" + this.grid.getGridEl().id;
55711     Roo.grid.HeaderDragZone.superclass.constructor.call(this, hd);
55712     if(hd2){
55713         this.setHandleElId(Roo.id(hd));
55714         this.setOuterHandleElId(Roo.id(hd2));
55715     }
55716     this.scroll = false;
55717 };
55718 Roo.extend(Roo.grid.HeaderDragZone, Roo.dd.DragZone, {
55719     maxDragWidth: 120,
55720     getDragData : function(e){
55721         var t = Roo.lib.Event.getTarget(e);
55722         var h = this.view.findHeaderCell(t);
55723         if(h){
55724             return {ddel: h.firstChild, header:h};
55725         }
55726         return false;
55727     },
55728
55729     onInitDrag : function(e){
55730         this.view.headersDisabled = true;
55731         var clone = this.dragData.ddel.cloneNode(true);
55732         clone.id = Roo.id();
55733         clone.style.width = Math.min(this.dragData.header.offsetWidth,this.maxDragWidth) + "px";
55734         this.proxy.update(clone);
55735         return true;
55736     },
55737
55738     afterValidDrop : function(){
55739         var v = this.view;
55740         setTimeout(function(){
55741             v.headersDisabled = false;
55742         }, 50);
55743     },
55744
55745     afterInvalidDrop : function(){
55746         var v = this.view;
55747         setTimeout(function(){
55748             v.headersDisabled = false;
55749         }, 50);
55750     }
55751 });
55752 /*
55753  * Based on:
55754  * Ext JS Library 1.1.1
55755  * Copyright(c) 2006-2007, Ext JS, LLC.
55756  *
55757  * Originally Released Under LGPL - original licence link has changed is not relivant.
55758  *
55759  * Fork - LGPL
55760  * <script type="text/javascript">
55761  */
55762 // private
55763 // This is a support class used internally by the Grid components
55764 Roo.grid.HeaderDropZone = function(grid, hd, hd2){
55765     this.grid = grid;
55766     this.view = grid.getView();
55767     // split the proxies so they don't interfere with mouse events
55768     this.proxyTop = Roo.DomHelper.append(document.body, {
55769         cls:"col-move-top", html:"&#160;"
55770     }, true);
55771     this.proxyBottom = Roo.DomHelper.append(document.body, {
55772         cls:"col-move-bottom", html:"&#160;"
55773     }, true);
55774     this.proxyTop.hide = this.proxyBottom.hide = function(){
55775         this.setLeftTop(-100,-100);
55776         this.setStyle("visibility", "hidden");
55777     };
55778     this.ddGroup = "gridHeader" + this.grid.getGridEl().id;
55779     // temporarily disabled
55780     //Roo.dd.ScrollManager.register(this.view.scroller.dom);
55781     Roo.grid.HeaderDropZone.superclass.constructor.call(this, grid.getGridEl().dom);
55782 };
55783 Roo.extend(Roo.grid.HeaderDropZone, Roo.dd.DropZone, {
55784     proxyOffsets : [-4, -9],
55785     fly: Roo.Element.fly,
55786
55787     getTargetFromEvent : function(e){
55788         var t = Roo.lib.Event.getTarget(e);
55789         var cindex = this.view.findCellIndex(t);
55790         if(cindex !== false){
55791             return this.view.getHeaderCell(cindex);
55792         }
55793         return null;
55794     },
55795
55796     nextVisible : function(h){
55797         var v = this.view, cm = this.grid.colModel;
55798         h = h.nextSibling;
55799         while(h){
55800             if(!cm.isHidden(v.getCellIndex(h))){
55801                 return h;
55802             }
55803             h = h.nextSibling;
55804         }
55805         return null;
55806     },
55807
55808     prevVisible : function(h){
55809         var v = this.view, cm = this.grid.colModel;
55810         h = h.prevSibling;
55811         while(h){
55812             if(!cm.isHidden(v.getCellIndex(h))){
55813                 return h;
55814             }
55815             h = h.prevSibling;
55816         }
55817         return null;
55818     },
55819
55820     positionIndicator : function(h, n, e){
55821         var x = Roo.lib.Event.getPageX(e);
55822         var r = Roo.lib.Dom.getRegion(n.firstChild);
55823         var px, pt, py = r.top + this.proxyOffsets[1];
55824         if((r.right - x) <= (r.right-r.left)/2){
55825             px = r.right+this.view.borderWidth;
55826             pt = "after";
55827         }else{
55828             px = r.left;
55829             pt = "before";
55830         }
55831         var oldIndex = this.view.getCellIndex(h);
55832         var newIndex = this.view.getCellIndex(n);
55833
55834         if(this.grid.colModel.isFixed(newIndex)){
55835             return false;
55836         }
55837
55838         var locked = this.grid.colModel.isLocked(newIndex);
55839
55840         if(pt == "after"){
55841             newIndex++;
55842         }
55843         if(oldIndex < newIndex){
55844             newIndex--;
55845         }
55846         if(oldIndex == newIndex && (locked == this.grid.colModel.isLocked(oldIndex))){
55847             return false;
55848         }
55849         px +=  this.proxyOffsets[0];
55850         this.proxyTop.setLeftTop(px, py);
55851         this.proxyTop.show();
55852         if(!this.bottomOffset){
55853             this.bottomOffset = this.view.mainHd.getHeight();
55854         }
55855         this.proxyBottom.setLeftTop(px, py+this.proxyTop.dom.offsetHeight+this.bottomOffset);
55856         this.proxyBottom.show();
55857         return pt;
55858     },
55859
55860     onNodeEnter : function(n, dd, e, data){
55861         if(data.header != n){
55862             this.positionIndicator(data.header, n, e);
55863         }
55864     },
55865
55866     onNodeOver : function(n, dd, e, data){
55867         var result = false;
55868         if(data.header != n){
55869             result = this.positionIndicator(data.header, n, e);
55870         }
55871         if(!result){
55872             this.proxyTop.hide();
55873             this.proxyBottom.hide();
55874         }
55875         return result ? this.dropAllowed : this.dropNotAllowed;
55876     },
55877
55878     onNodeOut : function(n, dd, e, data){
55879         this.proxyTop.hide();
55880         this.proxyBottom.hide();
55881     },
55882
55883     onNodeDrop : function(n, dd, e, data){
55884         var h = data.header;
55885         if(h != n){
55886             var cm = this.grid.colModel;
55887             var x = Roo.lib.Event.getPageX(e);
55888             var r = Roo.lib.Dom.getRegion(n.firstChild);
55889             var pt = (r.right - x) <= ((r.right-r.left)/2) ? "after" : "before";
55890             var oldIndex = this.view.getCellIndex(h);
55891             var newIndex = this.view.getCellIndex(n);
55892             var locked = cm.isLocked(newIndex);
55893             if(pt == "after"){
55894                 newIndex++;
55895             }
55896             if(oldIndex < newIndex){
55897                 newIndex--;
55898             }
55899             if(oldIndex == newIndex && (locked == cm.isLocked(oldIndex))){
55900                 return false;
55901             }
55902             cm.setLocked(oldIndex, locked, true);
55903             cm.moveColumn(oldIndex, newIndex);
55904             this.grid.fireEvent("columnmove", oldIndex, newIndex);
55905             return true;
55906         }
55907         return false;
55908     }
55909 });
55910 /*
55911  * Based on:
55912  * Ext JS Library 1.1.1
55913  * Copyright(c) 2006-2007, Ext JS, LLC.
55914  *
55915  * Originally Released Under LGPL - original licence link has changed is not relivant.
55916  *
55917  * Fork - LGPL
55918  * <script type="text/javascript">
55919  */
55920   
55921 /**
55922  * @class Roo.grid.GridView
55923  * @extends Roo.util.Observable
55924  *
55925  * @constructor
55926  * @param {Object} config
55927  */
55928 Roo.grid.GridView = function(config){
55929     Roo.grid.GridView.superclass.constructor.call(this);
55930     this.el = null;
55931
55932     Roo.apply(this, config);
55933 };
55934
55935 Roo.extend(Roo.grid.GridView, Roo.grid.AbstractGridView, {
55936
55937     unselectable :  'unselectable="on"',
55938     unselectableCls :  'x-unselectable',
55939     
55940     
55941     rowClass : "x-grid-row",
55942
55943     cellClass : "x-grid-col",
55944
55945     tdClass : "x-grid-td",
55946
55947     hdClass : "x-grid-hd",
55948
55949     splitClass : "x-grid-split",
55950
55951     sortClasses : ["sort-asc", "sort-desc"],
55952
55953     enableMoveAnim : false,
55954
55955     hlColor: "C3DAF9",
55956
55957     dh : Roo.DomHelper,
55958
55959     fly : Roo.Element.fly,
55960
55961     css : Roo.util.CSS,
55962
55963     borderWidth: 1,
55964
55965     splitOffset: 3,
55966
55967     scrollIncrement : 22,
55968
55969     cellRE: /(?:.*?)x-grid-(?:hd|cell|csplit)-(?:[\d]+)-([\d]+)(?:.*?)/,
55970
55971     findRE: /\s?(?:x-grid-hd|x-grid-col|x-grid-csplit)\s/,
55972
55973     bind : function(ds, cm){
55974         if(this.ds){
55975             this.ds.un("load", this.onLoad, this);
55976             this.ds.un("datachanged", this.onDataChange, this);
55977             this.ds.un("add", this.onAdd, this);
55978             this.ds.un("remove", this.onRemove, this);
55979             this.ds.un("update", this.onUpdate, this);
55980             this.ds.un("clear", this.onClear, this);
55981         }
55982         if(ds){
55983             ds.on("load", this.onLoad, this);
55984             ds.on("datachanged", this.onDataChange, this);
55985             ds.on("add", this.onAdd, this);
55986             ds.on("remove", this.onRemove, this);
55987             ds.on("update", this.onUpdate, this);
55988             ds.on("clear", this.onClear, this);
55989         }
55990         this.ds = ds;
55991
55992         if(this.cm){
55993             this.cm.un("widthchange", this.onColWidthChange, this);
55994             this.cm.un("headerchange", this.onHeaderChange, this);
55995             this.cm.un("hiddenchange", this.onHiddenChange, this);
55996             this.cm.un("columnmoved", this.onColumnMove, this);
55997             this.cm.un("columnlockchange", this.onColumnLock, this);
55998         }
55999         if(cm){
56000             this.generateRules(cm);
56001             cm.on("widthchange", this.onColWidthChange, this);
56002             cm.on("headerchange", this.onHeaderChange, this);
56003             cm.on("hiddenchange", this.onHiddenChange, this);
56004             cm.on("columnmoved", this.onColumnMove, this);
56005             cm.on("columnlockchange", this.onColumnLock, this);
56006         }
56007         this.cm = cm;
56008     },
56009
56010     init: function(grid){
56011         Roo.grid.GridView.superclass.init.call(this, grid);
56012
56013         this.bind(grid.dataSource, grid.colModel);
56014
56015         grid.on("headerclick", this.handleHeaderClick, this);
56016
56017         if(grid.trackMouseOver){
56018             grid.on("mouseover", this.onRowOver, this);
56019             grid.on("mouseout", this.onRowOut, this);
56020         }
56021         grid.cancelTextSelection = function(){};
56022         this.gridId = grid.id;
56023
56024         var tpls = this.templates || {};
56025
56026         if(!tpls.master){
56027             tpls.master = new Roo.Template(
56028                '<div class="x-grid" hidefocus="true">',
56029                 '<a href="#" class="x-grid-focus" tabIndex="-1"></a>',
56030                   '<div class="x-grid-topbar"></div>',
56031                   '<div class="x-grid-scroller"><div></div></div>',
56032                   '<div class="x-grid-locked">',
56033                       '<div class="x-grid-header">{lockedHeader}</div>',
56034                       '<div class="x-grid-body">{lockedBody}</div>',
56035                   "</div>",
56036                   '<div class="x-grid-viewport">',
56037                       '<div class="x-grid-header">{header}</div>',
56038                       '<div class="x-grid-body">{body}</div>',
56039                   "</div>",
56040                   '<div class="x-grid-bottombar"></div>',
56041                  
56042                   '<div class="x-grid-resize-proxy">&#160;</div>',
56043                "</div>"
56044             );
56045             tpls.master.disableformats = true;
56046         }
56047
56048         if(!tpls.header){
56049             tpls.header = new Roo.Template(
56050                '<table border="0" cellspacing="0" cellpadding="0">',
56051                '<tbody><tr class="x-grid-hd-row">{cells}</tr></tbody>',
56052                "</table>{splits}"
56053             );
56054             tpls.header.disableformats = true;
56055         }
56056         tpls.header.compile();
56057
56058         if(!tpls.hcell){
56059             tpls.hcell = new Roo.Template(
56060                 '<td class="x-grid-hd x-grid-td-{id} {cellId}"><div title="{title}" class="x-grid-hd-inner x-grid-hd-{id}">',
56061                 '<div class="x-grid-hd-text ' + this.unselectableCls +  '" ' + this.unselectable +'>{value}<img class="x-grid-sort-icon" src="', Roo.BLANK_IMAGE_URL, '" /></div>',
56062                 "</div></td>"
56063              );
56064              tpls.hcell.disableFormats = true;
56065         }
56066         tpls.hcell.compile();
56067
56068         if(!tpls.hsplit){
56069             tpls.hsplit = new Roo.Template('<div class="x-grid-split {splitId} x-grid-split-{id}" style="{style} ' +
56070                                             this.unselectableCls +  '" ' + this.unselectable +'>&#160;</div>');
56071             tpls.hsplit.disableFormats = true;
56072         }
56073         tpls.hsplit.compile();
56074
56075         if(!tpls.body){
56076             tpls.body = new Roo.Template(
56077                '<table border="0" cellspacing="0" cellpadding="0">',
56078                "<tbody>{rows}</tbody>",
56079                "</table>"
56080             );
56081             tpls.body.disableFormats = true;
56082         }
56083         tpls.body.compile();
56084
56085         if(!tpls.row){
56086             tpls.row = new Roo.Template('<tr class="x-grid-row {alt}">{cells}</tr>');
56087             tpls.row.disableFormats = true;
56088         }
56089         tpls.row.compile();
56090
56091         if(!tpls.cell){
56092             tpls.cell = new Roo.Template(
56093                 '<td class="x-grid-col x-grid-td-{id} {cellId} {css}" tabIndex="0">',
56094                 '<div class="x-grid-col-{id} x-grid-cell-inner"><div class="x-grid-cell-text ' +
56095                     this.unselectableCls +  '" ' + this.unselectable +'" {attr}>{value}</div></div>',
56096                 "</td>"
56097             );
56098             tpls.cell.disableFormats = true;
56099         }
56100         tpls.cell.compile();
56101
56102         this.templates = tpls;
56103     },
56104
56105     // remap these for backwards compat
56106     onColWidthChange : function(){
56107         this.updateColumns.apply(this, arguments);
56108     },
56109     onHeaderChange : function(){
56110         this.updateHeaders.apply(this, arguments);
56111     }, 
56112     onHiddenChange : function(){
56113         this.handleHiddenChange.apply(this, arguments);
56114     },
56115     onColumnMove : function(){
56116         this.handleColumnMove.apply(this, arguments);
56117     },
56118     onColumnLock : function(){
56119         this.handleLockChange.apply(this, arguments);
56120     },
56121
56122     onDataChange : function(){
56123         this.refresh();
56124         this.updateHeaderSortState();
56125     },
56126
56127     onClear : function(){
56128         this.refresh();
56129     },
56130
56131     onUpdate : function(ds, record){
56132         this.refreshRow(record);
56133     },
56134
56135     refreshRow : function(record){
56136         var ds = this.ds, index;
56137         if(typeof record == 'number'){
56138             index = record;
56139             record = ds.getAt(index);
56140         }else{
56141             index = ds.indexOf(record);
56142         }
56143         this.insertRows(ds, index, index, true);
56144         this.onRemove(ds, record, index+1, true);
56145         this.syncRowHeights(index, index);
56146         this.layout();
56147         this.fireEvent("rowupdated", this, index, record);
56148     },
56149
56150     onAdd : function(ds, records, index){
56151         this.insertRows(ds, index, index + (records.length-1));
56152     },
56153
56154     onRemove : function(ds, record, index, isUpdate){
56155         if(isUpdate !== true){
56156             this.fireEvent("beforerowremoved", this, index, record);
56157         }
56158         var bt = this.getBodyTable(), lt = this.getLockedTable();
56159         if(bt.rows[index]){
56160             bt.firstChild.removeChild(bt.rows[index]);
56161         }
56162         if(lt.rows[index]){
56163             lt.firstChild.removeChild(lt.rows[index]);
56164         }
56165         if(isUpdate !== true){
56166             this.stripeRows(index);
56167             this.syncRowHeights(index, index);
56168             this.layout();
56169             this.fireEvent("rowremoved", this, index, record);
56170         }
56171     },
56172
56173     onLoad : function(){
56174         this.scrollToTop();
56175     },
56176
56177     /**
56178      * Scrolls the grid to the top
56179      */
56180     scrollToTop : function(){
56181         if(this.scroller){
56182             this.scroller.dom.scrollTop = 0;
56183             this.syncScroll();
56184         }
56185     },
56186
56187     /**
56188      * Gets a panel in the header of the grid that can be used for toolbars etc.
56189      * After modifying the contents of this panel a call to grid.autoSize() may be
56190      * required to register any changes in size.
56191      * @param {Boolean} doShow By default the header is hidden. Pass true to show the panel
56192      * @return Roo.Element
56193      */
56194     getHeaderPanel : function(doShow){
56195         if(doShow){
56196             this.headerPanel.show();
56197         }
56198         return this.headerPanel;
56199     },
56200
56201     /**
56202      * Gets a panel in the footer of the grid that can be used for toolbars etc.
56203      * After modifying the contents of this panel a call to grid.autoSize() may be
56204      * required to register any changes in size.
56205      * @param {Boolean} doShow By default the footer is hidden. Pass true to show the panel
56206      * @return Roo.Element
56207      */
56208     getFooterPanel : function(doShow){
56209         if(doShow){
56210             this.footerPanel.show();
56211         }
56212         return this.footerPanel;
56213     },
56214
56215     initElements : function(){
56216         var E = Roo.Element;
56217         var el = this.grid.getGridEl().dom.firstChild;
56218         var cs = el.childNodes;
56219
56220         this.el = new E(el);
56221         
56222          this.focusEl = new E(el.firstChild);
56223         this.focusEl.swallowEvent("click", true);
56224         
56225         this.headerPanel = new E(cs[1]);
56226         this.headerPanel.enableDisplayMode("block");
56227
56228         this.scroller = new E(cs[2]);
56229         this.scrollSizer = new E(this.scroller.dom.firstChild);
56230
56231         this.lockedWrap = new E(cs[3]);
56232         this.lockedHd = new E(this.lockedWrap.dom.firstChild);
56233         this.lockedBody = new E(this.lockedWrap.dom.childNodes[1]);
56234
56235         this.mainWrap = new E(cs[4]);
56236         this.mainHd = new E(this.mainWrap.dom.firstChild);
56237         this.mainBody = new E(this.mainWrap.dom.childNodes[1]);
56238
56239         this.footerPanel = new E(cs[5]);
56240         this.footerPanel.enableDisplayMode("block");
56241
56242         this.resizeProxy = new E(cs[6]);
56243
56244         this.headerSelector = String.format(
56245            '#{0} td.x-grid-hd, #{1} td.x-grid-hd',
56246            this.lockedHd.id, this.mainHd.id
56247         );
56248
56249         this.splitterSelector = String.format(
56250            '#{0} div.x-grid-split, #{1} div.x-grid-split',
56251            this.idToCssName(this.lockedHd.id), this.idToCssName(this.mainHd.id)
56252         );
56253     },
56254     idToCssName : function(s)
56255     {
56256         return s.replace(/[^a-z0-9]+/ig, '-');
56257     },
56258
56259     getHeaderCell : function(index){
56260         return Roo.DomQuery.select(this.headerSelector)[index];
56261     },
56262
56263     getHeaderCellMeasure : function(index){
56264         return this.getHeaderCell(index).firstChild;
56265     },
56266
56267     getHeaderCellText : function(index){
56268         return this.getHeaderCell(index).firstChild.firstChild;
56269     },
56270
56271     getLockedTable : function(){
56272         return this.lockedBody.dom.firstChild;
56273     },
56274
56275     getBodyTable : function(){
56276         return this.mainBody.dom.firstChild;
56277     },
56278
56279     getLockedRow : function(index){
56280         return this.getLockedTable().rows[index];
56281     },
56282
56283     getRow : function(index){
56284         return this.getBodyTable().rows[index];
56285     },
56286
56287     getRowComposite : function(index){
56288         if(!this.rowEl){
56289             this.rowEl = new Roo.CompositeElementLite();
56290         }
56291         var els = [], lrow, mrow;
56292         if(lrow = this.getLockedRow(index)){
56293             els.push(lrow);
56294         }
56295         if(mrow = this.getRow(index)){
56296             els.push(mrow);
56297         }
56298         this.rowEl.elements = els;
56299         return this.rowEl;
56300     },
56301     /**
56302      * Gets the 'td' of the cell
56303      * 
56304      * @param {Integer} rowIndex row to select
56305      * @param {Integer} colIndex column to select
56306      * 
56307      * @return {Object} 
56308      */
56309     getCell : function(rowIndex, colIndex){
56310         var locked = this.cm.getLockedCount();
56311         var source;
56312         if(colIndex < locked){
56313             source = this.lockedBody.dom.firstChild;
56314         }else{
56315             source = this.mainBody.dom.firstChild;
56316             colIndex -= locked;
56317         }
56318         return source.rows[rowIndex].childNodes[colIndex];
56319     },
56320
56321     getCellText : function(rowIndex, colIndex){
56322         return this.getCell(rowIndex, colIndex).firstChild.firstChild;
56323     },
56324
56325     getCellBox : function(cell){
56326         var b = this.fly(cell).getBox();
56327         if(Roo.isOpera){ // opera fails to report the Y
56328             b.y = cell.offsetTop + this.mainBody.getY();
56329         }
56330         return b;
56331     },
56332
56333     getCellIndex : function(cell){
56334         var id = String(cell.className).match(this.cellRE);
56335         if(id){
56336             return parseInt(id[1], 10);
56337         }
56338         return 0;
56339     },
56340
56341     findHeaderIndex : function(n){
56342         var r = Roo.fly(n).findParent("td." + this.hdClass, 6);
56343         return r ? this.getCellIndex(r) : false;
56344     },
56345
56346     findHeaderCell : function(n){
56347         var r = Roo.fly(n).findParent("td." + this.hdClass, 6);
56348         return r ? r : false;
56349     },
56350
56351     findRowIndex : function(n){
56352         if(!n){
56353             return false;
56354         }
56355         var r = Roo.fly(n).findParent("tr." + this.rowClass, 6);
56356         return r ? r.rowIndex : false;
56357     },
56358
56359     findCellIndex : function(node){
56360         var stop = this.el.dom;
56361         while(node && node != stop){
56362             if(this.findRE.test(node.className)){
56363                 return this.getCellIndex(node);
56364             }
56365             node = node.parentNode;
56366         }
56367         return false;
56368     },
56369
56370     getColumnId : function(index){
56371         return this.cm.getColumnId(index);
56372     },
56373
56374     getSplitters : function()
56375     {
56376         if(this.splitterSelector){
56377            return Roo.DomQuery.select(this.splitterSelector);
56378         }else{
56379             return null;
56380       }
56381     },
56382
56383     getSplitter : function(index){
56384         return this.getSplitters()[index];
56385     },
56386
56387     onRowOver : function(e, t){
56388         var row;
56389         if((row = this.findRowIndex(t)) !== false){
56390             this.getRowComposite(row).addClass("x-grid-row-over");
56391         }
56392     },
56393
56394     onRowOut : function(e, t){
56395         var row;
56396         if((row = this.findRowIndex(t)) !== false && row !== this.findRowIndex(e.getRelatedTarget())){
56397             this.getRowComposite(row).removeClass("x-grid-row-over");
56398         }
56399     },
56400
56401     renderHeaders : function(){
56402         var cm = this.cm;
56403         var ct = this.templates.hcell, ht = this.templates.header, st = this.templates.hsplit;
56404         var cb = [], lb = [], sb = [], lsb = [], p = {};
56405         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56406             p.cellId = "x-grid-hd-0-" + i;
56407             p.splitId = "x-grid-csplit-0-" + i;
56408             p.id = cm.getColumnId(i);
56409             p.value = cm.getColumnHeader(i) || "";
56410             p.title = cm.getColumnTooltip(i) || (''+p.value).match(/\</)  ? '' :  p.value  || "";
56411             p.style = (this.grid.enableColumnResize === false || !cm.isResizable(i) || cm.isFixed(i)) ? 'cursor:default' : '';
56412             if(!cm.isLocked(i)){
56413                 cb[cb.length] = ct.apply(p);
56414                 sb[sb.length] = st.apply(p);
56415             }else{
56416                 lb[lb.length] = ct.apply(p);
56417                 lsb[lsb.length] = st.apply(p);
56418             }
56419         }
56420         return [ht.apply({cells: lb.join(""), splits:lsb.join("")}),
56421                 ht.apply({cells: cb.join(""), splits:sb.join("")})];
56422     },
56423
56424     updateHeaders : function(){
56425         var html = this.renderHeaders();
56426         this.lockedHd.update(html[0]);
56427         this.mainHd.update(html[1]);
56428     },
56429
56430     /**
56431      * Focuses the specified row.
56432      * @param {Number} row The row index
56433      */
56434     focusRow : function(row)
56435     {
56436         //Roo.log('GridView.focusRow');
56437         var x = this.scroller.dom.scrollLeft;
56438         this.focusCell(row, 0, false);
56439         this.scroller.dom.scrollLeft = x;
56440     },
56441
56442     /**
56443      * Focuses the specified cell.
56444      * @param {Number} row The row index
56445      * @param {Number} col The column index
56446      * @param {Boolean} hscroll false to disable horizontal scrolling
56447      */
56448     focusCell : function(row, col, hscroll)
56449     {
56450         //Roo.log('GridView.focusCell');
56451         var el = this.ensureVisible(row, col, hscroll);
56452         this.focusEl.alignTo(el, "tl-tl");
56453         if(Roo.isGecko){
56454             this.focusEl.focus();
56455         }else{
56456             this.focusEl.focus.defer(1, this.focusEl);
56457         }
56458     },
56459
56460     /**
56461      * Scrolls the specified cell into view
56462      * @param {Number} row The row index
56463      * @param {Number} col The column index
56464      * @param {Boolean} hscroll false to disable horizontal scrolling
56465      */
56466     ensureVisible : function(row, col, hscroll)
56467     {
56468         //Roo.log('GridView.ensureVisible,' + row + ',' + col);
56469         //return null; //disable for testing.
56470         if(typeof row != "number"){
56471             row = row.rowIndex;
56472         }
56473         if(row < 0 && row >= this.ds.getCount()){
56474             return  null;
56475         }
56476         col = (col !== undefined ? col : 0);
56477         var cm = this.grid.colModel;
56478         while(cm.isHidden(col)){
56479             col++;
56480         }
56481
56482         var el = this.getCell(row, col);
56483         if(!el){
56484             return null;
56485         }
56486         var c = this.scroller.dom;
56487
56488         var ctop = parseInt(el.offsetTop, 10);
56489         var cleft = parseInt(el.offsetLeft, 10);
56490         var cbot = ctop + el.offsetHeight;
56491         var cright = cleft + el.offsetWidth;
56492         
56493         var ch = c.clientHeight - this.mainHd.dom.offsetHeight;
56494         var stop = parseInt(c.scrollTop, 10);
56495         var sleft = parseInt(c.scrollLeft, 10);
56496         var sbot = stop + ch;
56497         var sright = sleft + c.clientWidth;
56498         /*
56499         Roo.log('GridView.ensureVisible:' +
56500                 ' ctop:' + ctop +
56501                 ' c.clientHeight:' + c.clientHeight +
56502                 ' this.mainHd.dom.offsetHeight:' + this.mainHd.dom.offsetHeight +
56503                 ' stop:' + stop +
56504                 ' cbot:' + cbot +
56505                 ' sbot:' + sbot +
56506                 ' ch:' + ch  
56507                 );
56508         */
56509         if(ctop < stop){
56510              c.scrollTop = ctop;
56511             //Roo.log("set scrolltop to ctop DISABLE?");
56512         }else if(cbot > sbot){
56513             //Roo.log("set scrolltop to cbot-ch");
56514             c.scrollTop = cbot-ch;
56515         }
56516         
56517         if(hscroll !== false){
56518             if(cleft < sleft){
56519                 c.scrollLeft = cleft;
56520             }else if(cright > sright){
56521                 c.scrollLeft = cright-c.clientWidth;
56522             }
56523         }
56524          
56525         return el;
56526     },
56527
56528     updateColumns : function(){
56529         this.grid.stopEditing();
56530         var cm = this.grid.colModel, colIds = this.getColumnIds();
56531         //var totalWidth = cm.getTotalWidth();
56532         var pos = 0;
56533         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56534             //if(cm.isHidden(i)) continue;
56535             var w = cm.getColumnWidth(i);
56536             this.css.updateRule(this.colSelector+this.idToCssName(colIds[i]), "width", (w - this.borderWidth) + "px");
56537             this.css.updateRule(this.hdSelector+this.idToCssName(colIds[i]), "width", (w - this.borderWidth) + "px");
56538         }
56539         this.updateSplitters();
56540     },
56541
56542     generateRules : function(cm){
56543         var ruleBuf = [], rulesId = this.idToCssName(this.grid.id)+ '-cssrules';
56544         Roo.util.CSS.removeStyleSheet(rulesId);
56545         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56546             var cid = cm.getColumnId(i);
56547             var align = '';
56548             if(cm.config[i].align){
56549                 align = 'text-align:'+cm.config[i].align+';';
56550             }
56551             var hidden = '';
56552             if(cm.isHidden(i)){
56553                 hidden = 'display:none;';
56554             }
56555             var width = "width:" + (cm.getColumnWidth(i) - this.borderWidth) + "px;";
56556             ruleBuf.push(
56557                     this.colSelector, cid, " {\n", cm.config[i].css, align, width, "\n}\n",
56558                     this.hdSelector, cid, " {\n", align, width, "}\n",
56559                     this.tdSelector, cid, " {\n",hidden,"\n}\n",
56560                     this.splitSelector, cid, " {\n", hidden , "\n}\n");
56561         }
56562         return Roo.util.CSS.createStyleSheet(ruleBuf.join(""), rulesId);
56563     },
56564
56565     updateSplitters : function(){
56566         var cm = this.cm, s = this.getSplitters();
56567         if(s){ // splitters not created yet
56568             var pos = 0, locked = true;
56569             for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56570                 if(cm.isHidden(i)) {
56571                     continue;
56572                 }
56573                 var w = cm.getColumnWidth(i); // make sure it's a number
56574                 if(!cm.isLocked(i) && locked){
56575                     pos = 0;
56576                     locked = false;
56577                 }
56578                 pos += w;
56579                 s[i].style.left = (pos-this.splitOffset) + "px";
56580             }
56581         }
56582     },
56583
56584     handleHiddenChange : function(colModel, colIndex, hidden){
56585         if(hidden){
56586             this.hideColumn(colIndex);
56587         }else{
56588             this.unhideColumn(colIndex);
56589         }
56590     },
56591
56592     hideColumn : function(colIndex){
56593         var cid = this.getColumnId(colIndex);
56594         this.css.updateRule(this.tdSelector+this.idToCssName(cid), "display", "none");
56595         this.css.updateRule(this.splitSelector+this.idToCssName(cid), "display", "none");
56596         if(Roo.isSafari){
56597             this.updateHeaders();
56598         }
56599         this.updateSplitters();
56600         this.layout();
56601     },
56602
56603     unhideColumn : function(colIndex){
56604         var cid = this.getColumnId(colIndex);
56605         this.css.updateRule(this.tdSelector+this.idToCssName(cid), "display", "");
56606         this.css.updateRule(this.splitSelector+this.idToCssName(cid), "display", "");
56607
56608         if(Roo.isSafari){
56609             this.updateHeaders();
56610         }
56611         this.updateSplitters();
56612         this.layout();
56613     },
56614
56615     insertRows : function(dm, firstRow, lastRow, isUpdate){
56616         if(firstRow == 0 && lastRow == dm.getCount()-1){
56617             this.refresh();
56618         }else{
56619             if(!isUpdate){
56620                 this.fireEvent("beforerowsinserted", this, firstRow, lastRow);
56621             }
56622             var s = this.getScrollState();
56623             var markup = this.renderRows(firstRow, lastRow);
56624             this.bufferRows(markup[0], this.getLockedTable(), firstRow);
56625             this.bufferRows(markup[1], this.getBodyTable(), firstRow);
56626             this.restoreScroll(s);
56627             if(!isUpdate){
56628                 this.fireEvent("rowsinserted", this, firstRow, lastRow);
56629                 this.syncRowHeights(firstRow, lastRow);
56630                 this.stripeRows(firstRow);
56631                 this.layout();
56632             }
56633         }
56634     },
56635
56636     bufferRows : function(markup, target, index){
56637         var before = null, trows = target.rows, tbody = target.tBodies[0];
56638         if(index < trows.length){
56639             before = trows[index];
56640         }
56641         var b = document.createElement("div");
56642         b.innerHTML = "<table><tbody>"+markup+"</tbody></table>";
56643         var rows = b.firstChild.rows;
56644         for(var i = 0, len = rows.length; i < len; i++){
56645             if(before){
56646                 tbody.insertBefore(rows[0], before);
56647             }else{
56648                 tbody.appendChild(rows[0]);
56649             }
56650         }
56651         b.innerHTML = "";
56652         b = null;
56653     },
56654
56655     deleteRows : function(dm, firstRow, lastRow){
56656         if(dm.getRowCount()<1){
56657             this.fireEvent("beforerefresh", this);
56658             this.mainBody.update("");
56659             this.lockedBody.update("");
56660             this.fireEvent("refresh", this);
56661         }else{
56662             this.fireEvent("beforerowsdeleted", this, firstRow, lastRow);
56663             var bt = this.getBodyTable();
56664             var tbody = bt.firstChild;
56665             var rows = bt.rows;
56666             for(var rowIndex = firstRow; rowIndex <= lastRow; rowIndex++){
56667                 tbody.removeChild(rows[firstRow]);
56668             }
56669             this.stripeRows(firstRow);
56670             this.fireEvent("rowsdeleted", this, firstRow, lastRow);
56671         }
56672     },
56673
56674     updateRows : function(dataSource, firstRow, lastRow){
56675         var s = this.getScrollState();
56676         this.refresh();
56677         this.restoreScroll(s);
56678     },
56679
56680     handleSort : function(dataSource, sortColumnIndex, sortDir, noRefresh){
56681         if(!noRefresh){
56682            this.refresh();
56683         }
56684         this.updateHeaderSortState();
56685     },
56686
56687     getScrollState : function(){
56688         
56689         var sb = this.scroller.dom;
56690         return {left: sb.scrollLeft, top: sb.scrollTop};
56691     },
56692
56693     stripeRows : function(startRow){
56694         if(!this.grid.stripeRows || this.ds.getCount() < 1){
56695             return;
56696         }
56697         startRow = startRow || 0;
56698         var rows = this.getBodyTable().rows;
56699         var lrows = this.getLockedTable().rows;
56700         var cls = ' x-grid-row-alt ';
56701         for(var i = startRow, len = rows.length; i < len; i++){
56702             var row = rows[i], lrow = lrows[i];
56703             var isAlt = ((i+1) % 2 == 0);
56704             var hasAlt = (' '+row.className + ' ').indexOf(cls) != -1;
56705             if(isAlt == hasAlt){
56706                 continue;
56707             }
56708             if(isAlt){
56709                 row.className += " x-grid-row-alt";
56710             }else{
56711                 row.className = row.className.replace("x-grid-row-alt", "");
56712             }
56713             if(lrow){
56714                 lrow.className = row.className;
56715             }
56716         }
56717     },
56718
56719     restoreScroll : function(state){
56720         //Roo.log('GridView.restoreScroll');
56721         var sb = this.scroller.dom;
56722         sb.scrollLeft = state.left;
56723         sb.scrollTop = state.top;
56724         this.syncScroll();
56725     },
56726
56727     syncScroll : function(){
56728         //Roo.log('GridView.syncScroll');
56729         var sb = this.scroller.dom;
56730         var sh = this.mainHd.dom;
56731         var bs = this.mainBody.dom;
56732         var lv = this.lockedBody.dom;
56733         sh.scrollLeft = bs.scrollLeft = sb.scrollLeft;
56734         lv.scrollTop = bs.scrollTop = sb.scrollTop;
56735     },
56736
56737     handleScroll : function(e){
56738         this.syncScroll();
56739         var sb = this.scroller.dom;
56740         this.grid.fireEvent("bodyscroll", sb.scrollLeft, sb.scrollTop);
56741         e.stopEvent();
56742     },
56743
56744     handleWheel : function(e){
56745         var d = e.getWheelDelta();
56746         this.scroller.dom.scrollTop -= d*22;
56747         // set this here to prevent jumpy scrolling on large tables
56748         this.lockedBody.dom.scrollTop = this.mainBody.dom.scrollTop = this.scroller.dom.scrollTop;
56749         e.stopEvent();
56750     },
56751
56752     renderRows : function(startRow, endRow){
56753         // pull in all the crap needed to render rows
56754         var g = this.grid, cm = g.colModel, ds = g.dataSource, stripe = g.stripeRows;
56755         var colCount = cm.getColumnCount();
56756
56757         if(ds.getCount() < 1){
56758             return ["", ""];
56759         }
56760
56761         // build a map for all the columns
56762         var cs = [];
56763         for(var i = 0; i < colCount; i++){
56764             var name = cm.getDataIndex(i);
56765             cs[i] = {
56766                 name : typeof name == 'undefined' ? ds.fields.get(i).name : name,
56767                 renderer : cm.getRenderer(i),
56768                 id : cm.getColumnId(i),
56769                 locked : cm.isLocked(i),
56770                 has_editor : cm.isCellEditable(i)
56771             };
56772         }
56773
56774         startRow = startRow || 0;
56775         endRow = typeof endRow == "undefined"? ds.getCount()-1 : endRow;
56776
56777         // records to render
56778         var rs = ds.getRange(startRow, endRow);
56779
56780         return this.doRender(cs, rs, ds, startRow, colCount, stripe);
56781     },
56782
56783     // As much as I hate to duplicate code, this was branched because FireFox really hates
56784     // [].join("") on strings. The performance difference was substantial enough to
56785     // branch this function
56786     doRender : Roo.isGecko ?
56787             function(cs, rs, ds, startRow, colCount, stripe){
56788                 var ts = this.templates, ct = ts.cell, rt = ts.row;
56789                 // buffers
56790                 var buf = "", lbuf = "", cb, lcb, c, p = {}, rp = {}, r, rowIndex;
56791                 
56792                 var hasListener = this.grid.hasListener('rowclass');
56793                 var rowcfg = {};
56794                 for(var j = 0, len = rs.length; j < len; j++){
56795                     r = rs[j]; cb = ""; lcb = ""; rowIndex = (j+startRow);
56796                     for(var i = 0; i < colCount; i++){
56797                         c = cs[i];
56798                         p.cellId = "x-grid-cell-" + rowIndex + "-" + i;
56799                         p.id = c.id;
56800                         p.css = p.attr = "";
56801                         p.value = c.renderer(r.data[c.name], p, r, rowIndex, i, ds);
56802                         if(p.value == undefined || p.value === "") {
56803                             p.value = "&#160;";
56804                         }
56805                         if(c.has_editor){
56806                             p.css += ' x-grid-editable-cell';
56807                         }
56808                         if(c.dirty && typeof r.modified[c.name] !== 'undefined'){
56809                             p.css +=  ' x-grid-dirty-cell';
56810                         }
56811                         var markup = ct.apply(p);
56812                         if(!c.locked){
56813                             cb+= markup;
56814                         }else{
56815                             lcb+= markup;
56816                         }
56817                     }
56818                     var alt = [];
56819                     if(stripe && ((rowIndex+1) % 2 == 0)){
56820                         alt.push("x-grid-row-alt")
56821                     }
56822                     if(r.dirty){
56823                         alt.push(  " x-grid-dirty-row");
56824                     }
56825                     rp.cells = lcb;
56826                     if(this.getRowClass){
56827                         alt.push(this.getRowClass(r, rowIndex));
56828                     }
56829                     if (hasListener) {
56830                         rowcfg = {
56831                              
56832                             record: r,
56833                             rowIndex : rowIndex,
56834                             rowClass : ''
56835                         };
56836                         this.grid.fireEvent('rowclass', this, rowcfg);
56837                         alt.push(rowcfg.rowClass);
56838                     }
56839                     rp.alt = alt.join(" ");
56840                     lbuf+= rt.apply(rp);
56841                     rp.cells = cb;
56842                     buf+=  rt.apply(rp);
56843                 }
56844                 return [lbuf, buf];
56845             } :
56846             function(cs, rs, ds, startRow, colCount, stripe){
56847                 var ts = this.templates, ct = ts.cell, rt = ts.row;
56848                 // buffers
56849                 var buf = [], lbuf = [], cb, lcb, c, p = {}, rp = {}, r, rowIndex;
56850                 var hasListener = this.grid.hasListener('rowclass');
56851  
56852                 var rowcfg = {};
56853                 for(var j = 0, len = rs.length; j < len; j++){
56854                     r = rs[j]; cb = []; lcb = []; rowIndex = (j+startRow);
56855                     for(var i = 0; i < colCount; i++){
56856                         c = cs[i];
56857                         p.cellId = "x-grid-cell-" + rowIndex + "-" + i;
56858                         p.id = c.id;
56859                         p.css = p.attr = "";
56860                         p.value = c.renderer(r.data[c.name], p, r, rowIndex, i, ds);
56861                         if(p.value == undefined || p.value === "") {
56862                             p.value = "&#160;";
56863                         }
56864                         //Roo.log(c);
56865                          if(c.has_editor){
56866                             p.css += ' x-grid-editable-cell';
56867                         }
56868                         if(r.dirty && typeof r.modified[c.name] !== 'undefined'){
56869                             p.css += ' x-grid-dirty-cell' 
56870                         }
56871                         
56872                         var markup = ct.apply(p);
56873                         if(!c.locked){
56874                             cb[cb.length] = markup;
56875                         }else{
56876                             lcb[lcb.length] = markup;
56877                         }
56878                     }
56879                     var alt = [];
56880                     if(stripe && ((rowIndex+1) % 2 == 0)){
56881                         alt.push( "x-grid-row-alt");
56882                     }
56883                     if(r.dirty){
56884                         alt.push(" x-grid-dirty-row");
56885                     }
56886                     rp.cells = lcb;
56887                     if(this.getRowClass){
56888                         alt.push( this.getRowClass(r, rowIndex));
56889                     }
56890                     if (hasListener) {
56891                         rowcfg = {
56892                              
56893                             record: r,
56894                             rowIndex : rowIndex,
56895                             rowClass : ''
56896                         };
56897                         this.grid.fireEvent('rowclass', this, rowcfg);
56898                         alt.push(rowcfg.rowClass);
56899                     }
56900                     
56901                     rp.alt = alt.join(" ");
56902                     rp.cells = lcb.join("");
56903                     lbuf[lbuf.length] = rt.apply(rp);
56904                     rp.cells = cb.join("");
56905                     buf[buf.length] =  rt.apply(rp);
56906                 }
56907                 return [lbuf.join(""), buf.join("")];
56908             },
56909
56910     renderBody : function(){
56911         var markup = this.renderRows();
56912         var bt = this.templates.body;
56913         return [bt.apply({rows: markup[0]}), bt.apply({rows: markup[1]})];
56914     },
56915
56916     /**
56917      * Refreshes the grid
56918      * @param {Boolean} headersToo
56919      */
56920     refresh : function(headersToo){
56921         this.fireEvent("beforerefresh", this);
56922         this.grid.stopEditing();
56923         var result = this.renderBody();
56924         this.lockedBody.update(result[0]);
56925         this.mainBody.update(result[1]);
56926         if(headersToo === true){
56927             this.updateHeaders();
56928             this.updateColumns();
56929             this.updateSplitters();
56930             this.updateHeaderSortState();
56931         }
56932         this.syncRowHeights();
56933         this.layout();
56934         this.fireEvent("refresh", this);
56935     },
56936
56937     handleColumnMove : function(cm, oldIndex, newIndex){
56938         this.indexMap = null;
56939         var s = this.getScrollState();
56940         this.refresh(true);
56941         this.restoreScroll(s);
56942         this.afterMove(newIndex);
56943     },
56944
56945     afterMove : function(colIndex){
56946         if(this.enableMoveAnim && Roo.enableFx){
56947             this.fly(this.getHeaderCell(colIndex).firstChild).highlight(this.hlColor);
56948         }
56949         // if multisort - fix sortOrder, and reload..
56950         if (this.grid.dataSource.multiSort) {
56951             // the we can call sort again..
56952             var dm = this.grid.dataSource;
56953             var cm = this.grid.colModel;
56954             var so = [];
56955             for(var i = 0; i < cm.config.length; i++ ) {
56956                 
56957                 if ((typeof(dm.sortToggle[cm.config[i].dataIndex]) == 'undefined')) {
56958                     continue; // dont' bother, it's not in sort list or being set.
56959                 }
56960                 
56961                 so.push(cm.config[i].dataIndex);
56962             };
56963             dm.sortOrder = so;
56964             dm.load(dm.lastOptions);
56965             
56966             
56967         }
56968         
56969     },
56970
56971     updateCell : function(dm, rowIndex, dataIndex){
56972         var colIndex = this.getColumnIndexByDataIndex(dataIndex);
56973         if(typeof colIndex == "undefined"){ // not present in grid
56974             return;
56975         }
56976         var cm = this.grid.colModel;
56977         var cell = this.getCell(rowIndex, colIndex);
56978         var cellText = this.getCellText(rowIndex, colIndex);
56979
56980         var p = {
56981             cellId : "x-grid-cell-" + rowIndex + "-" + colIndex,
56982             id : cm.getColumnId(colIndex),
56983             css: colIndex == cm.getColumnCount()-1 ? "x-grid-col-last" : ""
56984         };
56985         var renderer = cm.getRenderer(colIndex);
56986         var val = renderer(dm.getValueAt(rowIndex, dataIndex), p, rowIndex, colIndex, dm);
56987         if(typeof val == "undefined" || val === "") {
56988             val = "&#160;";
56989         }
56990         cellText.innerHTML = val;
56991         cell.className = this.cellClass + " " + this.idToCssName(p.cellId) + " " + p.css;
56992         this.syncRowHeights(rowIndex, rowIndex);
56993     },
56994
56995     calcColumnWidth : function(colIndex, maxRowsToMeasure){
56996         var maxWidth = 0;
56997         if(this.grid.autoSizeHeaders){
56998             var h = this.getHeaderCellMeasure(colIndex);
56999             maxWidth = Math.max(maxWidth, h.scrollWidth);
57000         }
57001         var tb, index;
57002         if(this.cm.isLocked(colIndex)){
57003             tb = this.getLockedTable();
57004             index = colIndex;
57005         }else{
57006             tb = this.getBodyTable();
57007             index = colIndex - this.cm.getLockedCount();
57008         }
57009         if(tb && tb.rows){
57010             var rows = tb.rows;
57011             var stopIndex = Math.min(maxRowsToMeasure || rows.length, rows.length);
57012             for(var i = 0; i < stopIndex; i++){
57013                 var cell = rows[i].childNodes[index].firstChild;
57014                 maxWidth = Math.max(maxWidth, cell.scrollWidth);
57015             }
57016         }
57017         return maxWidth + /*margin for error in IE*/ 5;
57018     },
57019     /**
57020      * Autofit a column to its content.
57021      * @param {Number} colIndex
57022      * @param {Boolean} forceMinSize true to force the column to go smaller if possible
57023      */
57024      autoSizeColumn : function(colIndex, forceMinSize, suppressEvent){
57025          if(this.cm.isHidden(colIndex)){
57026              return; // can't calc a hidden column
57027          }
57028         if(forceMinSize){
57029             var cid = this.cm.getColumnId(colIndex);
57030             this.css.updateRule(this.colSelector +this.idToCssName( cid), "width", this.grid.minColumnWidth + "px");
57031            if(this.grid.autoSizeHeaders){
57032                this.css.updateRule(this.hdSelector + this.idToCssName(cid), "width", this.grid.minColumnWidth + "px");
57033            }
57034         }
57035         var newWidth = this.calcColumnWidth(colIndex);
57036         this.cm.setColumnWidth(colIndex,
57037             Math.max(this.grid.minColumnWidth, newWidth), suppressEvent);
57038         if(!suppressEvent){
57039             this.grid.fireEvent("columnresize", colIndex, newWidth);
57040         }
57041     },
57042
57043     /**
57044      * Autofits all columns to their content and then expands to fit any extra space in the grid
57045      */
57046      autoSizeColumns : function(){
57047         var cm = this.grid.colModel;
57048         var colCount = cm.getColumnCount();
57049         for(var i = 0; i < colCount; i++){
57050             this.autoSizeColumn(i, true, true);
57051         }
57052         if(cm.getTotalWidth() < this.scroller.dom.clientWidth){
57053             this.fitColumns();
57054         }else{
57055             this.updateColumns();
57056             this.layout();
57057         }
57058     },
57059
57060     /**
57061      * Autofits all columns to the grid's width proportionate with their current size
57062      * @param {Boolean} reserveScrollSpace Reserve space for a scrollbar
57063      */
57064     fitColumns : function(reserveScrollSpace){
57065         var cm = this.grid.colModel;
57066         var colCount = cm.getColumnCount();
57067         var cols = [];
57068         var width = 0;
57069         var i, w;
57070         for (i = 0; i < colCount; i++){
57071             if(!cm.isHidden(i) && !cm.isFixed(i)){
57072                 w = cm.getColumnWidth(i);
57073                 cols.push(i);
57074                 cols.push(w);
57075                 width += w;
57076             }
57077         }
57078         var avail = Math.min(this.scroller.dom.clientWidth, this.el.getWidth());
57079         if(reserveScrollSpace){
57080             avail -= 17;
57081         }
57082         var frac = (avail - cm.getTotalWidth())/width;
57083         while (cols.length){
57084             w = cols.pop();
57085             i = cols.pop();
57086             cm.setColumnWidth(i, Math.floor(w + w*frac), true);
57087         }
57088         this.updateColumns();
57089         this.layout();
57090     },
57091
57092     onRowSelect : function(rowIndex){
57093         var row = this.getRowComposite(rowIndex);
57094         row.addClass("x-grid-row-selected");
57095     },
57096
57097     onRowDeselect : function(rowIndex){
57098         var row = this.getRowComposite(rowIndex);
57099         row.removeClass("x-grid-row-selected");
57100     },
57101
57102     onCellSelect : function(row, col){
57103         var cell = this.getCell(row, col);
57104         if(cell){
57105             Roo.fly(cell).addClass("x-grid-cell-selected");
57106         }
57107     },
57108
57109     onCellDeselect : function(row, col){
57110         var cell = this.getCell(row, col);
57111         if(cell){
57112             Roo.fly(cell).removeClass("x-grid-cell-selected");
57113         }
57114     },
57115
57116     updateHeaderSortState : function(){
57117         
57118         // sort state can be single { field: xxx, direction : yyy}
57119         // or   { xxx=>ASC , yyy : DESC ..... }
57120         
57121         var mstate = {};
57122         if (!this.ds.multiSort) { 
57123             var state = this.ds.getSortState();
57124             if(!state){
57125                 return;
57126             }
57127             mstate[state.field] = state.direction;
57128             // FIXME... - this is not used here.. but might be elsewhere..
57129             this.sortState = state;
57130             
57131         } else {
57132             mstate = this.ds.sortToggle;
57133         }
57134         //remove existing sort classes..
57135         
57136         var sc = this.sortClasses;
57137         var hds = this.el.select(this.headerSelector).removeClass(sc);
57138         
57139         for(var f in mstate) {
57140         
57141             var sortColumn = this.cm.findColumnIndex(f);
57142             
57143             if(sortColumn != -1){
57144                 var sortDir = mstate[f];        
57145                 hds.item(sortColumn).addClass(sc[sortDir == "DESC" ? 1 : 0]);
57146             }
57147         }
57148         
57149          
57150         
57151     },
57152
57153
57154     handleHeaderClick : function(g, index,e){
57155         
57156         Roo.log("header click");
57157         
57158         if (Roo.isTouch) {
57159             // touch events on header are handled by context
57160             this.handleHdCtx(g,index,e);
57161             return;
57162         }
57163         
57164         
57165         if(this.headersDisabled){
57166             return;
57167         }
57168         var dm = g.dataSource, cm = g.colModel;
57169         if(!cm.isSortable(index)){
57170             return;
57171         }
57172         g.stopEditing();
57173         
57174         if (dm.multiSort) {
57175             // update the sortOrder
57176             var so = [];
57177             for(var i = 0; i < cm.config.length; i++ ) {
57178                 
57179                 if ((typeof(dm.sortToggle[cm.config[i].dataIndex]) == 'undefined') && (index != i)) {
57180                     continue; // dont' bother, it's not in sort list or being set.
57181                 }
57182                 
57183                 so.push(cm.config[i].dataIndex);
57184             };
57185             dm.sortOrder = so;
57186         }
57187         
57188         
57189         dm.sort(cm.getDataIndex(index));
57190     },
57191
57192
57193     destroy : function(){
57194         if(this.colMenu){
57195             this.colMenu.removeAll();
57196             Roo.menu.MenuMgr.unregister(this.colMenu);
57197             this.colMenu.getEl().remove();
57198             delete this.colMenu;
57199         }
57200         if(this.hmenu){
57201             this.hmenu.removeAll();
57202             Roo.menu.MenuMgr.unregister(this.hmenu);
57203             this.hmenu.getEl().remove();
57204             delete this.hmenu;
57205         }
57206         if(this.grid.enableColumnMove){
57207             var dds = Roo.dd.DDM.ids['gridHeader' + this.grid.getGridEl().id];
57208             if(dds){
57209                 for(var dd in dds){
57210                     if(!dds[dd].config.isTarget && dds[dd].dragElId){
57211                         var elid = dds[dd].dragElId;
57212                         dds[dd].unreg();
57213                         Roo.get(elid).remove();
57214                     } else if(dds[dd].config.isTarget){
57215                         dds[dd].proxyTop.remove();
57216                         dds[dd].proxyBottom.remove();
57217                         dds[dd].unreg();
57218                     }
57219                     if(Roo.dd.DDM.locationCache[dd]){
57220                         delete Roo.dd.DDM.locationCache[dd];
57221                     }
57222                 }
57223                 delete Roo.dd.DDM.ids['gridHeader' + this.grid.getGridEl().id];
57224             }
57225         }
57226         Roo.util.CSS.removeStyleSheet(this.idToCssName(this.grid.id) + '-cssrules');
57227         this.bind(null, null);
57228         Roo.EventManager.removeResizeListener(this.onWindowResize, this);
57229     },
57230
57231     handleLockChange : function(){
57232         this.refresh(true);
57233     },
57234
57235     onDenyColumnLock : function(){
57236
57237     },
57238
57239     onDenyColumnHide : function(){
57240
57241     },
57242
57243     handleHdMenuClick : function(item){
57244         var index = this.hdCtxIndex;
57245         var cm = this.cm, ds = this.ds;
57246         switch(item.id){
57247             case "asc":
57248                 ds.sort(cm.getDataIndex(index), "ASC");
57249                 break;
57250             case "desc":
57251                 ds.sort(cm.getDataIndex(index), "DESC");
57252                 break;
57253             case "lock":
57254                 var lc = cm.getLockedCount();
57255                 if(cm.getColumnCount(true) <= lc+1){
57256                     this.onDenyColumnLock();
57257                     return;
57258                 }
57259                 if(lc != index){
57260                     cm.setLocked(index, true, true);
57261                     cm.moveColumn(index, lc);
57262                     this.grid.fireEvent("columnmove", index, lc);
57263                 }else{
57264                     cm.setLocked(index, true);
57265                 }
57266             break;
57267             case "unlock":
57268                 var lc = cm.getLockedCount();
57269                 if((lc-1) != index){
57270                     cm.setLocked(index, false, true);
57271                     cm.moveColumn(index, lc-1);
57272                     this.grid.fireEvent("columnmove", index, lc-1);
57273                 }else{
57274                     cm.setLocked(index, false);
57275                 }
57276             break;
57277             case 'wider': // used to expand cols on touch..
57278             case 'narrow':
57279                 var cw = cm.getColumnWidth(index);
57280                 cw += (item.id == 'wider' ? 1 : -1) * 50;
57281                 cw = Math.max(0, cw);
57282                 cw = Math.min(cw,4000);
57283                 cm.setColumnWidth(index, cw);
57284                 break;
57285                 
57286             default:
57287                 index = cm.getIndexById(item.id.substr(4));
57288                 if(index != -1){
57289                     if(item.checked && cm.getColumnCount(true) <= 1){
57290                         this.onDenyColumnHide();
57291                         return false;
57292                     }
57293                     cm.setHidden(index, item.checked);
57294                 }
57295         }
57296         return true;
57297     },
57298
57299     beforeColMenuShow : function(){
57300         var cm = this.cm,  colCount = cm.getColumnCount();
57301         this.colMenu.removeAll();
57302         for(var i = 0; i < colCount; i++){
57303             this.colMenu.add(new Roo.menu.CheckItem({
57304                 id: "col-"+cm.getColumnId(i),
57305                 text: cm.getColumnHeader(i),
57306                 checked: !cm.isHidden(i),
57307                 hideOnClick:false
57308             }));
57309         }
57310     },
57311
57312     handleHdCtx : function(g, index, e){
57313         e.stopEvent();
57314         var hd = this.getHeaderCell(index);
57315         this.hdCtxIndex = index;
57316         var ms = this.hmenu.items, cm = this.cm;
57317         ms.get("asc").setDisabled(!cm.isSortable(index));
57318         ms.get("desc").setDisabled(!cm.isSortable(index));
57319         if(this.grid.enableColLock !== false){
57320             ms.get("lock").setDisabled(cm.isLocked(index));
57321             ms.get("unlock").setDisabled(!cm.isLocked(index));
57322         }
57323         this.hmenu.show(hd, "tl-bl");
57324     },
57325
57326     handleHdOver : function(e){
57327         var hd = this.findHeaderCell(e.getTarget());
57328         if(hd && !this.headersDisabled){
57329             if(this.grid.colModel.isSortable(this.getCellIndex(hd))){
57330                this.fly(hd).addClass("x-grid-hd-over");
57331             }
57332         }
57333     },
57334
57335     handleHdOut : function(e){
57336         var hd = this.findHeaderCell(e.getTarget());
57337         if(hd){
57338             this.fly(hd).removeClass("x-grid-hd-over");
57339         }
57340     },
57341
57342     handleSplitDblClick : function(e, t){
57343         var i = this.getCellIndex(t);
57344         if(this.grid.enableColumnResize !== false && this.cm.isResizable(i) && !this.cm.isFixed(i)){
57345             this.autoSizeColumn(i, true);
57346             this.layout();
57347         }
57348     },
57349
57350     render : function(){
57351
57352         var cm = this.cm;
57353         var colCount = cm.getColumnCount();
57354
57355         if(this.grid.monitorWindowResize === true){
57356             Roo.EventManager.onWindowResize(this.onWindowResize, this, true);
57357         }
57358         var header = this.renderHeaders();
57359         var body = this.templates.body.apply({rows:""});
57360         var html = this.templates.master.apply({
57361             lockedBody: body,
57362             body: body,
57363             lockedHeader: header[0],
57364             header: header[1]
57365         });
57366
57367         //this.updateColumns();
57368
57369         this.grid.getGridEl().dom.innerHTML = html;
57370
57371         this.initElements();
57372         
57373         // a kludge to fix the random scolling effect in webkit
57374         this.el.on("scroll", function() {
57375             this.el.dom.scrollTop=0; // hopefully not recursive..
57376         },this);
57377
57378         this.scroller.on("scroll", this.handleScroll, this);
57379         this.lockedBody.on("mousewheel", this.handleWheel, this);
57380         this.mainBody.on("mousewheel", this.handleWheel, this);
57381
57382         this.mainHd.on("mouseover", this.handleHdOver, this);
57383         this.mainHd.on("mouseout", this.handleHdOut, this);
57384         this.mainHd.on("dblclick", this.handleSplitDblClick, this,
57385                 {delegate: "."+this.splitClass});
57386
57387         this.lockedHd.on("mouseover", this.handleHdOver, this);
57388         this.lockedHd.on("mouseout", this.handleHdOut, this);
57389         this.lockedHd.on("dblclick", this.handleSplitDblClick, this,
57390                 {delegate: "."+this.splitClass});
57391
57392         if(this.grid.enableColumnResize !== false && Roo.grid.SplitDragZone){
57393             new Roo.grid.SplitDragZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57394         }
57395
57396         this.updateSplitters();
57397
57398         if(this.grid.enableColumnMove && Roo.grid.HeaderDragZone){
57399             new Roo.grid.HeaderDragZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57400             new Roo.grid.HeaderDropZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57401         }
57402
57403         if(this.grid.enableCtxMenu !== false && Roo.menu.Menu){
57404             this.hmenu = new Roo.menu.Menu({id: this.grid.id + "-hctx"});
57405             this.hmenu.add(
57406                 {id:"asc", text: this.sortAscText, cls: "xg-hmenu-sort-asc"},
57407                 {id:"desc", text: this.sortDescText, cls: "xg-hmenu-sort-desc"}
57408             );
57409             if(this.grid.enableColLock !== false){
57410                 this.hmenu.add('-',
57411                     {id:"lock", text: this.lockText, cls: "xg-hmenu-lock"},
57412                     {id:"unlock", text: this.unlockText, cls: "xg-hmenu-unlock"}
57413                 );
57414             }
57415             if (Roo.isTouch) {
57416                  this.hmenu.add('-',
57417                     {id:"wider", text: this.columnsWiderText},
57418                     {id:"narrow", text: this.columnsNarrowText }
57419                 );
57420                 
57421                  
57422             }
57423             
57424             if(this.grid.enableColumnHide !== false){
57425
57426                 this.colMenu = new Roo.menu.Menu({id:this.grid.id + "-hcols-menu"});
57427                 this.colMenu.on("beforeshow", this.beforeColMenuShow, this);
57428                 this.colMenu.on("itemclick", this.handleHdMenuClick, this);
57429
57430                 this.hmenu.add('-',
57431                     {id:"columns", text: this.columnsText, menu: this.colMenu}
57432                 );
57433             }
57434             this.hmenu.on("itemclick", this.handleHdMenuClick, this);
57435
57436             this.grid.on("headercontextmenu", this.handleHdCtx, this);
57437         }
57438
57439         if((this.grid.enableDragDrop || this.grid.enableDrag) && Roo.grid.GridDragZone){
57440             this.dd = new Roo.grid.GridDragZone(this.grid, {
57441                 ddGroup : this.grid.ddGroup || 'GridDD'
57442             });
57443             
57444         }
57445
57446         /*
57447         for(var i = 0; i < colCount; i++){
57448             if(cm.isHidden(i)){
57449                 this.hideColumn(i);
57450             }
57451             if(cm.config[i].align){
57452                 this.css.updateRule(this.colSelector + i, "textAlign", cm.config[i].align);
57453                 this.css.updateRule(this.hdSelector + i, "textAlign", cm.config[i].align);
57454             }
57455         }*/
57456         
57457         this.updateHeaderSortState();
57458
57459         this.beforeInitialResize();
57460         this.layout(true);
57461
57462         // two part rendering gives faster view to the user
57463         this.renderPhase2.defer(1, this);
57464     },
57465
57466     renderPhase2 : function(){
57467         // render the rows now
57468         this.refresh();
57469         if(this.grid.autoSizeColumns){
57470             this.autoSizeColumns();
57471         }
57472     },
57473
57474     beforeInitialResize : function(){
57475
57476     },
57477
57478     onColumnSplitterMoved : function(i, w){
57479         this.userResized = true;
57480         var cm = this.grid.colModel;
57481         cm.setColumnWidth(i, w, true);
57482         var cid = cm.getColumnId(i);
57483         this.css.updateRule(this.colSelector + this.idToCssName(cid), "width", (w-this.borderWidth) + "px");
57484         this.css.updateRule(this.hdSelector + this.idToCssName(cid), "width", (w-this.borderWidth) + "px");
57485         this.updateSplitters();
57486         this.layout();
57487         this.grid.fireEvent("columnresize", i, w);
57488     },
57489
57490     syncRowHeights : function(startIndex, endIndex){
57491         if(this.grid.enableRowHeightSync === true && this.cm.getLockedCount() > 0){
57492             startIndex = startIndex || 0;
57493             var mrows = this.getBodyTable().rows;
57494             var lrows = this.getLockedTable().rows;
57495             var len = mrows.length-1;
57496             endIndex = Math.min(endIndex || len, len);
57497             for(var i = startIndex; i <= endIndex; i++){
57498                 var m = mrows[i], l = lrows[i];
57499                 var h = Math.max(m.offsetHeight, l.offsetHeight);
57500                 m.style.height = l.style.height = h + "px";
57501             }
57502         }
57503     },
57504
57505     layout : function(initialRender, is2ndPass){
57506         var g = this.grid;
57507         var auto = g.autoHeight;
57508         var scrollOffset = 16;
57509         var c = g.getGridEl(), cm = this.cm,
57510                 expandCol = g.autoExpandColumn,
57511                 gv = this;
57512         //c.beginMeasure();
57513
57514         if(!c.dom.offsetWidth){ // display:none?
57515             if(initialRender){
57516                 this.lockedWrap.show();
57517                 this.mainWrap.show();
57518             }
57519             return;
57520         }
57521
57522         var hasLock = this.cm.isLocked(0);
57523
57524         var tbh = this.headerPanel.getHeight();
57525         var bbh = this.footerPanel.getHeight();
57526
57527         if(auto){
57528             var ch = this.getBodyTable().offsetHeight + tbh + bbh + this.mainHd.getHeight();
57529             var newHeight = ch + c.getBorderWidth("tb");
57530             if(g.maxHeight){
57531                 newHeight = Math.min(g.maxHeight, newHeight);
57532             }
57533             c.setHeight(newHeight);
57534         }
57535
57536         if(g.autoWidth){
57537             c.setWidth(cm.getTotalWidth()+c.getBorderWidth('lr'));
57538         }
57539
57540         var s = this.scroller;
57541
57542         var csize = c.getSize(true);
57543
57544         this.el.setSize(csize.width, csize.height);
57545
57546         this.headerPanel.setWidth(csize.width);
57547         this.footerPanel.setWidth(csize.width);
57548
57549         var hdHeight = this.mainHd.getHeight();
57550         var vw = csize.width;
57551         var vh = csize.height - (tbh + bbh);
57552
57553         s.setSize(vw, vh);
57554
57555         var bt = this.getBodyTable();
57556         
57557         if(cm.getLockedCount() == cm.config.length){
57558             bt = this.getLockedTable();
57559         }
57560         
57561         var ltWidth = hasLock ?
57562                       Math.max(this.getLockedTable().offsetWidth, this.lockedHd.dom.firstChild.offsetWidth) : 0;
57563
57564         var scrollHeight = bt.offsetHeight;
57565         var scrollWidth = ltWidth + bt.offsetWidth;
57566         var vscroll = false, hscroll = false;
57567
57568         this.scrollSizer.setSize(scrollWidth, scrollHeight+hdHeight);
57569
57570         var lw = this.lockedWrap, mw = this.mainWrap;
57571         var lb = this.lockedBody, mb = this.mainBody;
57572
57573         setTimeout(function(){
57574             var t = s.dom.offsetTop;
57575             var w = s.dom.clientWidth,
57576                 h = s.dom.clientHeight;
57577
57578             lw.setTop(t);
57579             lw.setSize(ltWidth, h);
57580
57581             mw.setLeftTop(ltWidth, t);
57582             mw.setSize(w-ltWidth, h);
57583
57584             lb.setHeight(h-hdHeight);
57585             mb.setHeight(h-hdHeight);
57586
57587             if(is2ndPass !== true && !gv.userResized && expandCol){
57588                 // high speed resize without full column calculation
57589                 
57590                 var ci = cm.getIndexById(expandCol);
57591                 if (ci < 0) {
57592                     ci = cm.findColumnIndex(expandCol);
57593                 }
57594                 ci = Math.max(0, ci); // make sure it's got at least the first col.
57595                 var expandId = cm.getColumnId(ci);
57596                 var  tw = cm.getTotalWidth(false);
57597                 var currentWidth = cm.getColumnWidth(ci);
57598                 var cw = Math.min(Math.max(((w-tw)+currentWidth-2)-/*scrollbar*/(w <= s.dom.offsetWidth ? 0 : 18), g.autoExpandMin), g.autoExpandMax);
57599                 if(currentWidth != cw){
57600                     cm.setColumnWidth(ci, cw, true);
57601                     gv.css.updateRule(gv.colSelector+gv.idToCssName(expandId), "width", (cw - gv.borderWidth) + "px");
57602                     gv.css.updateRule(gv.hdSelector+gv.idToCssName(expandId), "width", (cw - gv.borderWidth) + "px");
57603                     gv.updateSplitters();
57604                     gv.layout(false, true);
57605                 }
57606             }
57607
57608             if(initialRender){
57609                 lw.show();
57610                 mw.show();
57611             }
57612             //c.endMeasure();
57613         }, 10);
57614     },
57615
57616     onWindowResize : function(){
57617         if(!this.grid.monitorWindowResize || this.grid.autoHeight){
57618             return;
57619         }
57620         this.layout();
57621     },
57622
57623     appendFooter : function(parentEl){
57624         return null;
57625     },
57626
57627     sortAscText : "Sort Ascending",
57628     sortDescText : "Sort Descending",
57629     lockText : "Lock Column",
57630     unlockText : "Unlock Column",
57631     columnsText : "Columns",
57632  
57633     columnsWiderText : "Wider",
57634     columnsNarrowText : "Thinner"
57635 });
57636
57637
57638 Roo.grid.GridView.ColumnDragZone = function(grid, hd){
57639     Roo.grid.GridView.ColumnDragZone.superclass.constructor.call(this, grid, hd, null);
57640     this.proxy.el.addClass('x-grid3-col-dd');
57641 };
57642
57643 Roo.extend(Roo.grid.GridView.ColumnDragZone, Roo.grid.HeaderDragZone, {
57644     handleMouseDown : function(e){
57645
57646     },
57647
57648     callHandleMouseDown : function(e){
57649         Roo.grid.GridView.ColumnDragZone.superclass.handleMouseDown.call(this, e);
57650     }
57651 });
57652 /*
57653  * Based on:
57654  * Ext JS Library 1.1.1
57655  * Copyright(c) 2006-2007, Ext JS, LLC.
57656  *
57657  * Originally Released Under LGPL - original licence link has changed is not relivant.
57658  *
57659  * Fork - LGPL
57660  * <script type="text/javascript">
57661  */
57662  
57663 // private
57664 // This is a support class used internally by the Grid components
57665 Roo.grid.SplitDragZone = function(grid, hd, hd2){
57666     this.grid = grid;
57667     this.view = grid.getView();
57668     this.proxy = this.view.resizeProxy;
57669     Roo.grid.SplitDragZone.superclass.constructor.call(this, hd,
57670         "gridSplitters" + this.grid.getGridEl().id, {
57671         dragElId : Roo.id(this.proxy.dom), resizeFrame:false
57672     });
57673     this.setHandleElId(Roo.id(hd));
57674     this.setOuterHandleElId(Roo.id(hd2));
57675     this.scroll = false;
57676 };
57677 Roo.extend(Roo.grid.SplitDragZone, Roo.dd.DDProxy, {
57678     fly: Roo.Element.fly,
57679
57680     b4StartDrag : function(x, y){
57681         this.view.headersDisabled = true;
57682         this.proxy.setHeight(this.view.mainWrap.getHeight());
57683         var w = this.cm.getColumnWidth(this.cellIndex);
57684         var minw = Math.max(w-this.grid.minColumnWidth, 0);
57685         this.resetConstraints();
57686         this.setXConstraint(minw, 1000);
57687         this.setYConstraint(0, 0);
57688         this.minX = x - minw;
57689         this.maxX = x + 1000;
57690         this.startPos = x;
57691         Roo.dd.DDProxy.prototype.b4StartDrag.call(this, x, y);
57692     },
57693
57694
57695     handleMouseDown : function(e){
57696         ev = Roo.EventObject.setEvent(e);
57697         var t = this.fly(ev.getTarget());
57698         if(t.hasClass("x-grid-split")){
57699             this.cellIndex = this.view.getCellIndex(t.dom);
57700             this.split = t.dom;
57701             this.cm = this.grid.colModel;
57702             if(this.cm.isResizable(this.cellIndex) && !this.cm.isFixed(this.cellIndex)){
57703                 Roo.grid.SplitDragZone.superclass.handleMouseDown.apply(this, arguments);
57704             }
57705         }
57706     },
57707
57708     endDrag : function(e){
57709         this.view.headersDisabled = false;
57710         var endX = Math.max(this.minX, Roo.lib.Event.getPageX(e));
57711         var diff = endX - this.startPos;
57712         this.view.onColumnSplitterMoved(this.cellIndex, this.cm.getColumnWidth(this.cellIndex)+diff);
57713     },
57714
57715     autoOffset : function(){
57716         this.setDelta(0,0);
57717     }
57718 });/*
57719  * Based on:
57720  * Ext JS Library 1.1.1
57721  * Copyright(c) 2006-2007, Ext JS, LLC.
57722  *
57723  * Originally Released Under LGPL - original licence link has changed is not relivant.
57724  *
57725  * Fork - LGPL
57726  * <script type="text/javascript">
57727  */
57728  
57729 // private
57730 // This is a support class used internally by the Grid components
57731 Roo.grid.GridDragZone = function(grid, config){
57732     this.view = grid.getView();
57733     Roo.grid.GridDragZone.superclass.constructor.call(this, this.view.mainBody.dom, config);
57734     if(this.view.lockedBody){
57735         this.setHandleElId(Roo.id(this.view.mainBody.dom));
57736         this.setOuterHandleElId(Roo.id(this.view.lockedBody.dom));
57737     }
57738     this.scroll = false;
57739     this.grid = grid;
57740     this.ddel = document.createElement('div');
57741     this.ddel.className = 'x-grid-dd-wrap';
57742 };
57743
57744 Roo.extend(Roo.grid.GridDragZone, Roo.dd.DragZone, {
57745     ddGroup : "GridDD",
57746
57747     getDragData : function(e){
57748         var t = Roo.lib.Event.getTarget(e);
57749         var rowIndex = this.view.findRowIndex(t);
57750         var sm = this.grid.selModel;
57751             
57752         //Roo.log(rowIndex);
57753         
57754         if (sm.getSelectedCell) {
57755             // cell selection..
57756             if (!sm.getSelectedCell()) {
57757                 return false;
57758             }
57759             if (rowIndex != sm.getSelectedCell()[0]) {
57760                 return false;
57761             }
57762         
57763         }
57764         
57765         if(rowIndex !== false){
57766             
57767             // if editorgrid.. 
57768             
57769             
57770             //Roo.log([ sm.getSelectedCell() ? sm.getSelectedCell()[0] : 'NO' , rowIndex ]);
57771                
57772             //if(!sm.isSelected(rowIndex) || e.hasModifier()){
57773               //  
57774             //}
57775             if (e.hasModifier()){
57776                 sm.handleMouseDown(e, t); // non modifier buttons are handled by row select.
57777             }
57778             
57779             Roo.log("getDragData");
57780             
57781             return {
57782                 grid: this.grid,
57783                 ddel: this.ddel,
57784                 rowIndex: rowIndex,
57785                 selections:sm.getSelections ? sm.getSelections() : (
57786                     sm.getSelectedCell() ? [ this.grid.ds.getAt(sm.getSelectedCell()[0]) ] : []
57787                 )
57788             };
57789         }
57790         return false;
57791     },
57792
57793     onInitDrag : function(e){
57794         var data = this.dragData;
57795         this.ddel.innerHTML = this.grid.getDragDropText();
57796         this.proxy.update(this.ddel);
57797         // fire start drag?
57798     },
57799
57800     afterRepair : function(){
57801         this.dragging = false;
57802     },
57803
57804     getRepairXY : function(e, data){
57805         return false;
57806     },
57807
57808     onEndDrag : function(data, e){
57809         // fire end drag?
57810     },
57811
57812     onValidDrop : function(dd, e, id){
57813         // fire drag drop?
57814         this.hideProxy();
57815     },
57816
57817     beforeInvalidDrop : function(e, id){
57818
57819     }
57820 });/*
57821  * Based on:
57822  * Ext JS Library 1.1.1
57823  * Copyright(c) 2006-2007, Ext JS, LLC.
57824  *
57825  * Originally Released Under LGPL - original licence link has changed is not relivant.
57826  *
57827  * Fork - LGPL
57828  * <script type="text/javascript">
57829  */
57830  
57831
57832 /**
57833  * @class Roo.grid.ColumnModel
57834  * @extends Roo.util.Observable
57835  * This is the default implementation of a ColumnModel used by the Grid. It defines
57836  * the columns in the grid.
57837  * <br>Usage:<br>
57838  <pre><code>
57839  var colModel = new Roo.grid.ColumnModel([
57840         {header: "Ticker", width: 60, sortable: true, locked: true},
57841         {header: "Company Name", width: 150, sortable: true},
57842         {header: "Market Cap.", width: 100, sortable: true},
57843         {header: "$ Sales", width: 100, sortable: true, renderer: money},
57844         {header: "Employees", width: 100, sortable: true, resizable: false}
57845  ]);
57846  </code></pre>
57847  * <p>
57848  
57849  * The config options listed for this class are options which may appear in each
57850  * individual column definition.
57851  * <br/>RooJS Fix - column id's are not sequential but use Roo.id() - fixes bugs with layouts.
57852  * @constructor
57853  * @param {Object} config An Array of column config objects. See this class's
57854  * config objects for details.
57855 */
57856 Roo.grid.ColumnModel = function(config){
57857         /**
57858      * The config passed into the constructor
57859      */
57860     this.config = config;
57861     this.lookup = {};
57862
57863     // if no id, create one
57864     // if the column does not have a dataIndex mapping,
57865     // map it to the order it is in the config
57866     for(var i = 0, len = config.length; i < len; i++){
57867         var c = config[i];
57868         if(typeof c.dataIndex == "undefined"){
57869             c.dataIndex = i;
57870         }
57871         if(typeof c.renderer == "string"){
57872             c.renderer = Roo.util.Format[c.renderer];
57873         }
57874         if(typeof c.id == "undefined"){
57875             c.id = Roo.id();
57876         }
57877         if(c.editor && c.editor.xtype){
57878             c.editor  = Roo.factory(c.editor, Roo.grid);
57879         }
57880         if(c.editor && c.editor.isFormField){
57881             c.editor = new Roo.grid.GridEditor(c.editor);
57882         }
57883         this.lookup[c.id] = c;
57884     }
57885
57886     /**
57887      * The width of columns which have no width specified (defaults to 100)
57888      * @type Number
57889      */
57890     this.defaultWidth = 100;
57891
57892     /**
57893      * Default sortable of columns which have no sortable specified (defaults to false)
57894      * @type Boolean
57895      */
57896     this.defaultSortable = false;
57897
57898     this.addEvents({
57899         /**
57900              * @event widthchange
57901              * Fires when the width of a column changes.
57902              * @param {ColumnModel} this
57903              * @param {Number} columnIndex The column index
57904              * @param {Number} newWidth The new width
57905              */
57906             "widthchange": true,
57907         /**
57908              * @event headerchange
57909              * Fires when the text of a header changes.
57910              * @param {ColumnModel} this
57911              * @param {Number} columnIndex The column index
57912              * @param {Number} newText The new header text
57913              */
57914             "headerchange": true,
57915         /**
57916              * @event hiddenchange
57917              * Fires when a column is hidden or "unhidden".
57918              * @param {ColumnModel} this
57919              * @param {Number} columnIndex The column index
57920              * @param {Boolean} hidden true if hidden, false otherwise
57921              */
57922             "hiddenchange": true,
57923             /**
57924          * @event columnmoved
57925          * Fires when a column is moved.
57926          * @param {ColumnModel} this
57927          * @param {Number} oldIndex
57928          * @param {Number} newIndex
57929          */
57930         "columnmoved" : true,
57931         /**
57932          * @event columlockchange
57933          * Fires when a column's locked state is changed
57934          * @param {ColumnModel} this
57935          * @param {Number} colIndex
57936          * @param {Boolean} locked true if locked
57937          */
57938         "columnlockchange" : true
57939     });
57940     Roo.grid.ColumnModel.superclass.constructor.call(this);
57941 };
57942 Roo.extend(Roo.grid.ColumnModel, Roo.util.Observable, {
57943     /**
57944      * @cfg {String} header The header text to display in the Grid view.
57945      */
57946     /**
57947      * @cfg {String} dataIndex (Optional) The name of the field in the grid's {@link Roo.data.Store}'s
57948      * {@link Roo.data.Record} definition from which to draw the column's value. If not
57949      * specified, the column's index is used as an index into the Record's data Array.
57950      */
57951     /**
57952      * @cfg {Number} width (Optional) The initial width in pixels of the column. Using this
57953      * instead of {@link Roo.grid.Grid#autoSizeColumns} is more efficient.
57954      */
57955     /**
57956      * @cfg {Boolean} sortable (Optional) True if sorting is to be allowed on this column.
57957      * Defaults to the value of the {@link #defaultSortable} property.
57958      * Whether local/remote sorting is used is specified in {@link Roo.data.Store#remoteSort}.
57959      */
57960     /**
57961      * @cfg {Boolean} locked (Optional) True to lock the column in place while scrolling the Grid.  Defaults to false.
57962      */
57963     /**
57964      * @cfg {Boolean} fixed (Optional) True if the column width cannot be changed.  Defaults to false.
57965      */
57966     /**
57967      * @cfg {Boolean} resizable (Optional) False to disable column resizing. Defaults to true.
57968      */
57969     /**
57970      * @cfg {Boolean} hidden (Optional) True to hide the column. Defaults to false.
57971      */
57972     /**
57973      * @cfg {Function} renderer (Optional) A function used to generate HTML markup for a cell
57974      * given the cell's data value. See {@link #setRenderer}. If not specified, the
57975      * default renderer returns the escaped data value. If an object is returned (bootstrap only)
57976      * then it is treated as a Roo Component object instance, and it is rendered after the initial row is rendered
57977      */
57978        /**
57979      * @cfg {Roo.grid.GridEditor} editor (Optional) For grid editors - returns the grid editor 
57980      */
57981     /**
57982      * @cfg {String} align (Optional) Set the CSS text-align property of the column.  Defaults to undefined.
57983      */
57984     /**
57985      * @cfg {String} valign (Optional) Set the CSS vertical-align property of the column (eg. middle, top, bottom etc).  Defaults to undefined.
57986      */
57987     /**
57988      * @cfg {String} cursor (Optional)
57989      */
57990     /**
57991      * @cfg {String} tooltip (Optional)
57992      */
57993     /**
57994      * @cfg {Number} xs (Optional)
57995      */
57996     /**
57997      * @cfg {Number} sm (Optional)
57998      */
57999     /**
58000      * @cfg {Number} md (Optional)
58001      */
58002     /**
58003      * @cfg {Number} lg (Optional)
58004      */
58005     /**
58006      * Returns the id of the column at the specified index.
58007      * @param {Number} index The column index
58008      * @return {String} the id
58009      */
58010     getColumnId : function(index){
58011         return this.config[index].id;
58012     },
58013
58014     /**
58015      * Returns the column for a specified id.
58016      * @param {String} id The column id
58017      * @return {Object} the column
58018      */
58019     getColumnById : function(id){
58020         return this.lookup[id];
58021     },
58022
58023     
58024     /**
58025      * Returns the column for a specified dataIndex.
58026      * @param {String} dataIndex The column dataIndex
58027      * @return {Object|Boolean} the column or false if not found
58028      */
58029     getColumnByDataIndex: function(dataIndex){
58030         var index = this.findColumnIndex(dataIndex);
58031         return index > -1 ? this.config[index] : false;
58032     },
58033     
58034     /**
58035      * Returns the index for a specified column id.
58036      * @param {String} id The column id
58037      * @return {Number} the index, or -1 if not found
58038      */
58039     getIndexById : function(id){
58040         for(var i = 0, len = this.config.length; i < len; i++){
58041             if(this.config[i].id == id){
58042                 return i;
58043             }
58044         }
58045         return -1;
58046     },
58047     
58048     /**
58049      * Returns the index for a specified column dataIndex.
58050      * @param {String} dataIndex The column dataIndex
58051      * @return {Number} the index, or -1 if not found
58052      */
58053     
58054     findColumnIndex : function(dataIndex){
58055         for(var i = 0, len = this.config.length; i < len; i++){
58056             if(this.config[i].dataIndex == dataIndex){
58057                 return i;
58058             }
58059         }
58060         return -1;
58061     },
58062     
58063     
58064     moveColumn : function(oldIndex, newIndex){
58065         var c = this.config[oldIndex];
58066         this.config.splice(oldIndex, 1);
58067         this.config.splice(newIndex, 0, c);
58068         this.dataMap = null;
58069         this.fireEvent("columnmoved", this, oldIndex, newIndex);
58070     },
58071
58072     isLocked : function(colIndex){
58073         return this.config[colIndex].locked === true;
58074     },
58075
58076     setLocked : function(colIndex, value, suppressEvent){
58077         if(this.isLocked(colIndex) == value){
58078             return;
58079         }
58080         this.config[colIndex].locked = value;
58081         if(!suppressEvent){
58082             this.fireEvent("columnlockchange", this, colIndex, value);
58083         }
58084     },
58085
58086     getTotalLockedWidth : function(){
58087         var totalWidth = 0;
58088         for(var i = 0; i < this.config.length; i++){
58089             if(this.isLocked(i) && !this.isHidden(i)){
58090                 this.totalWidth += this.getColumnWidth(i);
58091             }
58092         }
58093         return totalWidth;
58094     },
58095
58096     getLockedCount : function(){
58097         for(var i = 0, len = this.config.length; i < len; i++){
58098             if(!this.isLocked(i)){
58099                 return i;
58100             }
58101         }
58102         
58103         return this.config.length;
58104     },
58105
58106     /**
58107      * Returns the number of columns.
58108      * @return {Number}
58109      */
58110     getColumnCount : function(visibleOnly){
58111         if(visibleOnly === true){
58112             var c = 0;
58113             for(var i = 0, len = this.config.length; i < len; i++){
58114                 if(!this.isHidden(i)){
58115                     c++;
58116                 }
58117             }
58118             return c;
58119         }
58120         return this.config.length;
58121     },
58122
58123     /**
58124      * Returns the column configs that return true by the passed function that is called with (columnConfig, index)
58125      * @param {Function} fn
58126      * @param {Object} scope (optional)
58127      * @return {Array} result
58128      */
58129     getColumnsBy : function(fn, scope){
58130         var r = [];
58131         for(var i = 0, len = this.config.length; i < len; i++){
58132             var c = this.config[i];
58133             if(fn.call(scope||this, c, i) === true){
58134                 r[r.length] = c;
58135             }
58136         }
58137         return r;
58138     },
58139
58140     /**
58141      * Returns true if the specified column is sortable.
58142      * @param {Number} col The column index
58143      * @return {Boolean}
58144      */
58145     isSortable : function(col){
58146         if(typeof this.config[col].sortable == "undefined"){
58147             return this.defaultSortable;
58148         }
58149         return this.config[col].sortable;
58150     },
58151
58152     /**
58153      * Returns the rendering (formatting) function defined for the column.
58154      * @param {Number} col The column index.
58155      * @return {Function} The function used to render the cell. See {@link #setRenderer}.
58156      */
58157     getRenderer : function(col){
58158         if(!this.config[col].renderer){
58159             return Roo.grid.ColumnModel.defaultRenderer;
58160         }
58161         return this.config[col].renderer;
58162     },
58163
58164     /**
58165      * Sets the rendering (formatting) function for a column.
58166      * @param {Number} col The column index
58167      * @param {Function} fn The function to use to process the cell's raw data
58168      * to return HTML markup for the grid view. The render function is called with
58169      * the following parameters:<ul>
58170      * <li>Data value.</li>
58171      * <li>Cell metadata. An object in which you may set the following attributes:<ul>
58172      * <li>css A CSS style string to apply to the table cell.</li>
58173      * <li>attr An HTML attribute definition string to apply to the data container element <i>within</i> the table cell.</li></ul>
58174      * <li>The {@link Roo.data.Record} from which the data was extracted.</li>
58175      * <li>Row index</li>
58176      * <li>Column index</li>
58177      * <li>The {@link Roo.data.Store} object from which the Record was extracted</li></ul>
58178      */
58179     setRenderer : function(col, fn){
58180         this.config[col].renderer = fn;
58181     },
58182
58183     /**
58184      * Returns the width for the specified column.
58185      * @param {Number} col The column index
58186      * @return {Number}
58187      */
58188     getColumnWidth : function(col){
58189         return this.config[col].width * 1 || this.defaultWidth;
58190     },
58191
58192     /**
58193      * Sets the width for a column.
58194      * @param {Number} col The column index
58195      * @param {Number} width The new width
58196      */
58197     setColumnWidth : function(col, width, suppressEvent){
58198         this.config[col].width = width;
58199         this.totalWidth = null;
58200         if(!suppressEvent){
58201              this.fireEvent("widthchange", this, col, width);
58202         }
58203     },
58204
58205     /**
58206      * Returns the total width of all columns.
58207      * @param {Boolean} includeHidden True to include hidden column widths
58208      * @return {Number}
58209      */
58210     getTotalWidth : function(includeHidden){
58211         if(!this.totalWidth){
58212             this.totalWidth = 0;
58213             for(var i = 0, len = this.config.length; i < len; i++){
58214                 if(includeHidden || !this.isHidden(i)){
58215                     this.totalWidth += this.getColumnWidth(i);
58216                 }
58217             }
58218         }
58219         return this.totalWidth;
58220     },
58221
58222     /**
58223      * Returns the header for the specified column.
58224      * @param {Number} col The column index
58225      * @return {String}
58226      */
58227     getColumnHeader : function(col){
58228         return this.config[col].header;
58229     },
58230
58231     /**
58232      * Sets the header for a column.
58233      * @param {Number} col The column index
58234      * @param {String} header The new header
58235      */
58236     setColumnHeader : function(col, header){
58237         this.config[col].header = header;
58238         this.fireEvent("headerchange", this, col, header);
58239     },
58240
58241     /**
58242      * Returns the tooltip for the specified column.
58243      * @param {Number} col The column index
58244      * @return {String}
58245      */
58246     getColumnTooltip : function(col){
58247             return this.config[col].tooltip;
58248     },
58249     /**
58250      * Sets the tooltip for a column.
58251      * @param {Number} col The column index
58252      * @param {String} tooltip The new tooltip
58253      */
58254     setColumnTooltip : function(col, tooltip){
58255             this.config[col].tooltip = tooltip;
58256     },
58257
58258     /**
58259      * Returns the dataIndex for the specified column.
58260      * @param {Number} col The column index
58261      * @return {Number}
58262      */
58263     getDataIndex : function(col){
58264         return this.config[col].dataIndex;
58265     },
58266
58267     /**
58268      * Sets the dataIndex for a column.
58269      * @param {Number} col The column index
58270      * @param {Number} dataIndex The new dataIndex
58271      */
58272     setDataIndex : function(col, dataIndex){
58273         this.config[col].dataIndex = dataIndex;
58274     },
58275
58276     
58277     
58278     /**
58279      * Returns true if the cell is editable.
58280      * @param {Number} colIndex The column index
58281      * @param {Number} rowIndex The row index - this is nto actually used..?
58282      * @return {Boolean}
58283      */
58284     isCellEditable : function(colIndex, rowIndex){
58285         return (this.config[colIndex].editable || (typeof this.config[colIndex].editable == "undefined" && this.config[colIndex].editor)) ? true : false;
58286     },
58287
58288     /**
58289      * Returns the editor defined for the cell/column.
58290      * return false or null to disable editing.
58291      * @param {Number} colIndex The column index
58292      * @param {Number} rowIndex The row index
58293      * @return {Object}
58294      */
58295     getCellEditor : function(colIndex, rowIndex){
58296         return this.config[colIndex].editor;
58297     },
58298
58299     /**
58300      * Sets if a column is editable.
58301      * @param {Number} col The column index
58302      * @param {Boolean} editable True if the column is editable
58303      */
58304     setEditable : function(col, editable){
58305         this.config[col].editable = editable;
58306     },
58307
58308
58309     /**
58310      * Returns true if the column is hidden.
58311      * @param {Number} colIndex The column index
58312      * @return {Boolean}
58313      */
58314     isHidden : function(colIndex){
58315         return this.config[colIndex].hidden;
58316     },
58317
58318
58319     /**
58320      * Returns true if the column width cannot be changed
58321      */
58322     isFixed : function(colIndex){
58323         return this.config[colIndex].fixed;
58324     },
58325
58326     /**
58327      * Returns true if the column can be resized
58328      * @return {Boolean}
58329      */
58330     isResizable : function(colIndex){
58331         return colIndex >= 0 && this.config[colIndex].resizable !== false && this.config[colIndex].fixed !== true;
58332     },
58333     /**
58334      * Sets if a column is hidden.
58335      * @param {Number} colIndex The column index
58336      * @param {Boolean} hidden True if the column is hidden
58337      */
58338     setHidden : function(colIndex, hidden){
58339         this.config[colIndex].hidden = hidden;
58340         this.totalWidth = null;
58341         this.fireEvent("hiddenchange", this, colIndex, hidden);
58342     },
58343
58344     /**
58345      * Sets the editor for a column.
58346      * @param {Number} col The column index
58347      * @param {Object} editor The editor object
58348      */
58349     setEditor : function(col, editor){
58350         this.config[col].editor = editor;
58351     }
58352 });
58353
58354 Roo.grid.ColumnModel.defaultRenderer = function(value)
58355 {
58356     if(typeof value == "object") {
58357         return value;
58358     }
58359         if(typeof value == "string" && value.length < 1){
58360             return "&#160;";
58361         }
58362     
58363         return String.format("{0}", value);
58364 };
58365
58366 // Alias for backwards compatibility
58367 Roo.grid.DefaultColumnModel = Roo.grid.ColumnModel;
58368 /*
58369  * Based on:
58370  * Ext JS Library 1.1.1
58371  * Copyright(c) 2006-2007, Ext JS, LLC.
58372  *
58373  * Originally Released Under LGPL - original licence link has changed is not relivant.
58374  *
58375  * Fork - LGPL
58376  * <script type="text/javascript">
58377  */
58378
58379 /**
58380  * @class Roo.grid.AbstractSelectionModel
58381  * @extends Roo.util.Observable
58382  * Abstract base class for grid SelectionModels.  It provides the interface that should be
58383  * implemented by descendant classes.  This class should not be directly instantiated.
58384  * @constructor
58385  */
58386 Roo.grid.AbstractSelectionModel = function(){
58387     this.locked = false;
58388     Roo.grid.AbstractSelectionModel.superclass.constructor.call(this);
58389 };
58390
58391 Roo.extend(Roo.grid.AbstractSelectionModel, Roo.util.Observable,  {
58392     /** @ignore Called by the grid automatically. Do not call directly. */
58393     init : function(grid){
58394         this.grid = grid;
58395         this.initEvents();
58396     },
58397
58398     /**
58399      * Locks the selections.
58400      */
58401     lock : function(){
58402         this.locked = true;
58403     },
58404
58405     /**
58406      * Unlocks the selections.
58407      */
58408     unlock : function(){
58409         this.locked = false;
58410     },
58411
58412     /**
58413      * Returns true if the selections are locked.
58414      * @return {Boolean}
58415      */
58416     isLocked : function(){
58417         return this.locked;
58418     }
58419 });/*
58420  * Based on:
58421  * Ext JS Library 1.1.1
58422  * Copyright(c) 2006-2007, Ext JS, LLC.
58423  *
58424  * Originally Released Under LGPL - original licence link has changed is not relivant.
58425  *
58426  * Fork - LGPL
58427  * <script type="text/javascript">
58428  */
58429 /**
58430  * @extends Roo.grid.AbstractSelectionModel
58431  * @class Roo.grid.RowSelectionModel
58432  * The default SelectionModel used by {@link Roo.grid.Grid}.
58433  * It supports multiple selections and keyboard selection/navigation. 
58434  * @constructor
58435  * @param {Object} config
58436  */
58437 Roo.grid.RowSelectionModel = function(config){
58438     Roo.apply(this, config);
58439     this.selections = new Roo.util.MixedCollection(false, function(o){
58440         return o.id;
58441     });
58442
58443     this.last = false;
58444     this.lastActive = false;
58445
58446     this.addEvents({
58447         /**
58448              * @event selectionchange
58449              * Fires when the selection changes
58450              * @param {SelectionModel} this
58451              */
58452             "selectionchange" : true,
58453         /**
58454              * @event afterselectionchange
58455              * Fires after the selection changes (eg. by key press or clicking)
58456              * @param {SelectionModel} this
58457              */
58458             "afterselectionchange" : true,
58459         /**
58460              * @event beforerowselect
58461              * Fires when a row is selected being selected, return false to cancel.
58462              * @param {SelectionModel} this
58463              * @param {Number} rowIndex The selected index
58464              * @param {Boolean} keepExisting False if other selections will be cleared
58465              */
58466             "beforerowselect" : true,
58467         /**
58468              * @event rowselect
58469              * Fires when a row is selected.
58470              * @param {SelectionModel} this
58471              * @param {Number} rowIndex The selected index
58472              * @param {Roo.data.Record} r The record
58473              */
58474             "rowselect" : true,
58475         /**
58476              * @event rowdeselect
58477              * Fires when a row is deselected.
58478              * @param {SelectionModel} this
58479              * @param {Number} rowIndex The selected index
58480              */
58481         "rowdeselect" : true
58482     });
58483     Roo.grid.RowSelectionModel.superclass.constructor.call(this);
58484     this.locked = false;
58485 };
58486
58487 Roo.extend(Roo.grid.RowSelectionModel, Roo.grid.AbstractSelectionModel,  {
58488     /**
58489      * @cfg {Boolean} singleSelect
58490      * True to allow selection of only one row at a time (defaults to false)
58491      */
58492     singleSelect : false,
58493
58494     // private
58495     initEvents : function(){
58496
58497         if(!this.grid.enableDragDrop && !this.grid.enableDrag){
58498             this.grid.on("mousedown", this.handleMouseDown, this);
58499         }else{ // allow click to work like normal
58500             this.grid.on("rowclick", this.handleDragableRowClick, this);
58501         }
58502
58503         this.rowNav = new Roo.KeyNav(this.grid.getGridEl(), {
58504             "up" : function(e){
58505                 if(!e.shiftKey){
58506                     this.selectPrevious(e.shiftKey);
58507                 }else if(this.last !== false && this.lastActive !== false){
58508                     var last = this.last;
58509                     this.selectRange(this.last,  this.lastActive-1);
58510                     this.grid.getView().focusRow(this.lastActive);
58511                     if(last !== false){
58512                         this.last = last;
58513                     }
58514                 }else{
58515                     this.selectFirstRow();
58516                 }
58517                 this.fireEvent("afterselectionchange", this);
58518             },
58519             "down" : function(e){
58520                 if(!e.shiftKey){
58521                     this.selectNext(e.shiftKey);
58522                 }else if(this.last !== false && this.lastActive !== false){
58523                     var last = this.last;
58524                     this.selectRange(this.last,  this.lastActive+1);
58525                     this.grid.getView().focusRow(this.lastActive);
58526                     if(last !== false){
58527                         this.last = last;
58528                     }
58529                 }else{
58530                     this.selectFirstRow();
58531                 }
58532                 this.fireEvent("afterselectionchange", this);
58533             },
58534             scope: this
58535         });
58536
58537         var view = this.grid.view;
58538         view.on("refresh", this.onRefresh, this);
58539         view.on("rowupdated", this.onRowUpdated, this);
58540         view.on("rowremoved", this.onRemove, this);
58541     },
58542
58543     // private
58544     onRefresh : function(){
58545         var ds = this.grid.dataSource, i, v = this.grid.view;
58546         var s = this.selections;
58547         s.each(function(r){
58548             if((i = ds.indexOfId(r.id)) != -1){
58549                 v.onRowSelect(i);
58550                 s.add(ds.getAt(i)); // updating the selection relate data
58551             }else{
58552                 s.remove(r);
58553             }
58554         });
58555     },
58556
58557     // private
58558     onRemove : function(v, index, r){
58559         this.selections.remove(r);
58560     },
58561
58562     // private
58563     onRowUpdated : function(v, index, r){
58564         if(this.isSelected(r)){
58565             v.onRowSelect(index);
58566         }
58567     },
58568
58569     /**
58570      * Select records.
58571      * @param {Array} records The records to select
58572      * @param {Boolean} keepExisting (optional) True to keep existing selections
58573      */
58574     selectRecords : function(records, keepExisting){
58575         if(!keepExisting){
58576             this.clearSelections();
58577         }
58578         var ds = this.grid.dataSource;
58579         for(var i = 0, len = records.length; i < len; i++){
58580             this.selectRow(ds.indexOf(records[i]), true);
58581         }
58582     },
58583
58584     /**
58585      * Gets the number of selected rows.
58586      * @return {Number}
58587      */
58588     getCount : function(){
58589         return this.selections.length;
58590     },
58591
58592     /**
58593      * Selects the first row in the grid.
58594      */
58595     selectFirstRow : function(){
58596         this.selectRow(0);
58597     },
58598
58599     /**
58600      * Select the last row.
58601      * @param {Boolean} keepExisting (optional) True to keep existing selections
58602      */
58603     selectLastRow : function(keepExisting){
58604         this.selectRow(this.grid.dataSource.getCount() - 1, keepExisting);
58605     },
58606
58607     /**
58608      * Selects the row immediately following the last selected row.
58609      * @param {Boolean} keepExisting (optional) True to keep existing selections
58610      */
58611     selectNext : function(keepExisting){
58612         if(this.last !== false && (this.last+1) < this.grid.dataSource.getCount()){
58613             this.selectRow(this.last+1, keepExisting);
58614             this.grid.getView().focusRow(this.last);
58615         }
58616     },
58617
58618     /**
58619      * Selects the row that precedes the last selected row.
58620      * @param {Boolean} keepExisting (optional) True to keep existing selections
58621      */
58622     selectPrevious : function(keepExisting){
58623         if(this.last){
58624             this.selectRow(this.last-1, keepExisting);
58625             this.grid.getView().focusRow(this.last);
58626         }
58627     },
58628
58629     /**
58630      * Returns the selected records
58631      * @return {Array} Array of selected records
58632      */
58633     getSelections : function(){
58634         return [].concat(this.selections.items);
58635     },
58636
58637     /**
58638      * Returns the first selected record.
58639      * @return {Record}
58640      */
58641     getSelected : function(){
58642         return this.selections.itemAt(0);
58643     },
58644
58645
58646     /**
58647      * Clears all selections.
58648      */
58649     clearSelections : function(fast){
58650         if(this.locked) {
58651             return;
58652         }
58653         if(fast !== true){
58654             var ds = this.grid.dataSource;
58655             var s = this.selections;
58656             s.each(function(r){
58657                 this.deselectRow(ds.indexOfId(r.id));
58658             }, this);
58659             s.clear();
58660         }else{
58661             this.selections.clear();
58662         }
58663         this.last = false;
58664     },
58665
58666
58667     /**
58668      * Selects all rows.
58669      */
58670     selectAll : function(){
58671         if(this.locked) {
58672             return;
58673         }
58674         this.selections.clear();
58675         for(var i = 0, len = this.grid.dataSource.getCount(); i < len; i++){
58676             this.selectRow(i, true);
58677         }
58678     },
58679
58680     /**
58681      * Returns True if there is a selection.
58682      * @return {Boolean}
58683      */
58684     hasSelection : function(){
58685         return this.selections.length > 0;
58686     },
58687
58688     /**
58689      * Returns True if the specified row is selected.
58690      * @param {Number/Record} record The record or index of the record to check
58691      * @return {Boolean}
58692      */
58693     isSelected : function(index){
58694         var r = typeof index == "number" ? this.grid.dataSource.getAt(index) : index;
58695         return (r && this.selections.key(r.id) ? true : false);
58696     },
58697
58698     /**
58699      * Returns True if the specified record id is selected.
58700      * @param {String} id The id of record to check
58701      * @return {Boolean}
58702      */
58703     isIdSelected : function(id){
58704         return (this.selections.key(id) ? true : false);
58705     },
58706
58707     // private
58708     handleMouseDown : function(e, t){
58709         var view = this.grid.getView(), rowIndex;
58710         if(this.isLocked() || (rowIndex = view.findRowIndex(t)) === false){
58711             return;
58712         };
58713         if(e.shiftKey && this.last !== false){
58714             var last = this.last;
58715             this.selectRange(last, rowIndex, e.ctrlKey);
58716             this.last = last; // reset the last
58717             view.focusRow(rowIndex);
58718         }else{
58719             var isSelected = this.isSelected(rowIndex);
58720             if(e.button !== 0 && isSelected){
58721                 view.focusRow(rowIndex);
58722             }else if(e.ctrlKey && isSelected){
58723                 this.deselectRow(rowIndex);
58724             }else if(!isSelected){
58725                 this.selectRow(rowIndex, e.button === 0 && (e.ctrlKey || e.shiftKey));
58726                 view.focusRow(rowIndex);
58727             }
58728         }
58729         this.fireEvent("afterselectionchange", this);
58730     },
58731     // private
58732     handleDragableRowClick :  function(grid, rowIndex, e) 
58733     {
58734         if(e.button === 0 && !e.shiftKey && !e.ctrlKey) {
58735             this.selectRow(rowIndex, false);
58736             grid.view.focusRow(rowIndex);
58737              this.fireEvent("afterselectionchange", this);
58738         }
58739     },
58740     
58741     /**
58742      * Selects multiple rows.
58743      * @param {Array} rows Array of the indexes of the row to select
58744      * @param {Boolean} keepExisting (optional) True to keep existing selections
58745      */
58746     selectRows : function(rows, keepExisting){
58747         if(!keepExisting){
58748             this.clearSelections();
58749         }
58750         for(var i = 0, len = rows.length; i < len; i++){
58751             this.selectRow(rows[i], true);
58752         }
58753     },
58754
58755     /**
58756      * Selects a range of rows. All rows in between startRow and endRow are also selected.
58757      * @param {Number} startRow The index of the first row in the range
58758      * @param {Number} endRow The index of the last row in the range
58759      * @param {Boolean} keepExisting (optional) True to retain existing selections
58760      */
58761     selectRange : function(startRow, endRow, keepExisting){
58762         if(this.locked) {
58763             return;
58764         }
58765         if(!keepExisting){
58766             this.clearSelections();
58767         }
58768         if(startRow <= endRow){
58769             for(var i = startRow; i <= endRow; i++){
58770                 this.selectRow(i, true);
58771             }
58772         }else{
58773             for(var i = startRow; i >= endRow; i--){
58774                 this.selectRow(i, true);
58775             }
58776         }
58777     },
58778
58779     /**
58780      * Deselects a range of rows. All rows in between startRow and endRow are also deselected.
58781      * @param {Number} startRow The index of the first row in the range
58782      * @param {Number} endRow The index of the last row in the range
58783      */
58784     deselectRange : function(startRow, endRow, preventViewNotify){
58785         if(this.locked) {
58786             return;
58787         }
58788         for(var i = startRow; i <= endRow; i++){
58789             this.deselectRow(i, preventViewNotify);
58790         }
58791     },
58792
58793     /**
58794      * Selects a row.
58795      * @param {Number} row The index of the row to select
58796      * @param {Boolean} keepExisting (optional) True to keep existing selections
58797      */
58798     selectRow : function(index, keepExisting, preventViewNotify){
58799         if(this.locked || (index < 0 || index >= this.grid.dataSource.getCount())) {
58800             return;
58801         }
58802         if(this.fireEvent("beforerowselect", this, index, keepExisting) !== false){
58803             if(!keepExisting || this.singleSelect){
58804                 this.clearSelections();
58805             }
58806             var r = this.grid.dataSource.getAt(index);
58807             this.selections.add(r);
58808             this.last = this.lastActive = index;
58809             if(!preventViewNotify){
58810                 this.grid.getView().onRowSelect(index);
58811             }
58812             this.fireEvent("rowselect", this, index, r);
58813             this.fireEvent("selectionchange", this);
58814         }
58815     },
58816
58817     /**
58818      * Deselects a row.
58819      * @param {Number} row The index of the row to deselect
58820      */
58821     deselectRow : function(index, preventViewNotify){
58822         if(this.locked) {
58823             return;
58824         }
58825         if(this.last == index){
58826             this.last = false;
58827         }
58828         if(this.lastActive == index){
58829             this.lastActive = false;
58830         }
58831         var r = this.grid.dataSource.getAt(index);
58832         this.selections.remove(r);
58833         if(!preventViewNotify){
58834             this.grid.getView().onRowDeselect(index);
58835         }
58836         this.fireEvent("rowdeselect", this, index);
58837         this.fireEvent("selectionchange", this);
58838     },
58839
58840     // private
58841     restoreLast : function(){
58842         if(this._last){
58843             this.last = this._last;
58844         }
58845     },
58846
58847     // private
58848     acceptsNav : function(row, col, cm){
58849         return !cm.isHidden(col) && cm.isCellEditable(col, row);
58850     },
58851
58852     // private
58853     onEditorKey : function(field, e){
58854         var k = e.getKey(), newCell, g = this.grid, ed = g.activeEditor;
58855         if(k == e.TAB){
58856             e.stopEvent();
58857             ed.completeEdit();
58858             if(e.shiftKey){
58859                 newCell = g.walkCells(ed.row, ed.col-1, -1, this.acceptsNav, this);
58860             }else{
58861                 newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
58862             }
58863         }else if(k == e.ENTER && !e.ctrlKey){
58864             e.stopEvent();
58865             ed.completeEdit();
58866             if(e.shiftKey){
58867                 newCell = g.walkCells(ed.row-1, ed.col, -1, this.acceptsNav, this);
58868             }else{
58869                 newCell = g.walkCells(ed.row+1, ed.col, 1, this.acceptsNav, this);
58870             }
58871         }else if(k == e.ESC){
58872             ed.cancelEdit();
58873         }
58874         if(newCell){
58875             g.startEditing(newCell[0], newCell[1]);
58876         }
58877     }
58878 });/*
58879  * Based on:
58880  * Ext JS Library 1.1.1
58881  * Copyright(c) 2006-2007, Ext JS, LLC.
58882  *
58883  * Originally Released Under LGPL - original licence link has changed is not relivant.
58884  *
58885  * Fork - LGPL
58886  * <script type="text/javascript">
58887  */
58888 /**
58889  * @class Roo.grid.CellSelectionModel
58890  * @extends Roo.grid.AbstractSelectionModel
58891  * This class provides the basic implementation for cell selection in a grid.
58892  * @constructor
58893  * @param {Object} config The object containing the configuration of this model.
58894  * @cfg {Boolean} enter_is_tab Enter behaves the same as tab. (eg. goes to next cell) default: false
58895  */
58896 Roo.grid.CellSelectionModel = function(config){
58897     Roo.apply(this, config);
58898
58899     this.selection = null;
58900
58901     this.addEvents({
58902         /**
58903              * @event beforerowselect
58904              * Fires before a cell is selected.
58905              * @param {SelectionModel} this
58906              * @param {Number} rowIndex The selected row index
58907              * @param {Number} colIndex The selected cell index
58908              */
58909             "beforecellselect" : true,
58910         /**
58911              * @event cellselect
58912              * Fires when a cell is selected.
58913              * @param {SelectionModel} this
58914              * @param {Number} rowIndex The selected row index
58915              * @param {Number} colIndex The selected cell index
58916              */
58917             "cellselect" : true,
58918         /**
58919              * @event selectionchange
58920              * Fires when the active selection changes.
58921              * @param {SelectionModel} this
58922              * @param {Object} selection null for no selection or an object (o) with two properties
58923                 <ul>
58924                 <li>o.record: the record object for the row the selection is in</li>
58925                 <li>o.cell: An array of [rowIndex, columnIndex]</li>
58926                 </ul>
58927              */
58928             "selectionchange" : true,
58929         /**
58930              * @event tabend
58931              * Fires when the tab (or enter) was pressed on the last editable cell
58932              * You can use this to trigger add new row.
58933              * @param {SelectionModel} this
58934              */
58935             "tabend" : true,
58936          /**
58937              * @event beforeeditnext
58938              * Fires before the next editable sell is made active
58939              * You can use this to skip to another cell or fire the tabend
58940              *    if you set cell to false
58941              * @param {Object} eventdata object : { cell : [ row, col ] } 
58942              */
58943             "beforeeditnext" : true
58944     });
58945     Roo.grid.CellSelectionModel.superclass.constructor.call(this);
58946 };
58947
58948 Roo.extend(Roo.grid.CellSelectionModel, Roo.grid.AbstractSelectionModel,  {
58949     
58950     enter_is_tab: false,
58951
58952     /** @ignore */
58953     initEvents : function(){
58954         this.grid.on("mousedown", this.handleMouseDown, this);
58955         this.grid.getGridEl().on(Roo.isIE ? "keydown" : "keypress", this.handleKeyDown, this);
58956         var view = this.grid.view;
58957         view.on("refresh", this.onViewChange, this);
58958         view.on("rowupdated", this.onRowUpdated, this);
58959         view.on("beforerowremoved", this.clearSelections, this);
58960         view.on("beforerowsinserted", this.clearSelections, this);
58961         if(this.grid.isEditor){
58962             this.grid.on("beforeedit", this.beforeEdit,  this);
58963         }
58964     },
58965
58966         //private
58967     beforeEdit : function(e){
58968         this.select(e.row, e.column, false, true, e.record);
58969     },
58970
58971         //private
58972     onRowUpdated : function(v, index, r){
58973         if(this.selection && this.selection.record == r){
58974             v.onCellSelect(index, this.selection.cell[1]);
58975         }
58976     },
58977
58978         //private
58979     onViewChange : function(){
58980         this.clearSelections(true);
58981     },
58982
58983         /**
58984          * Returns the currently selected cell,.
58985          * @return {Array} The selected cell (row, column) or null if none selected.
58986          */
58987     getSelectedCell : function(){
58988         return this.selection ? this.selection.cell : null;
58989     },
58990
58991     /**
58992      * Clears all selections.
58993      * @param {Boolean} true to prevent the gridview from being notified about the change.
58994      */
58995     clearSelections : function(preventNotify){
58996         var s = this.selection;
58997         if(s){
58998             if(preventNotify !== true){
58999                 this.grid.view.onCellDeselect(s.cell[0], s.cell[1]);
59000             }
59001             this.selection = null;
59002             this.fireEvent("selectionchange", this, null);
59003         }
59004     },
59005
59006     /**
59007      * Returns true if there is a selection.
59008      * @return {Boolean}
59009      */
59010     hasSelection : function(){
59011         return this.selection ? true : false;
59012     },
59013
59014     /** @ignore */
59015     handleMouseDown : function(e, t){
59016         var v = this.grid.getView();
59017         if(this.isLocked()){
59018             return;
59019         };
59020         var row = v.findRowIndex(t);
59021         var cell = v.findCellIndex(t);
59022         if(row !== false && cell !== false){
59023             this.select(row, cell);
59024         }
59025     },
59026
59027     /**
59028      * Selects a cell.
59029      * @param {Number} rowIndex
59030      * @param {Number} collIndex
59031      */
59032     select : function(rowIndex, colIndex, preventViewNotify, preventFocus, /*internal*/ r){
59033         if(this.fireEvent("beforecellselect", this, rowIndex, colIndex) !== false){
59034             this.clearSelections();
59035             r = r || this.grid.dataSource.getAt(rowIndex);
59036             this.selection = {
59037                 record : r,
59038                 cell : [rowIndex, colIndex]
59039             };
59040             if(!preventViewNotify){
59041                 var v = this.grid.getView();
59042                 v.onCellSelect(rowIndex, colIndex);
59043                 if(preventFocus !== true){
59044                     v.focusCell(rowIndex, colIndex);
59045                 }
59046             }
59047             this.fireEvent("cellselect", this, rowIndex, colIndex);
59048             this.fireEvent("selectionchange", this, this.selection);
59049         }
59050     },
59051
59052         //private
59053     isSelectable : function(rowIndex, colIndex, cm){
59054         return !cm.isHidden(colIndex);
59055     },
59056
59057     /** @ignore */
59058     handleKeyDown : function(e){
59059         //Roo.log('Cell Sel Model handleKeyDown');
59060         if(!e.isNavKeyPress()){
59061             return;
59062         }
59063         var g = this.grid, s = this.selection;
59064         if(!s){
59065             e.stopEvent();
59066             var cell = g.walkCells(0, 0, 1, this.isSelectable,  this);
59067             if(cell){
59068                 this.select(cell[0], cell[1]);
59069             }
59070             return;
59071         }
59072         var sm = this;
59073         var walk = function(row, col, step){
59074             return g.walkCells(row, col, step, sm.isSelectable,  sm);
59075         };
59076         var k = e.getKey(), r = s.cell[0], c = s.cell[1];
59077         var newCell;
59078
59079       
59080
59081         switch(k){
59082             case e.TAB:
59083                 // handled by onEditorKey
59084                 if (g.isEditor && g.editing) {
59085                     return;
59086                 }
59087                 if(e.shiftKey) {
59088                     newCell = walk(r, c-1, -1);
59089                 } else {
59090                     newCell = walk(r, c+1, 1);
59091                 }
59092                 break;
59093             
59094             case e.DOWN:
59095                newCell = walk(r+1, c, 1);
59096                 break;
59097             
59098             case e.UP:
59099                 newCell = walk(r-1, c, -1);
59100                 break;
59101             
59102             case e.RIGHT:
59103                 newCell = walk(r, c+1, 1);
59104                 break;
59105             
59106             case e.LEFT:
59107                 newCell = walk(r, c-1, -1);
59108                 break;
59109             
59110             case e.ENTER:
59111                 
59112                 if(g.isEditor && !g.editing){
59113                    g.startEditing(r, c);
59114                    e.stopEvent();
59115                    return;
59116                 }
59117                 
59118                 
59119              break;
59120         };
59121         if(newCell){
59122             this.select(newCell[0], newCell[1]);
59123             e.stopEvent();
59124             
59125         }
59126     },
59127
59128     acceptsNav : function(row, col, cm){
59129         return !cm.isHidden(col) && cm.isCellEditable(col, row);
59130     },
59131     /**
59132      * Selects a cell.
59133      * @param {Number} field (not used) - as it's normally used as a listener
59134      * @param {Number} e - event - fake it by using
59135      *
59136      * var e = Roo.EventObjectImpl.prototype;
59137      * e.keyCode = e.TAB
59138      *
59139      * 
59140      */
59141     onEditorKey : function(field, e){
59142         
59143         var k = e.getKey(),
59144             newCell,
59145             g = this.grid,
59146             ed = g.activeEditor,
59147             forward = false;
59148         ///Roo.log('onEditorKey' + k);
59149         
59150         
59151         if (this.enter_is_tab && k == e.ENTER) {
59152             k = e.TAB;
59153         }
59154         
59155         if(k == e.TAB){
59156             if(e.shiftKey){
59157                 newCell = g.walkCells(ed.row, ed.col-1, -1, this.acceptsNav, this);
59158             }else{
59159                 newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
59160                 forward = true;
59161             }
59162             
59163             e.stopEvent();
59164             
59165         } else if(k == e.ENTER &&  !e.ctrlKey){
59166             ed.completeEdit();
59167             e.stopEvent();
59168             newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
59169         
59170                 } else if(k == e.ESC){
59171             ed.cancelEdit();
59172         }
59173                 
59174         if (newCell) {
59175             var ecall = { cell : newCell, forward : forward };
59176             this.fireEvent('beforeeditnext', ecall );
59177             newCell = ecall.cell;
59178                         forward = ecall.forward;
59179         }
59180                 
59181         if(newCell){
59182             //Roo.log('next cell after edit');
59183             g.startEditing.defer(100, g, [newCell[0], newCell[1]]);
59184         } else if (forward) {
59185             // tabbed past last
59186             this.fireEvent.defer(100, this, ['tabend',this]);
59187         }
59188     }
59189 });/*
59190  * Based on:
59191  * Ext JS Library 1.1.1
59192  * Copyright(c) 2006-2007, Ext JS, LLC.
59193  *
59194  * Originally Released Under LGPL - original licence link has changed is not relivant.
59195  *
59196  * Fork - LGPL
59197  * <script type="text/javascript">
59198  */
59199  
59200 /**
59201  * @class Roo.grid.EditorGrid
59202  * @extends Roo.grid.Grid
59203  * Class for creating and editable grid.
59204  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered - 
59205  * The container MUST have some type of size defined for the grid to fill. The container will be 
59206  * automatically set to position relative if it isn't already.
59207  * @param {Object} dataSource The data model to bind to
59208  * @param {Object} colModel The column model with info about this grid's columns
59209  */
59210 Roo.grid.EditorGrid = function(container, config){
59211     Roo.grid.EditorGrid.superclass.constructor.call(this, container, config);
59212     this.getGridEl().addClass("xedit-grid");
59213
59214     if(!this.selModel){
59215         this.selModel = new Roo.grid.CellSelectionModel();
59216     }
59217
59218     this.activeEditor = null;
59219
59220         this.addEvents({
59221             /**
59222              * @event beforeedit
59223              * Fires before cell editing is triggered. The edit event object has the following properties <br />
59224              * <ul style="padding:5px;padding-left:16px;">
59225              * <li>grid - This grid</li>
59226              * <li>record - The record being edited</li>
59227              * <li>field - The field name being edited</li>
59228              * <li>value - The value for the field being edited.</li>
59229              * <li>row - The grid row index</li>
59230              * <li>column - The grid column index</li>
59231              * <li>cancel - Set this to true to cancel the edit or return false from your handler.</li>
59232              * </ul>
59233              * @param {Object} e An edit event (see above for description)
59234              */
59235             "beforeedit" : true,
59236             /**
59237              * @event afteredit
59238              * Fires after a cell is edited. <br />
59239              * <ul style="padding:5px;padding-left:16px;">
59240              * <li>grid - This grid</li>
59241              * <li>record - The record being edited</li>
59242              * <li>field - The field name being edited</li>
59243              * <li>value - The value being set</li>
59244              * <li>originalValue - The original value for the field, before the edit.</li>
59245              * <li>row - The grid row index</li>
59246              * <li>column - The grid column index</li>
59247              * </ul>
59248              * @param {Object} e An edit event (see above for description)
59249              */
59250             "afteredit" : true,
59251             /**
59252              * @event validateedit
59253              * Fires after a cell is edited, but before the value is set in the record. 
59254          * You can use this to modify the value being set in the field, Return false
59255              * to cancel the change. The edit event object has the following properties <br />
59256              * <ul style="padding:5px;padding-left:16px;">
59257          * <li>editor - This editor</li>
59258              * <li>grid - This grid</li>
59259              * <li>record - The record being edited</li>
59260              * <li>field - The field name being edited</li>
59261              * <li>value - The value being set</li>
59262              * <li>originalValue - The original value for the field, before the edit.</li>
59263              * <li>row - The grid row index</li>
59264              * <li>column - The grid column index</li>
59265              * <li>cancel - Set this to true to cancel the edit or return false from your handler.</li>
59266              * </ul>
59267              * @param {Object} e An edit event (see above for description)
59268              */
59269             "validateedit" : true
59270         });
59271     this.on("bodyscroll", this.stopEditing,  this);
59272     this.on(this.clicksToEdit == 1 ? "cellclick" : "celldblclick", this.onCellDblClick,  this);
59273 };
59274
59275 Roo.extend(Roo.grid.EditorGrid, Roo.grid.Grid, {
59276     /**
59277      * @cfg {Number} clicksToEdit
59278      * The number of clicks on a cell required to display the cell's editor (defaults to 2)
59279      */
59280     clicksToEdit: 2,
59281
59282     // private
59283     isEditor : true,
59284     // private
59285     trackMouseOver: false, // causes very odd FF errors
59286
59287     onCellDblClick : function(g, row, col){
59288         this.startEditing(row, col);
59289     },
59290
59291     onEditComplete : function(ed, value, startValue){
59292         this.editing = false;
59293         this.activeEditor = null;
59294         ed.un("specialkey", this.selModel.onEditorKey, this.selModel);
59295         var r = ed.record;
59296         var field = this.colModel.getDataIndex(ed.col);
59297         var e = {
59298             grid: this,
59299             record: r,
59300             field: field,
59301             originalValue: startValue,
59302             value: value,
59303             row: ed.row,
59304             column: ed.col,
59305             cancel:false,
59306             editor: ed
59307         };
59308         var cell = Roo.get(this.view.getCell(ed.row,ed.col));
59309         cell.show();
59310           
59311         if(String(value) !== String(startValue)){
59312             
59313             if(this.fireEvent("validateedit", e) !== false && !e.cancel){
59314                 r.set(field, e.value);
59315                 // if we are dealing with a combo box..
59316                 // then we also set the 'name' colum to be the displayField
59317                 if (ed.field.displayField && ed.field.name) {
59318                     r.set(ed.field.name, ed.field.el.dom.value);
59319                 }
59320                 
59321                 delete e.cancel; //?? why!!!
59322                 this.fireEvent("afteredit", e);
59323             }
59324         } else {
59325             this.fireEvent("afteredit", e); // always fire it!
59326         }
59327         this.view.focusCell(ed.row, ed.col);
59328     },
59329
59330     /**
59331      * Starts editing the specified for the specified row/column
59332      * @param {Number} rowIndex
59333      * @param {Number} colIndex
59334      */
59335     startEditing : function(row, col){
59336         this.stopEditing();
59337         if(this.colModel.isCellEditable(col, row)){
59338             this.view.ensureVisible(row, col, true);
59339           
59340             var r = this.dataSource.getAt(row);
59341             var field = this.colModel.getDataIndex(col);
59342             var cell = Roo.get(this.view.getCell(row,col));
59343             var e = {
59344                 grid: this,
59345                 record: r,
59346                 field: field,
59347                 value: r.data[field],
59348                 row: row,
59349                 column: col,
59350                 cancel:false 
59351             };
59352             if(this.fireEvent("beforeedit", e) !== false && !e.cancel){
59353                 this.editing = true;
59354                 var ed = this.colModel.getCellEditor(col, row);
59355                 
59356                 if (!ed) {
59357                     return;
59358                 }
59359                 if(!ed.rendered){
59360                     ed.render(ed.parentEl || document.body);
59361                 }
59362                 ed.field.reset();
59363                
59364                 cell.hide();
59365                 
59366                 (function(){ // complex but required for focus issues in safari, ie and opera
59367                     ed.row = row;
59368                     ed.col = col;
59369                     ed.record = r;
59370                     ed.on("complete",   this.onEditComplete,        this,       {single: true});
59371                     ed.on("specialkey", this.selModel.onEditorKey,  this.selModel);
59372                     this.activeEditor = ed;
59373                     var v = r.data[field];
59374                     ed.startEdit(this.view.getCell(row, col), v);
59375                     // combo's with 'displayField and name set
59376                     if (ed.field.displayField && ed.field.name) {
59377                         ed.field.el.dom.value = r.data[ed.field.name];
59378                     }
59379                     
59380                     
59381                 }).defer(50, this);
59382             }
59383         }
59384     },
59385         
59386     /**
59387      * Stops any active editing
59388      */
59389     stopEditing : function(){
59390         if(this.activeEditor){
59391             this.activeEditor.completeEdit();
59392         }
59393         this.activeEditor = null;
59394     },
59395         
59396          /**
59397      * Called to get grid's drag proxy text, by default returns this.ddText.
59398      * @return {String}
59399      */
59400     getDragDropText : function(){
59401         var count = this.selModel.getSelectedCell() ? 1 : 0;
59402         return String.format(this.ddText, count, count == 1 ? '' : 's');
59403     }
59404         
59405 });/*
59406  * Based on:
59407  * Ext JS Library 1.1.1
59408  * Copyright(c) 2006-2007, Ext JS, LLC.
59409  *
59410  * Originally Released Under LGPL - original licence link has changed is not relivant.
59411  *
59412  * Fork - LGPL
59413  * <script type="text/javascript">
59414  */
59415
59416 // private - not really -- you end up using it !
59417 // This is a support class used internally by the Grid components
59418
59419 /**
59420  * @class Roo.grid.GridEditor
59421  * @extends Roo.Editor
59422  * Class for creating and editable grid elements.
59423  * @param {Object} config any settings (must include field)
59424  */
59425 Roo.grid.GridEditor = function(field, config){
59426     if (!config && field.field) {
59427         config = field;
59428         field = Roo.factory(config.field, Roo.form);
59429     }
59430     Roo.grid.GridEditor.superclass.constructor.call(this, field, config);
59431     field.monitorTab = false;
59432 };
59433
59434 Roo.extend(Roo.grid.GridEditor, Roo.Editor, {
59435     
59436     /**
59437      * @cfg {Roo.form.Field} field Field to wrap (or xtyped)
59438      */
59439     
59440     alignment: "tl-tl",
59441     autoSize: "width",
59442     hideEl : false,
59443     cls: "x-small-editor x-grid-editor",
59444     shim:false,
59445     shadow:"frame"
59446 });/*
59447  * Based on:
59448  * Ext JS Library 1.1.1
59449  * Copyright(c) 2006-2007, Ext JS, LLC.
59450  *
59451  * Originally Released Under LGPL - original licence link has changed is not relivant.
59452  *
59453  * Fork - LGPL
59454  * <script type="text/javascript">
59455  */
59456   
59457
59458   
59459 Roo.grid.PropertyRecord = Roo.data.Record.create([
59460     {name:'name',type:'string'},  'value'
59461 ]);
59462
59463
59464 Roo.grid.PropertyStore = function(grid, source){
59465     this.grid = grid;
59466     this.store = new Roo.data.Store({
59467         recordType : Roo.grid.PropertyRecord
59468     });
59469     this.store.on('update', this.onUpdate,  this);
59470     if(source){
59471         this.setSource(source);
59472     }
59473     Roo.grid.PropertyStore.superclass.constructor.call(this);
59474 };
59475
59476
59477
59478 Roo.extend(Roo.grid.PropertyStore, Roo.util.Observable, {
59479     setSource : function(o){
59480         this.source = o;
59481         this.store.removeAll();
59482         var data = [];
59483         for(var k in o){
59484             if(this.isEditableValue(o[k])){
59485                 data.push(new Roo.grid.PropertyRecord({name: k, value: o[k]}, k));
59486             }
59487         }
59488         this.store.loadRecords({records: data}, {}, true);
59489     },
59490
59491     onUpdate : function(ds, record, type){
59492         if(type == Roo.data.Record.EDIT){
59493             var v = record.data['value'];
59494             var oldValue = record.modified['value'];
59495             if(this.grid.fireEvent('beforepropertychange', this.source, record.id, v, oldValue) !== false){
59496                 this.source[record.id] = v;
59497                 record.commit();
59498                 this.grid.fireEvent('propertychange', this.source, record.id, v, oldValue);
59499             }else{
59500                 record.reject();
59501             }
59502         }
59503     },
59504
59505     getProperty : function(row){
59506        return this.store.getAt(row);
59507     },
59508
59509     isEditableValue: function(val){
59510         if(val && val instanceof Date){
59511             return true;
59512         }else if(typeof val == 'object' || typeof val == 'function'){
59513             return false;
59514         }
59515         return true;
59516     },
59517
59518     setValue : function(prop, value){
59519         this.source[prop] = value;
59520         this.store.getById(prop).set('value', value);
59521     },
59522
59523     getSource : function(){
59524         return this.source;
59525     }
59526 });
59527
59528 Roo.grid.PropertyColumnModel = function(grid, store){
59529     this.grid = grid;
59530     var g = Roo.grid;
59531     g.PropertyColumnModel.superclass.constructor.call(this, [
59532         {header: this.nameText, sortable: true, dataIndex:'name', id: 'name'},
59533         {header: this.valueText, resizable:false, dataIndex: 'value', id: 'value'}
59534     ]);
59535     this.store = store;
59536     this.bselect = Roo.DomHelper.append(document.body, {
59537         tag: 'select', style:'display:none', cls: 'x-grid-editor', children: [
59538             {tag: 'option', value: 'true', html: 'true'},
59539             {tag: 'option', value: 'false', html: 'false'}
59540         ]
59541     });
59542     Roo.id(this.bselect);
59543     var f = Roo.form;
59544     this.editors = {
59545         'date' : new g.GridEditor(new f.DateField({selectOnFocus:true})),
59546         'string' : new g.GridEditor(new f.TextField({selectOnFocus:true})),
59547         'number' : new g.GridEditor(new f.NumberField({selectOnFocus:true, style:'text-align:left;'})),
59548         'int' : new g.GridEditor(new f.NumberField({selectOnFocus:true, allowDecimals:false, style:'text-align:left;'})),
59549         'boolean' : new g.GridEditor(new f.Field({el:this.bselect,selectOnFocus:true}))
59550     };
59551     this.renderCellDelegate = this.renderCell.createDelegate(this);
59552     this.renderPropDelegate = this.renderProp.createDelegate(this);
59553 };
59554
59555 Roo.extend(Roo.grid.PropertyColumnModel, Roo.grid.ColumnModel, {
59556     
59557     
59558     nameText : 'Name',
59559     valueText : 'Value',
59560     
59561     dateFormat : 'm/j/Y',
59562     
59563     
59564     renderDate : function(dateVal){
59565         return dateVal.dateFormat(this.dateFormat);
59566     },
59567
59568     renderBool : function(bVal){
59569         return bVal ? 'true' : 'false';
59570     },
59571
59572     isCellEditable : function(colIndex, rowIndex){
59573         return colIndex == 1;
59574     },
59575
59576     getRenderer : function(col){
59577         return col == 1 ?
59578             this.renderCellDelegate : this.renderPropDelegate;
59579     },
59580
59581     renderProp : function(v){
59582         return this.getPropertyName(v);
59583     },
59584
59585     renderCell : function(val){
59586         var rv = val;
59587         if(val instanceof Date){
59588             rv = this.renderDate(val);
59589         }else if(typeof val == 'boolean'){
59590             rv = this.renderBool(val);
59591         }
59592         return Roo.util.Format.htmlEncode(rv);
59593     },
59594
59595     getPropertyName : function(name){
59596         var pn = this.grid.propertyNames;
59597         return pn && pn[name] ? pn[name] : name;
59598     },
59599
59600     getCellEditor : function(colIndex, rowIndex){
59601         var p = this.store.getProperty(rowIndex);
59602         var n = p.data['name'], val = p.data['value'];
59603         
59604         if(typeof(this.grid.customEditors[n]) == 'string'){
59605             return this.editors[this.grid.customEditors[n]];
59606         }
59607         if(typeof(this.grid.customEditors[n]) != 'undefined'){
59608             return this.grid.customEditors[n];
59609         }
59610         if(val instanceof Date){
59611             return this.editors['date'];
59612         }else if(typeof val == 'number'){
59613             return this.editors['number'];
59614         }else if(typeof val == 'boolean'){
59615             return this.editors['boolean'];
59616         }else{
59617             return this.editors['string'];
59618         }
59619     }
59620 });
59621
59622 /**
59623  * @class Roo.grid.PropertyGrid
59624  * @extends Roo.grid.EditorGrid
59625  * This class represents the  interface of a component based property grid control.
59626  * <br><br>Usage:<pre><code>
59627  var grid = new Roo.grid.PropertyGrid("my-container-id", {
59628       
59629  });
59630  // set any options
59631  grid.render();
59632  * </code></pre>
59633   
59634  * @constructor
59635  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
59636  * The container MUST have some type of size defined for the grid to fill. The container will be
59637  * automatically set to position relative if it isn't already.
59638  * @param {Object} config A config object that sets properties on this grid.
59639  */
59640 Roo.grid.PropertyGrid = function(container, config){
59641     config = config || {};
59642     var store = new Roo.grid.PropertyStore(this);
59643     this.store = store;
59644     var cm = new Roo.grid.PropertyColumnModel(this, store);
59645     store.store.sort('name', 'ASC');
59646     Roo.grid.PropertyGrid.superclass.constructor.call(this, container, Roo.apply({
59647         ds: store.store,
59648         cm: cm,
59649         enableColLock:false,
59650         enableColumnMove:false,
59651         stripeRows:false,
59652         trackMouseOver: false,
59653         clicksToEdit:1
59654     }, config));
59655     this.getGridEl().addClass('x-props-grid');
59656     this.lastEditRow = null;
59657     this.on('columnresize', this.onColumnResize, this);
59658     this.addEvents({
59659          /**
59660              * @event beforepropertychange
59661              * Fires before a property changes (return false to stop?)
59662              * @param {Roo.grid.PropertyGrid} grid property grid? (check could be store)
59663              * @param {String} id Record Id
59664              * @param {String} newval New Value
59665          * @param {String} oldval Old Value
59666              */
59667         "beforepropertychange": true,
59668         /**
59669              * @event propertychange
59670              * Fires after a property changes
59671              * @param {Roo.grid.PropertyGrid} grid property grid? (check could be store)
59672              * @param {String} id Record Id
59673              * @param {String} newval New Value
59674          * @param {String} oldval Old Value
59675              */
59676         "propertychange": true
59677     });
59678     this.customEditors = this.customEditors || {};
59679 };
59680 Roo.extend(Roo.grid.PropertyGrid, Roo.grid.EditorGrid, {
59681     
59682      /**
59683      * @cfg {Object} customEditors map of colnames=> custom editors.
59684      * the custom editor can be one of the standard ones (date|string|number|int|boolean), or a
59685      * grid editor eg. Roo.grid.GridEditor(new Roo.form.TextArea({selectOnFocus:true})),
59686      * false disables editing of the field.
59687          */
59688     
59689       /**
59690      * @cfg {Object} propertyNames map of property Names to their displayed value
59691          */
59692     
59693     render : function(){
59694         Roo.grid.PropertyGrid.superclass.render.call(this);
59695         this.autoSize.defer(100, this);
59696     },
59697
59698     autoSize : function(){
59699         Roo.grid.PropertyGrid.superclass.autoSize.call(this);
59700         if(this.view){
59701             this.view.fitColumns();
59702         }
59703     },
59704
59705     onColumnResize : function(){
59706         this.colModel.setColumnWidth(1, this.container.getWidth(true)-this.colModel.getColumnWidth(0));
59707         this.autoSize();
59708     },
59709     /**
59710      * Sets the data for the Grid
59711      * accepts a Key => Value object of all the elements avaiable.
59712      * @param {Object} data  to appear in grid.
59713      */
59714     setSource : function(source){
59715         this.store.setSource(source);
59716         //this.autoSize();
59717     },
59718     /**
59719      * Gets all the data from the grid.
59720      * @return {Object} data  data stored in grid
59721      */
59722     getSource : function(){
59723         return this.store.getSource();
59724     }
59725 });/*
59726   
59727  * Licence LGPL
59728  
59729  */
59730  
59731 /**
59732  * @class Roo.grid.Calendar
59733  * @extends Roo.util.Grid
59734  * This class extends the Grid to provide a calendar widget
59735  * <br><br>Usage:<pre><code>
59736  var grid = new Roo.grid.Calendar("my-container-id", {
59737      ds: myDataStore,
59738      cm: myColModel,
59739      selModel: mySelectionModel,
59740      autoSizeColumns: true,
59741      monitorWindowResize: false,
59742      trackMouseOver: true
59743      eventstore : real data store..
59744  });
59745  // set any options
59746  grid.render();
59747   
59748   * @constructor
59749  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
59750  * The container MUST have some type of size defined for the grid to fill. The container will be
59751  * automatically set to position relative if it isn't already.
59752  * @param {Object} config A config object that sets properties on this grid.
59753  */
59754 Roo.grid.Calendar = function(container, config){
59755         // initialize the container
59756         this.container = Roo.get(container);
59757         this.container.update("");
59758         this.container.setStyle("overflow", "hidden");
59759     this.container.addClass('x-grid-container');
59760
59761     this.id = this.container.id;
59762
59763     Roo.apply(this, config);
59764     // check and correct shorthanded configs
59765     
59766     var rows = [];
59767     var d =1;
59768     for (var r = 0;r < 6;r++) {
59769         
59770         rows[r]=[];
59771         for (var c =0;c < 7;c++) {
59772             rows[r][c]= '';
59773         }
59774     }
59775     if (this.eventStore) {
59776         this.eventStore= Roo.factory(this.eventStore, Roo.data);
59777         this.eventStore.on('load',this.onLoad, this);
59778         this.eventStore.on('beforeload',this.clearEvents, this);
59779          
59780     }
59781     
59782     this.dataSource = new Roo.data.Store({
59783             proxy: new Roo.data.MemoryProxy(rows),
59784             reader: new Roo.data.ArrayReader({}, [
59785                    'weekday0', 'weekday1', 'weekday2', 'weekday3', 'weekday4', 'weekday5', 'weekday6' ])
59786     });
59787
59788     this.dataSource.load();
59789     this.ds = this.dataSource;
59790     this.ds.xmodule = this.xmodule || false;
59791     
59792     
59793     var cellRender = function(v,x,r)
59794     {
59795         return String.format(
59796             '<div class="fc-day  fc-widget-content"><div>' +
59797                 '<div class="fc-event-container"></div>' +
59798                 '<div class="fc-day-number">{0}</div>'+
59799                 
59800                 '<div class="fc-day-content"><div style="position:relative"></div></div>' +
59801             '</div></div>', v);
59802     
59803     }
59804     
59805     
59806     this.colModel = new Roo.grid.ColumnModel( [
59807         {
59808             xtype: 'ColumnModel',
59809             xns: Roo.grid,
59810             dataIndex : 'weekday0',
59811             header : 'Sunday',
59812             renderer : cellRender
59813         },
59814         {
59815             xtype: 'ColumnModel',
59816             xns: Roo.grid,
59817             dataIndex : 'weekday1',
59818             header : 'Monday',
59819             renderer : cellRender
59820         },
59821         {
59822             xtype: 'ColumnModel',
59823             xns: Roo.grid,
59824             dataIndex : 'weekday2',
59825             header : 'Tuesday',
59826             renderer : cellRender
59827         },
59828         {
59829             xtype: 'ColumnModel',
59830             xns: Roo.grid,
59831             dataIndex : 'weekday3',
59832             header : 'Wednesday',
59833             renderer : cellRender
59834         },
59835         {
59836             xtype: 'ColumnModel',
59837             xns: Roo.grid,
59838             dataIndex : 'weekday4',
59839             header : 'Thursday',
59840             renderer : cellRender
59841         },
59842         {
59843             xtype: 'ColumnModel',
59844             xns: Roo.grid,
59845             dataIndex : 'weekday5',
59846             header : 'Friday',
59847             renderer : cellRender
59848         },
59849         {
59850             xtype: 'ColumnModel',
59851             xns: Roo.grid,
59852             dataIndex : 'weekday6',
59853             header : 'Saturday',
59854             renderer : cellRender
59855         }
59856     ]);
59857     this.cm = this.colModel;
59858     this.cm.xmodule = this.xmodule || false;
59859  
59860         
59861           
59862     //this.selModel = new Roo.grid.CellSelectionModel();
59863     //this.sm = this.selModel;
59864     //this.selModel.init(this);
59865     
59866     
59867     if(this.width){
59868         this.container.setWidth(this.width);
59869     }
59870
59871     if(this.height){
59872         this.container.setHeight(this.height);
59873     }
59874     /** @private */
59875         this.addEvents({
59876         // raw events
59877         /**
59878          * @event click
59879          * The raw click event for the entire grid.
59880          * @param {Roo.EventObject} e
59881          */
59882         "click" : true,
59883         /**
59884          * @event dblclick
59885          * The raw dblclick event for the entire grid.
59886          * @param {Roo.EventObject} e
59887          */
59888         "dblclick" : true,
59889         /**
59890          * @event contextmenu
59891          * The raw contextmenu event for the entire grid.
59892          * @param {Roo.EventObject} e
59893          */
59894         "contextmenu" : true,
59895         /**
59896          * @event mousedown
59897          * The raw mousedown event for the entire grid.
59898          * @param {Roo.EventObject} e
59899          */
59900         "mousedown" : true,
59901         /**
59902          * @event mouseup
59903          * The raw mouseup event for the entire grid.
59904          * @param {Roo.EventObject} e
59905          */
59906         "mouseup" : true,
59907         /**
59908          * @event mouseover
59909          * The raw mouseover event for the entire grid.
59910          * @param {Roo.EventObject} e
59911          */
59912         "mouseover" : true,
59913         /**
59914          * @event mouseout
59915          * The raw mouseout event for the entire grid.
59916          * @param {Roo.EventObject} e
59917          */
59918         "mouseout" : true,
59919         /**
59920          * @event keypress
59921          * The raw keypress event for the entire grid.
59922          * @param {Roo.EventObject} e
59923          */
59924         "keypress" : true,
59925         /**
59926          * @event keydown
59927          * The raw keydown event for the entire grid.
59928          * @param {Roo.EventObject} e
59929          */
59930         "keydown" : true,
59931
59932         // custom events
59933
59934         /**
59935          * @event cellclick
59936          * Fires when a cell is clicked
59937          * @param {Grid} this
59938          * @param {Number} rowIndex
59939          * @param {Number} columnIndex
59940          * @param {Roo.EventObject} e
59941          */
59942         "cellclick" : true,
59943         /**
59944          * @event celldblclick
59945          * Fires when a cell is double clicked
59946          * @param {Grid} this
59947          * @param {Number} rowIndex
59948          * @param {Number} columnIndex
59949          * @param {Roo.EventObject} e
59950          */
59951         "celldblclick" : true,
59952         /**
59953          * @event rowclick
59954          * Fires when a row is clicked
59955          * @param {Grid} this
59956          * @param {Number} rowIndex
59957          * @param {Roo.EventObject} e
59958          */
59959         "rowclick" : true,
59960         /**
59961          * @event rowdblclick
59962          * Fires when a row is double clicked
59963          * @param {Grid} this
59964          * @param {Number} rowIndex
59965          * @param {Roo.EventObject} e
59966          */
59967         "rowdblclick" : true,
59968         /**
59969          * @event headerclick
59970          * Fires when a header is clicked
59971          * @param {Grid} this
59972          * @param {Number} columnIndex
59973          * @param {Roo.EventObject} e
59974          */
59975         "headerclick" : true,
59976         /**
59977          * @event headerdblclick
59978          * Fires when a header cell is double clicked
59979          * @param {Grid} this
59980          * @param {Number} columnIndex
59981          * @param {Roo.EventObject} e
59982          */
59983         "headerdblclick" : true,
59984         /**
59985          * @event rowcontextmenu
59986          * Fires when a row is right clicked
59987          * @param {Grid} this
59988          * @param {Number} rowIndex
59989          * @param {Roo.EventObject} e
59990          */
59991         "rowcontextmenu" : true,
59992         /**
59993          * @event cellcontextmenu
59994          * Fires when a cell is right clicked
59995          * @param {Grid} this
59996          * @param {Number} rowIndex
59997          * @param {Number} cellIndex
59998          * @param {Roo.EventObject} e
59999          */
60000          "cellcontextmenu" : true,
60001         /**
60002          * @event headercontextmenu
60003          * Fires when a header is right clicked
60004          * @param {Grid} this
60005          * @param {Number} columnIndex
60006          * @param {Roo.EventObject} e
60007          */
60008         "headercontextmenu" : true,
60009         /**
60010          * @event bodyscroll
60011          * Fires when the body element is scrolled
60012          * @param {Number} scrollLeft
60013          * @param {Number} scrollTop
60014          */
60015         "bodyscroll" : true,
60016         /**
60017          * @event columnresize
60018          * Fires when the user resizes a column
60019          * @param {Number} columnIndex
60020          * @param {Number} newSize
60021          */
60022         "columnresize" : true,
60023         /**
60024          * @event columnmove
60025          * Fires when the user moves a column
60026          * @param {Number} oldIndex
60027          * @param {Number} newIndex
60028          */
60029         "columnmove" : true,
60030         /**
60031          * @event startdrag
60032          * Fires when row(s) start being dragged
60033          * @param {Grid} this
60034          * @param {Roo.GridDD} dd The drag drop object
60035          * @param {event} e The raw browser event
60036          */
60037         "startdrag" : true,
60038         /**
60039          * @event enddrag
60040          * Fires when a drag operation is complete
60041          * @param {Grid} this
60042          * @param {Roo.GridDD} dd The drag drop object
60043          * @param {event} e The raw browser event
60044          */
60045         "enddrag" : true,
60046         /**
60047          * @event dragdrop
60048          * Fires when dragged row(s) are dropped on a valid DD target
60049          * @param {Grid} this
60050          * @param {Roo.GridDD} dd The drag drop object
60051          * @param {String} targetId The target drag drop object
60052          * @param {event} e The raw browser event
60053          */
60054         "dragdrop" : true,
60055         /**
60056          * @event dragover
60057          * Fires while row(s) are being dragged. "targetId" is the id of the Yahoo.util.DD object the selected rows are being dragged over.
60058          * @param {Grid} this
60059          * @param {Roo.GridDD} dd The drag drop object
60060          * @param {String} targetId The target drag drop object
60061          * @param {event} e The raw browser event
60062          */
60063         "dragover" : true,
60064         /**
60065          * @event dragenter
60066          *  Fires when the dragged row(s) first cross another DD target while being dragged
60067          * @param {Grid} this
60068          * @param {Roo.GridDD} dd The drag drop object
60069          * @param {String} targetId The target drag drop object
60070          * @param {event} e The raw browser event
60071          */
60072         "dragenter" : true,
60073         /**
60074          * @event dragout
60075          * Fires when the dragged row(s) leave another DD target while being dragged
60076          * @param {Grid} this
60077          * @param {Roo.GridDD} dd The drag drop object
60078          * @param {String} targetId The target drag drop object
60079          * @param {event} e The raw browser event
60080          */
60081         "dragout" : true,
60082         /**
60083          * @event rowclass
60084          * Fires when a row is rendered, so you can change add a style to it.
60085          * @param {GridView} gridview   The grid view
60086          * @param {Object} rowcfg   contains record  rowIndex and rowClass - set rowClass to add a style.
60087          */
60088         'rowclass' : true,
60089
60090         /**
60091          * @event render
60092          * Fires when the grid is rendered
60093          * @param {Grid} grid
60094          */
60095         'render' : true,
60096             /**
60097              * @event select
60098              * Fires when a date is selected
60099              * @param {DatePicker} this
60100              * @param {Date} date The selected date
60101              */
60102         'select': true,
60103         /**
60104              * @event monthchange
60105              * Fires when the displayed month changes 
60106              * @param {DatePicker} this
60107              * @param {Date} date The selected month
60108              */
60109         'monthchange': true,
60110         /**
60111              * @event evententer
60112              * Fires when mouse over an event
60113              * @param {Calendar} this
60114              * @param {event} Event
60115              */
60116         'evententer': true,
60117         /**
60118              * @event eventleave
60119              * Fires when the mouse leaves an
60120              * @param {Calendar} this
60121              * @param {event}
60122              */
60123         'eventleave': true,
60124         /**
60125              * @event eventclick
60126              * Fires when the mouse click an
60127              * @param {Calendar} this
60128              * @param {event}
60129              */
60130         'eventclick': true,
60131         /**
60132              * @event eventrender
60133              * Fires before each cell is rendered, so you can modify the contents, like cls / title / qtip
60134              * @param {Calendar} this
60135              * @param {data} data to be modified
60136              */
60137         'eventrender': true
60138         
60139     });
60140
60141     Roo.grid.Grid.superclass.constructor.call(this);
60142     this.on('render', function() {
60143         this.view.el.addClass('x-grid-cal'); 
60144         
60145         (function() { this.setDate(new Date()); }).defer(100,this); //default today..
60146
60147     },this);
60148     
60149     if (!Roo.grid.Calendar.style) {
60150         Roo.grid.Calendar.style = Roo.util.CSS.createStyleSheet({
60151             
60152             
60153             '.x-grid-cal .x-grid-col' :  {
60154                 height: 'auto !important',
60155                 'vertical-align': 'top'
60156             },
60157             '.x-grid-cal  .fc-event-hori' : {
60158                 height: '14px'
60159             }
60160              
60161             
60162         }, Roo.id());
60163     }
60164
60165     
60166     
60167 };
60168 Roo.extend(Roo.grid.Calendar, Roo.grid.Grid, {
60169     /**
60170      * @cfg {Store} eventStore The store that loads events.
60171      */
60172     eventStore : 25,
60173
60174      
60175     activeDate : false,
60176     startDay : 0,
60177     autoWidth : true,
60178     monitorWindowResize : false,
60179
60180     
60181     resizeColumns : function() {
60182         var col = (this.view.el.getWidth() / 7) - 3;
60183         // loop through cols, and setWidth
60184         for(var i =0 ; i < 7 ; i++){
60185             this.cm.setColumnWidth(i, col);
60186         }
60187     },
60188      setDate :function(date) {
60189         
60190         Roo.log('setDate?');
60191         
60192         this.resizeColumns();
60193         var vd = this.activeDate;
60194         this.activeDate = date;
60195 //        if(vd && this.el){
60196 //            var t = date.getTime();
60197 //            if(vd.getMonth() == date.getMonth() && vd.getFullYear() == date.getFullYear()){
60198 //                Roo.log('using add remove');
60199 //                
60200 //                this.fireEvent('monthchange', this, date);
60201 //                
60202 //                this.cells.removeClass("fc-state-highlight");
60203 //                this.cells.each(function(c){
60204 //                   if(c.dateValue == t){
60205 //                       c.addClass("fc-state-highlight");
60206 //                       setTimeout(function(){
60207 //                            try{c.dom.firstChild.focus();}catch(e){}
60208 //                       }, 50);
60209 //                       return false;
60210 //                   }
60211 //                   return true;
60212 //                });
60213 //                return;
60214 //            }
60215 //        }
60216         
60217         var days = date.getDaysInMonth();
60218         
60219         var firstOfMonth = date.getFirstDateOfMonth();
60220         var startingPos = firstOfMonth.getDay()-this.startDay;
60221         
60222         if(startingPos < this.startDay){
60223             startingPos += 7;
60224         }
60225         
60226         var pm = date.add(Date.MONTH, -1);
60227         var prevStart = pm.getDaysInMonth()-startingPos;
60228 //        
60229         
60230         
60231         this.cells = this.view.el.select('.x-grid-row .x-grid-col',true);
60232         
60233         this.textNodes = this.view.el.query('.x-grid-row .x-grid-col .x-grid-cell-text');
60234         //this.cells.addClassOnOver('fc-state-hover');
60235         
60236         var cells = this.cells.elements;
60237         var textEls = this.textNodes;
60238         
60239         //Roo.each(cells, function(cell){
60240         //    cell.removeClass([ 'fc-past', 'fc-other-month', 'fc-future', 'fc-state-highlight', 'fc-state-disabled']);
60241         //});
60242         
60243         days += startingPos;
60244
60245         // convert everything to numbers so it's fast
60246         var day = 86400000;
60247         var d = (new Date(pm.getFullYear(), pm.getMonth(), prevStart)).clearTime();
60248         //Roo.log(d);
60249         //Roo.log(pm);
60250         //Roo.log(prevStart);
60251         
60252         var today = new Date().clearTime().getTime();
60253         var sel = date.clearTime().getTime();
60254         var min = this.minDate ? this.minDate.clearTime() : Number.NEGATIVE_INFINITY;
60255         var max = this.maxDate ? this.maxDate.clearTime() : Number.POSITIVE_INFINITY;
60256         var ddMatch = this.disabledDatesRE;
60257         var ddText = this.disabledDatesText;
60258         var ddays = this.disabledDays ? this.disabledDays.join("") : false;
60259         var ddaysText = this.disabledDaysText;
60260         var format = this.format;
60261         
60262         var setCellClass = function(cal, cell){
60263             
60264             //Roo.log('set Cell Class');
60265             cell.title = "";
60266             var t = d.getTime();
60267             
60268             //Roo.log(d);
60269             
60270             
60271             cell.dateValue = t;
60272             if(t == today){
60273                 cell.className += " fc-today";
60274                 cell.className += " fc-state-highlight";
60275                 cell.title = cal.todayText;
60276             }
60277             if(t == sel){
60278                 // disable highlight in other month..
60279                 cell.className += " fc-state-highlight";
60280                 
60281             }
60282             // disabling
60283             if(t < min) {
60284                 //cell.className = " fc-state-disabled";
60285                 cell.title = cal.minText;
60286                 return;
60287             }
60288             if(t > max) {
60289                 //cell.className = " fc-state-disabled";
60290                 cell.title = cal.maxText;
60291                 return;
60292             }
60293             if(ddays){
60294                 if(ddays.indexOf(d.getDay()) != -1){
60295                     // cell.title = ddaysText;
60296                    // cell.className = " fc-state-disabled";
60297                 }
60298             }
60299             if(ddMatch && format){
60300                 var fvalue = d.dateFormat(format);
60301                 if(ddMatch.test(fvalue)){
60302                     cell.title = ddText.replace("%0", fvalue);
60303                    cell.className = " fc-state-disabled";
60304                 }
60305             }
60306             
60307             if (!cell.initialClassName) {
60308                 cell.initialClassName = cell.dom.className;
60309             }
60310             
60311             cell.dom.className = cell.initialClassName  + ' ' +  cell.className;
60312         };
60313
60314         var i = 0;
60315         
60316         for(; i < startingPos; i++) {
60317             cells[i].dayName =  (++prevStart);
60318             Roo.log(textEls[i]);
60319             d.setDate(d.getDate()+1);
60320             
60321             //cells[i].className = "fc-past fc-other-month";
60322             setCellClass(this, cells[i]);
60323         }
60324         
60325         var intDay = 0;
60326         
60327         for(; i < days; i++){
60328             intDay = i - startingPos + 1;
60329             cells[i].dayName =  (intDay);
60330             d.setDate(d.getDate()+1);
60331             
60332             cells[i].className = ''; // "x-date-active";
60333             setCellClass(this, cells[i]);
60334         }
60335         var extraDays = 0;
60336         
60337         for(; i < 42; i++) {
60338             //textEls[i].innerHTML = (++extraDays);
60339             
60340             d.setDate(d.getDate()+1);
60341             cells[i].dayName = (++extraDays);
60342             cells[i].className = "fc-future fc-other-month";
60343             setCellClass(this, cells[i]);
60344         }
60345         
60346         //this.el.select('.fc-header-title h2',true).update(Date.monthNames[date.getMonth()] + " " + date.getFullYear());
60347         
60348         var totalRows = Math.ceil((date.getDaysInMonth() + date.getFirstDateOfMonth().getDay()) / 7);
60349         
60350         // this will cause all the cells to mis
60351         var rows= [];
60352         var i =0;
60353         for (var r = 0;r < 6;r++) {
60354             for (var c =0;c < 7;c++) {
60355                 this.ds.getAt(r).set('weekday' + c ,cells[i++].dayName );
60356             }    
60357         }
60358         
60359         this.cells = this.view.el.select('.x-grid-row .x-grid-col',true);
60360         for(i=0;i<cells.length;i++) {
60361             
60362             this.cells.elements[i].dayName = cells[i].dayName ;
60363             this.cells.elements[i].className = cells[i].className;
60364             this.cells.elements[i].initialClassName = cells[i].initialClassName ;
60365             this.cells.elements[i].title = cells[i].title ;
60366             this.cells.elements[i].dateValue = cells[i].dateValue ;
60367         }
60368         
60369         
60370         
60371         
60372         //this.el.select('tr.fc-week.fc-prev-last',true).removeClass('fc-last');
60373         //this.el.select('tr.fc-week.fc-next-last',true).addClass('fc-last').show();
60374         
60375         ////if(totalRows != 6){
60376             //this.el.select('tr.fc-week.fc-last',true).removeClass('fc-last').addClass('fc-next-last').hide();
60377            // this.el.select('tr.fc-week.fc-prev-last',true).addClass('fc-last');
60378        // }
60379         
60380         this.fireEvent('monthchange', this, date);
60381         
60382         
60383     },
60384  /**
60385      * Returns the grid's SelectionModel.
60386      * @return {SelectionModel}
60387      */
60388     getSelectionModel : function(){
60389         if(!this.selModel){
60390             this.selModel = new Roo.grid.CellSelectionModel();
60391         }
60392         return this.selModel;
60393     },
60394
60395     load: function() {
60396         this.eventStore.load()
60397         
60398         
60399         
60400     },
60401     
60402     findCell : function(dt) {
60403         dt = dt.clearTime().getTime();
60404         var ret = false;
60405         this.cells.each(function(c){
60406             //Roo.log("check " +c.dateValue + '?=' + dt);
60407             if(c.dateValue == dt){
60408                 ret = c;
60409                 return false;
60410             }
60411             return true;
60412         });
60413         
60414         return ret;
60415     },
60416     
60417     findCells : function(rec) {
60418         var s = rec.data.start_dt.clone().clearTime().getTime();
60419        // Roo.log(s);
60420         var e= rec.data.end_dt.clone().clearTime().getTime();
60421        // Roo.log(e);
60422         var ret = [];
60423         this.cells.each(function(c){
60424              ////Roo.log("check " +c.dateValue + '<' + e + ' > ' + s);
60425             
60426             if(c.dateValue > e){
60427                 return ;
60428             }
60429             if(c.dateValue < s){
60430                 return ;
60431             }
60432             ret.push(c);
60433         });
60434         
60435         return ret;    
60436     },
60437     
60438     findBestRow: function(cells)
60439     {
60440         var ret = 0;
60441         
60442         for (var i =0 ; i < cells.length;i++) {
60443             ret  = Math.max(cells[i].rows || 0,ret);
60444         }
60445         return ret;
60446         
60447     },
60448     
60449     
60450     addItem : function(rec)
60451     {
60452         // look for vertical location slot in
60453         var cells = this.findCells(rec);
60454         
60455         rec.row = this.findBestRow(cells);
60456         
60457         // work out the location.
60458         
60459         var crow = false;
60460         var rows = [];
60461         for(var i =0; i < cells.length; i++) {
60462             if (!crow) {
60463                 crow = {
60464                     start : cells[i],
60465                     end :  cells[i]
60466                 };
60467                 continue;
60468             }
60469             if (crow.start.getY() == cells[i].getY()) {
60470                 // on same row.
60471                 crow.end = cells[i];
60472                 continue;
60473             }
60474             // different row.
60475             rows.push(crow);
60476             crow = {
60477                 start: cells[i],
60478                 end : cells[i]
60479             };
60480             
60481         }
60482         
60483         rows.push(crow);
60484         rec.els = [];
60485         rec.rows = rows;
60486         rec.cells = cells;
60487         for (var i = 0; i < cells.length;i++) {
60488             cells[i].rows = Math.max(cells[i].rows || 0 , rec.row + 1 );
60489             
60490         }
60491         
60492         
60493     },
60494     
60495     clearEvents: function() {
60496         
60497         if (!this.eventStore.getCount()) {
60498             return;
60499         }
60500         // reset number of rows in cells.
60501         Roo.each(this.cells.elements, function(c){
60502             c.rows = 0;
60503         });
60504         
60505         this.eventStore.each(function(e) {
60506             this.clearEvent(e);
60507         },this);
60508         
60509     },
60510     
60511     clearEvent : function(ev)
60512     {
60513         if (ev.els) {
60514             Roo.each(ev.els, function(el) {
60515                 el.un('mouseenter' ,this.onEventEnter, this);
60516                 el.un('mouseleave' ,this.onEventLeave, this);
60517                 el.remove();
60518             },this);
60519             ev.els = [];
60520         }
60521     },
60522     
60523     
60524     renderEvent : function(ev,ctr) {
60525         if (!ctr) {
60526              ctr = this.view.el.select('.fc-event-container',true).first();
60527         }
60528         
60529          
60530         this.clearEvent(ev);
60531             //code
60532        
60533         
60534         
60535         ev.els = [];
60536         var cells = ev.cells;
60537         var rows = ev.rows;
60538         this.fireEvent('eventrender', this, ev);
60539         
60540         for(var i =0; i < rows.length; i++) {
60541             
60542             cls = '';
60543             if (i == 0) {
60544                 cls += ' fc-event-start';
60545             }
60546             if ((i+1) == rows.length) {
60547                 cls += ' fc-event-end';
60548             }
60549             
60550             //Roo.log(ev.data);
60551             // how many rows should it span..
60552             var cg = this.eventTmpl.append(ctr,Roo.apply({
60553                 fccls : cls
60554                 
60555             }, ev.data) , true);
60556             
60557             
60558             cg.on('mouseenter' ,this.onEventEnter, this, ev);
60559             cg.on('mouseleave' ,this.onEventLeave, this, ev);
60560             cg.on('click', this.onEventClick, this, ev);
60561             
60562             ev.els.push(cg);
60563             
60564             var sbox = rows[i].start.select('.fc-day-content',true).first().getBox();
60565             var ebox = rows[i].end.select('.fc-day-content',true).first().getBox();
60566             //Roo.log(cg);
60567              
60568             cg.setXY([sbox.x +2, sbox.y +(ev.row * 20)]);    
60569             cg.setWidth(ebox.right - sbox.x -2);
60570         }
60571     },
60572     
60573     renderEvents: function()
60574     {   
60575         // first make sure there is enough space..
60576         
60577         if (!this.eventTmpl) {
60578             this.eventTmpl = new Roo.Template(
60579                 '<div class="roo-dynamic fc-event fc-event-hori fc-event-draggable ui-draggable {fccls} {cls}"  style="position: absolute" unselectable="on">' +
60580                     '<div class="fc-event-inner">' +
60581                         '<span class="fc-event-time">{time}</span>' +
60582                         '<span class="fc-event-title" qtip="{qtip}">{title}</span>' +
60583                     '</div>' +
60584                     '<div class="ui-resizable-heandle ui-resizable-e">&nbsp;&nbsp;&nbsp;</div>' +
60585                 '</div>'
60586             );
60587                 
60588         }
60589                
60590         
60591         
60592         this.cells.each(function(c) {
60593             //Roo.log(c.select('.fc-day-content div',true).first());
60594             c.select('.fc-day-content div',true).first().setHeight(Math.max(34, (c.rows || 1) * 20));
60595         });
60596         
60597         var ctr = this.view.el.select('.fc-event-container',true).first();
60598         
60599         var cls;
60600         this.eventStore.each(function(ev){
60601             
60602             this.renderEvent(ev);
60603              
60604              
60605         }, this);
60606         this.view.layout();
60607         
60608     },
60609     
60610     onEventEnter: function (e, el,event,d) {
60611         this.fireEvent('evententer', this, el, event);
60612     },
60613     
60614     onEventLeave: function (e, el,event,d) {
60615         this.fireEvent('eventleave', this, el, event);
60616     },
60617     
60618     onEventClick: function (e, el,event,d) {
60619         this.fireEvent('eventclick', this, el, event);
60620     },
60621     
60622     onMonthChange: function () {
60623         this.store.load();
60624     },
60625     
60626     onLoad: function () {
60627         
60628         //Roo.log('calendar onload');
60629 //         
60630         if(this.eventStore.getCount() > 0){
60631             
60632            
60633             
60634             this.eventStore.each(function(d){
60635                 
60636                 
60637                 // FIXME..
60638                 var add =   d.data;
60639                 if (typeof(add.end_dt) == 'undefined')  {
60640                     Roo.log("Missing End time in calendar data: ");
60641                     Roo.log(d);
60642                     return;
60643                 }
60644                 if (typeof(add.start_dt) == 'undefined')  {
60645                     Roo.log("Missing Start time in calendar data: ");
60646                     Roo.log(d);
60647                     return;
60648                 }
60649                 add.start_dt = typeof(add.start_dt) == 'string' ? Date.parseDate(add.start_dt,'Y-m-d H:i:s') : add.start_dt,
60650                 add.end_dt = typeof(add.end_dt) == 'string' ? Date.parseDate(add.end_dt,'Y-m-d H:i:s') : add.end_dt,
60651                 add.id = add.id || d.id;
60652                 add.title = add.title || '??';
60653                 
60654                 this.addItem(d);
60655                 
60656              
60657             },this);
60658         }
60659         
60660         this.renderEvents();
60661     }
60662     
60663
60664 });
60665 /*
60666  grid : {
60667                 xtype: 'Grid',
60668                 xns: Roo.grid,
60669                 listeners : {
60670                     render : function ()
60671                     {
60672                         _this.grid = this;
60673                         
60674                         if (!this.view.el.hasClass('course-timesheet')) {
60675                             this.view.el.addClass('course-timesheet');
60676                         }
60677                         if (this.tsStyle) {
60678                             this.ds.load({});
60679                             return; 
60680                         }
60681                         Roo.log('width');
60682                         Roo.log(_this.grid.view.el.getWidth());
60683                         
60684                         
60685                         this.tsStyle =  Roo.util.CSS.createStyleSheet({
60686                             '.course-timesheet .x-grid-row' : {
60687                                 height: '80px'
60688                             },
60689                             '.x-grid-row td' : {
60690                                 'vertical-align' : 0
60691                             },
60692                             '.course-edit-link' : {
60693                                 'color' : 'blue',
60694                                 'text-overflow' : 'ellipsis',
60695                                 'overflow' : 'hidden',
60696                                 'white-space' : 'nowrap',
60697                                 'cursor' : 'pointer'
60698                             },
60699                             '.sub-link' : {
60700                                 'color' : 'green'
60701                             },
60702                             '.de-act-sup-link' : {
60703                                 'color' : 'purple',
60704                                 'text-decoration' : 'line-through'
60705                             },
60706                             '.de-act-link' : {
60707                                 'color' : 'red',
60708                                 'text-decoration' : 'line-through'
60709                             },
60710                             '.course-timesheet .course-highlight' : {
60711                                 'border-top-style': 'dashed !important',
60712                                 'border-bottom-bottom': 'dashed !important'
60713                             },
60714                             '.course-timesheet .course-item' : {
60715                                 'font-family'   : 'tahoma, arial, helvetica',
60716                                 'font-size'     : '11px',
60717                                 'overflow'      : 'hidden',
60718                                 'padding-left'  : '10px',
60719                                 'padding-right' : '10px',
60720                                 'padding-top' : '10px' 
60721                             }
60722                             
60723                         }, Roo.id());
60724                                 this.ds.load({});
60725                     }
60726                 },
60727                 autoWidth : true,
60728                 monitorWindowResize : false,
60729                 cellrenderer : function(v,x,r)
60730                 {
60731                     return v;
60732                 },
60733                 sm : {
60734                     xtype: 'CellSelectionModel',
60735                     xns: Roo.grid
60736                 },
60737                 dataSource : {
60738                     xtype: 'Store',
60739                     xns: Roo.data,
60740                     listeners : {
60741                         beforeload : function (_self, options)
60742                         {
60743                             options.params = options.params || {};
60744                             options.params._month = _this.monthField.getValue();
60745                             options.params.limit = 9999;
60746                             options.params['sort'] = 'when_dt';    
60747                             options.params['dir'] = 'ASC';    
60748                             this.proxy.loadResponse = this.loadResponse;
60749                             Roo.log("load?");
60750                             //this.addColumns();
60751                         },
60752                         load : function (_self, records, options)
60753                         {
60754                             _this.grid.view.el.select('.course-edit-link', true).on('click', function() {
60755                                 // if you click on the translation.. you can edit it...
60756                                 var el = Roo.get(this);
60757                                 var id = el.dom.getAttribute('data-id');
60758                                 var d = el.dom.getAttribute('data-date');
60759                                 var t = el.dom.getAttribute('data-time');
60760                                 //var id = this.child('span').dom.textContent;
60761                                 
60762                                 //Roo.log(this);
60763                                 Pman.Dialog.CourseCalendar.show({
60764                                     id : id,
60765                                     when_d : d,
60766                                     when_t : t,
60767                                     productitem_active : id ? 1 : 0
60768                                 }, function() {
60769                                     _this.grid.ds.load({});
60770                                 });
60771                            
60772                            });
60773                            
60774                            _this.panel.fireEvent('resize', [ '', '' ]);
60775                         }
60776                     },
60777                     loadResponse : function(o, success, response){
60778                             // this is overridden on before load..
60779                             
60780                             Roo.log("our code?");       
60781                             //Roo.log(success);
60782                             //Roo.log(response)
60783                             delete this.activeRequest;
60784                             if(!success){
60785                                 this.fireEvent("loadexception", this, o, response);
60786                                 o.request.callback.call(o.request.scope, null, o.request.arg, false);
60787                                 return;
60788                             }
60789                             var result;
60790                             try {
60791                                 result = o.reader.read(response);
60792                             }catch(e){
60793                                 Roo.log("load exception?");
60794                                 this.fireEvent("loadexception", this, o, response, e);
60795                                 o.request.callback.call(o.request.scope, null, o.request.arg, false);
60796                                 return;
60797                             }
60798                             Roo.log("ready...");        
60799                             // loop through result.records;
60800                             // and set this.tdate[date] = [] << array of records..
60801                             _this.tdata  = {};
60802                             Roo.each(result.records, function(r){
60803                                 //Roo.log(r.data);
60804                                 if(typeof(_this.tdata[r.data.when_dt.format('j')]) == 'undefined'){
60805                                     _this.tdata[r.data.when_dt.format('j')] = [];
60806                                 }
60807                                 _this.tdata[r.data.when_dt.format('j')].push(r.data);
60808                             });
60809                             
60810                             //Roo.log(_this.tdata);
60811                             
60812                             result.records = [];
60813                             result.totalRecords = 6;
60814                     
60815                             // let's generate some duumy records for the rows.
60816                             //var st = _this.dateField.getValue();
60817                             
60818                             // work out monday..
60819                             //st = st.add(Date.DAY, -1 * st.format('w'));
60820                             
60821                             var date = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60822                             
60823                             var firstOfMonth = date.getFirstDayOfMonth();
60824                             var days = date.getDaysInMonth();
60825                             var d = 1;
60826                             var firstAdded = false;
60827                             for (var i = 0; i < result.totalRecords ; i++) {
60828                                 //var d= st.add(Date.DAY, i);
60829                                 var row = {};
60830                                 var added = 0;
60831                                 for(var w = 0 ; w < 7 ; w++){
60832                                     if(!firstAdded && firstOfMonth != w){
60833                                         continue;
60834                                     }
60835                                     if(d > days){
60836                                         continue;
60837                                     }
60838                                     firstAdded = true;
60839                                     var dd = (d > 0 && d < 10) ? "0"+d : d;
60840                                     row['weekday'+w] = String.format(
60841                                                     '<span style="font-size: 16px;"><b>{0}</b></span>'+
60842                                                     '<span class="course-edit-link" style="color:blue;" data-id="0" data-date="{1}"> Add New</span>',
60843                                                     d,
60844                                                     date.format('Y-m-')+dd
60845                                                 );
60846                                     added++;
60847                                     if(typeof(_this.tdata[d]) != 'undefined'){
60848                                         Roo.each(_this.tdata[d], function(r){
60849                                             var is_sub = '';
60850                                             var deactive = '';
60851                                             var id = r.id;
60852                                             var desc = (r.productitem_id_descrip) ? r.productitem_id_descrip : '';
60853                                             if(r.parent_id*1>0){
60854                                                 is_sub = (r.productitem_id_visible*1 < 1) ? 'de-act-sup-link' :'sub-link';
60855                                                 id = r.parent_id;
60856                                             }
60857                                             if(r.productitem_id_visible*1 < 1 && r.parent_id*1 < 1){
60858                                                 deactive = 'de-act-link';
60859                                             }
60860                                             
60861                                             row['weekday'+w] += String.format(
60862                                                     '<br /><span class="course-edit-link {3} {4}" qtip="{5}" data-id="{0}">{2} - {1}</span>',
60863                                                     id, //0
60864                                                     r.product_id_name, //1
60865                                                     r.when_dt.format('h:ia'), //2
60866                                                     is_sub, //3
60867                                                     deactive, //4
60868                                                     desc // 5
60869                                             );
60870                                         });
60871                                     }
60872                                     d++;
60873                                 }
60874                                 
60875                                 // only do this if something added..
60876                                 if(added > 0){ 
60877                                     result.records.push(_this.grid.dataSource.reader.newRow(row));
60878                                 }
60879                                 
60880                                 
60881                                 // push it twice. (second one with an hour..
60882                                 
60883                             }
60884                             //Roo.log(result);
60885                             this.fireEvent("load", this, o, o.request.arg);
60886                             o.request.callback.call(o.request.scope, result, o.request.arg, true);
60887                         },
60888                     sortInfo : {field: 'when_dt', direction : 'ASC' },
60889                     proxy : {
60890                         xtype: 'HttpProxy',
60891                         xns: Roo.data,
60892                         method : 'GET',
60893                         url : baseURL + '/Roo/Shop_course.php'
60894                     },
60895                     reader : {
60896                         xtype: 'JsonReader',
60897                         xns: Roo.data,
60898                         id : 'id',
60899                         fields : [
60900                             {
60901                                 'name': 'id',
60902                                 'type': 'int'
60903                             },
60904                             {
60905                                 'name': 'when_dt',
60906                                 'type': 'string'
60907                             },
60908                             {
60909                                 'name': 'end_dt',
60910                                 'type': 'string'
60911                             },
60912                             {
60913                                 'name': 'parent_id',
60914                                 'type': 'int'
60915                             },
60916                             {
60917                                 'name': 'product_id',
60918                                 'type': 'int'
60919                             },
60920                             {
60921                                 'name': 'productitem_id',
60922                                 'type': 'int'
60923                             },
60924                             {
60925                                 'name': 'guid',
60926                                 'type': 'int'
60927                             }
60928                         ]
60929                     }
60930                 },
60931                 toolbar : {
60932                     xtype: 'Toolbar',
60933                     xns: Roo,
60934                     items : [
60935                         {
60936                             xtype: 'Button',
60937                             xns: Roo.Toolbar,
60938                             listeners : {
60939                                 click : function (_self, e)
60940                                 {
60941                                     var sd = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60942                                     sd.setMonth(sd.getMonth()-1);
60943                                     _this.monthField.setValue(sd.format('Y-m-d'));
60944                                     _this.grid.ds.load({});
60945                                 }
60946                             },
60947                             text : "Back"
60948                         },
60949                         {
60950                             xtype: 'Separator',
60951                             xns: Roo.Toolbar
60952                         },
60953                         {
60954                             xtype: 'MonthField',
60955                             xns: Roo.form,
60956                             listeners : {
60957                                 render : function (_self)
60958                                 {
60959                                     _this.monthField = _self;
60960                                    // _this.monthField.set  today
60961                                 },
60962                                 select : function (combo, date)
60963                                 {
60964                                     _this.grid.ds.load({});
60965                                 }
60966                             },
60967                             value : (function() { return new Date(); })()
60968                         },
60969                         {
60970                             xtype: 'Separator',
60971                             xns: Roo.Toolbar
60972                         },
60973                         {
60974                             xtype: 'TextItem',
60975                             xns: Roo.Toolbar,
60976                             text : "Blue: in-active, green: in-active sup-event, red: de-active, purple: de-active sup-event"
60977                         },
60978                         {
60979                             xtype: 'Fill',
60980                             xns: Roo.Toolbar
60981                         },
60982                         {
60983                             xtype: 'Button',
60984                             xns: Roo.Toolbar,
60985                             listeners : {
60986                                 click : function (_self, e)
60987                                 {
60988                                     var sd = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60989                                     sd.setMonth(sd.getMonth()+1);
60990                                     _this.monthField.setValue(sd.format('Y-m-d'));
60991                                     _this.grid.ds.load({});
60992                                 }
60993                             },
60994                             text : "Next"
60995                         }
60996                     ]
60997                 },
60998                  
60999             }
61000         };
61001         
61002         *//*
61003  * Based on:
61004  * Ext JS Library 1.1.1
61005  * Copyright(c) 2006-2007, Ext JS, LLC.
61006  *
61007  * Originally Released Under LGPL - original licence link has changed is not relivant.
61008  *
61009  * Fork - LGPL
61010  * <script type="text/javascript">
61011  */
61012  
61013 /**
61014  * @class Roo.LoadMask
61015  * A simple utility class for generically masking elements while loading data.  If the element being masked has
61016  * an underlying {@link Roo.data.Store}, the masking will be automatically synchronized with the store's loading
61017  * process and the mask element will be cached for reuse.  For all other elements, this mask will replace the
61018  * element's UpdateManager load indicator and will be destroyed after the initial load.
61019  * @constructor
61020  * Create a new LoadMask
61021  * @param {String/HTMLElement/Roo.Element} el The element or DOM node, or its id
61022  * @param {Object} config The config object
61023  */
61024 Roo.LoadMask = function(el, config){
61025     this.el = Roo.get(el);
61026     Roo.apply(this, config);
61027     if(this.store){
61028         this.store.on('beforeload', this.onBeforeLoad, this);
61029         this.store.on('load', this.onLoad, this);
61030         this.store.on('loadexception', this.onLoadException, this);
61031         this.removeMask = false;
61032     }else{
61033         var um = this.el.getUpdateManager();
61034         um.showLoadIndicator = false; // disable the default indicator
61035         um.on('beforeupdate', this.onBeforeLoad, this);
61036         um.on('update', this.onLoad, this);
61037         um.on('failure', this.onLoad, this);
61038         this.removeMask = true;
61039     }
61040 };
61041
61042 Roo.LoadMask.prototype = {
61043     /**
61044      * @cfg {Boolean} removeMask
61045      * True to create a single-use mask that is automatically destroyed after loading (useful for page loads),
61046      * False to persist the mask element reference for multiple uses (e.g., for paged data widgets).  Defaults to false.
61047      */
61048     /**
61049      * @cfg {String} msg
61050      * The text to display in a centered loading message box (defaults to 'Loading...')
61051      */
61052     msg : 'Loading...',
61053     /**
61054      * @cfg {String} msgCls
61055      * The CSS class to apply to the loading message element (defaults to "x-mask-loading")
61056      */
61057     msgCls : 'x-mask-loading',
61058
61059     /**
61060      * Read-only. True if the mask is currently disabled so that it will not be displayed (defaults to false)
61061      * @type Boolean
61062      */
61063     disabled: false,
61064
61065     /**
61066      * Disables the mask to prevent it from being displayed
61067      */
61068     disable : function(){
61069        this.disabled = true;
61070     },
61071
61072     /**
61073      * Enables the mask so that it can be displayed
61074      */
61075     enable : function(){
61076         this.disabled = false;
61077     },
61078     
61079     onLoadException : function()
61080     {
61081         Roo.log(arguments);
61082         
61083         if (typeof(arguments[3]) != 'undefined') {
61084             Roo.MessageBox.alert("Error loading",arguments[3]);
61085         } 
61086         /*
61087         try {
61088             if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
61089                 Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
61090             }   
61091         } catch(e) {
61092             
61093         }
61094         */
61095     
61096         (function() { this.el.unmask(this.removeMask); }).defer(50, this);
61097     },
61098     // private
61099     onLoad : function()
61100     {
61101         (function() { this.el.unmask(this.removeMask); }).defer(50, this);
61102     },
61103
61104     // private
61105     onBeforeLoad : function(){
61106         if(!this.disabled){
61107             (function() { this.el.mask(this.msg, this.msgCls); }).defer(50, this);
61108         }
61109     },
61110
61111     // private
61112     destroy : function(){
61113         if(this.store){
61114             this.store.un('beforeload', this.onBeforeLoad, this);
61115             this.store.un('load', this.onLoad, this);
61116             this.store.un('loadexception', this.onLoadException, this);
61117         }else{
61118             var um = this.el.getUpdateManager();
61119             um.un('beforeupdate', this.onBeforeLoad, this);
61120             um.un('update', this.onLoad, this);
61121             um.un('failure', this.onLoad, this);
61122         }
61123     }
61124 };/*
61125  * Based on:
61126  * Ext JS Library 1.1.1
61127  * Copyright(c) 2006-2007, Ext JS, LLC.
61128  *
61129  * Originally Released Under LGPL - original licence link has changed is not relivant.
61130  *
61131  * Fork - LGPL
61132  * <script type="text/javascript">
61133  */
61134
61135
61136 /**
61137  * @class Roo.XTemplate
61138  * @extends Roo.Template
61139  * Provides a template that can have nested templates for loops or conditionals. The syntax is:
61140 <pre><code>
61141 var t = new Roo.XTemplate(
61142         '&lt;select name="{name}"&gt;',
61143                 '&lt;tpl for="options"&gt;&lt;option value="{value:trim}"&gt;{text:ellipsis(10)}&lt;/option&gt;&lt;/tpl&gt;',
61144         '&lt;/select&gt;'
61145 );
61146  
61147 // then append, applying the master template values
61148  </code></pre>
61149  *
61150  * Supported features:
61151  *
61152  *  Tags:
61153
61154 <pre><code>
61155       {a_variable} - output encoded.
61156       {a_variable.format:("Y-m-d")} - call a method on the variable
61157       {a_variable:raw} - unencoded output
61158       {a_variable:toFixed(1,2)} - Roo.util.Format."toFixed"
61159       {a_variable:this.method_on_template(...)} - call a method on the template object.
61160  
61161 </code></pre>
61162  *  The tpl tag:
61163 <pre><code>
61164         &lt;tpl for="a_variable or condition.."&gt;&lt;/tpl&gt;
61165         &lt;tpl if="a_variable or condition"&gt;&lt;/tpl&gt;
61166         &lt;tpl exec="some javascript"&gt;&lt;/tpl&gt;
61167         &lt;tpl name="named_template"&gt;&lt;/tpl&gt; (experimental)
61168   
61169         &lt;tpl for="."&gt;&lt;/tpl&gt; - just iterate the property..
61170         &lt;tpl for=".."&gt;&lt;/tpl&gt; - iterates with the parent (probably the template) 
61171 </code></pre>
61172  *      
61173  */
61174 Roo.XTemplate = function()
61175 {
61176     Roo.XTemplate.superclass.constructor.apply(this, arguments);
61177     if (this.html) {
61178         this.compile();
61179     }
61180 };
61181
61182
61183 Roo.extend(Roo.XTemplate, Roo.Template, {
61184
61185     /**
61186      * The various sub templates
61187      */
61188     tpls : false,
61189     /**
61190      *
61191      * basic tag replacing syntax
61192      * WORD:WORD()
61193      *
61194      * // you can fake an object call by doing this
61195      *  x.t:(test,tesT) 
61196      * 
61197      */
61198     re : /\{([\w-\.]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
61199
61200     /**
61201      * compile the template
61202      *
61203      * This is not recursive, so I'm not sure how nested templates are really going to be handled..
61204      *
61205      */
61206     compile: function()
61207     {
61208         var s = this.html;
61209      
61210         s = ['<tpl>', s, '</tpl>'].join('');
61211     
61212         var re     = /<tpl\b[^>]*>((?:(?=([^<]+))\2|<(?!tpl\b[^>]*>))*?)<\/tpl>/,
61213             nameRe = /^<tpl\b[^>]*?for="(.*?)"/,
61214             ifRe   = /^<tpl\b[^>]*?if="(.*?)"/,
61215             execRe = /^<tpl\b[^>]*?exec="(.*?)"/,
61216             namedRe = /^<tpl\b[^>]*?name="(\w+)"/,  // named templates..
61217             m,
61218             id     = 0,
61219             tpls   = [];
61220     
61221         while(true == !!(m = s.match(re))){
61222             var forMatch   = m[0].match(nameRe),
61223                 ifMatch   = m[0].match(ifRe),
61224                 execMatch   = m[0].match(execRe),
61225                 namedMatch   = m[0].match(namedRe),
61226                 
61227                 exp  = null, 
61228                 fn   = null,
61229                 exec = null,
61230                 name = forMatch && forMatch[1] ? forMatch[1] : '';
61231                 
61232             if (ifMatch) {
61233                 // if - puts fn into test..
61234                 exp = ifMatch && ifMatch[1] ? ifMatch[1] : null;
61235                 if(exp){
61236                    fn = new Function('values', 'parent', 'with(values){ return '+(Roo.util.Format.htmlDecode(exp))+'; }');
61237                 }
61238             }
61239             
61240             if (execMatch) {
61241                 // exec - calls a function... returns empty if true is  returned.
61242                 exp = execMatch && execMatch[1] ? execMatch[1] : null;
61243                 if(exp){
61244                    exec = new Function('values', 'parent', 'with(values){ '+(Roo.util.Format.htmlDecode(exp))+'; }');
61245                 }
61246             }
61247             
61248             
61249             if (name) {
61250                 // for = 
61251                 switch(name){
61252                     case '.':  name = new Function('values', 'parent', 'with(values){ return values; }'); break;
61253                     case '..': name = new Function('values', 'parent', 'with(values){ return parent; }'); break;
61254                     default:   name = new Function('values', 'parent', 'with(values){ return '+name+'; }');
61255                 }
61256             }
61257             var uid = namedMatch ? namedMatch[1] : id;
61258             
61259             
61260             tpls.push({
61261                 id:     namedMatch ? namedMatch[1] : id,
61262                 target: name,
61263                 exec:   exec,
61264                 test:   fn,
61265                 body:   m[1] || ''
61266             });
61267             if (namedMatch) {
61268                 s = s.replace(m[0], '');
61269             } else { 
61270                 s = s.replace(m[0], '{xtpl'+ id + '}');
61271             }
61272             ++id;
61273         }
61274         this.tpls = [];
61275         for(var i = tpls.length-1; i >= 0; --i){
61276             this.compileTpl(tpls[i]);
61277             this.tpls[tpls[i].id] = tpls[i];
61278         }
61279         this.master = tpls[tpls.length-1];
61280         return this;
61281     },
61282     /**
61283      * same as applyTemplate, except it's done to one of the subTemplates
61284      * when using named templates, you can do:
61285      *
61286      * var str = pl.applySubTemplate('your-name', values);
61287      *
61288      * 
61289      * @param {Number} id of the template
61290      * @param {Object} values to apply to template
61291      * @param {Object} parent (normaly the instance of this object)
61292      */
61293     applySubTemplate : function(id, values, parent)
61294     {
61295         
61296         
61297         var t = this.tpls[id];
61298         
61299         
61300         try { 
61301             if(t.test && !t.test.call(this, values, parent)){
61302                 return '';
61303             }
61304         } catch(e) {
61305             Roo.log("Xtemplate.applySubTemplate 'test': Exception thrown");
61306             Roo.log(e.toString());
61307             Roo.log(t.test);
61308             return ''
61309         }
61310         try { 
61311             
61312             if(t.exec && t.exec.call(this, values, parent)){
61313                 return '';
61314             }
61315         } catch(e) {
61316             Roo.log("Xtemplate.applySubTemplate 'exec': Exception thrown");
61317             Roo.log(e.toString());
61318             Roo.log(t.exec);
61319             return ''
61320         }
61321         try {
61322             var vs = t.target ? t.target.call(this, values, parent) : values;
61323             parent = t.target ? values : parent;
61324             if(t.target && vs instanceof Array){
61325                 var buf = [];
61326                 for(var i = 0, len = vs.length; i < len; i++){
61327                     buf[buf.length] = t.compiled.call(this, vs[i], parent);
61328                 }
61329                 return buf.join('');
61330             }
61331             return t.compiled.call(this, vs, parent);
61332         } catch (e) {
61333             Roo.log("Xtemplate.applySubTemplate : Exception thrown");
61334             Roo.log(e.toString());
61335             Roo.log(t.compiled);
61336             return '';
61337         }
61338     },
61339
61340     compileTpl : function(tpl)
61341     {
61342         var fm = Roo.util.Format;
61343         var useF = this.disableFormats !== true;
61344         var sep = Roo.isGecko ? "+" : ",";
61345         var undef = function(str) {
61346             Roo.log("Property not found :"  + str);
61347             return '';
61348         };
61349         
61350         var fn = function(m, name, format, args)
61351         {
61352             //Roo.log(arguments);
61353             args = args ? args.replace(/\\'/g,"'") : args;
61354             //["{TEST:(a,b,c)}", "TEST", "", "a,b,c", 0, "{TEST:(a,b,c)}"]
61355             if (typeof(format) == 'undefined') {
61356                 format= 'htmlEncode';
61357             }
61358             if (format == 'raw' ) {
61359                 format = false;
61360             }
61361             
61362             if(name.substr(0, 4) == 'xtpl'){
61363                 return "'"+ sep +'this.applySubTemplate('+name.substr(4)+', values, parent)'+sep+"'";
61364             }
61365             
61366             // build an array of options to determine if value is undefined..
61367             
61368             // basically get 'xxxx.yyyy' then do
61369             // (typeof(xxxx) == 'undefined' || typeof(xxx.yyyy) == 'undefined') ?
61370             //    (function () { Roo.log("Property not found"); return ''; })() :
61371             //    ......
61372             
61373             var udef_ar = [];
61374             var lookfor = '';
61375             Roo.each(name.split('.'), function(st) {
61376                 lookfor += (lookfor.length ? '.': '') + st;
61377                 udef_ar.push(  "(typeof(" + lookfor + ") == 'undefined')"  );
61378             });
61379             
61380             var udef_st = '((' + udef_ar.join(" || ") +") ? undef('" + name + "') : "; // .. needs )
61381             
61382             
61383             if(format && useF){
61384                 
61385                 args = args ? ',' + args : "";
61386                  
61387                 if(format.substr(0, 5) != "this."){
61388                     format = "fm." + format + '(';
61389                 }else{
61390                     format = 'this.call("'+ format.substr(5) + '", ';
61391                     args = ", values";
61392                 }
61393                 
61394                 return "'"+ sep +   udef_st   +    format + name + args + "))"+sep+"'";
61395             }
61396              
61397             if (args.length) {
61398                 // called with xxyx.yuu:(test,test)
61399                 // change to ()
61400                 return "'"+ sep + udef_st  + name + '(' +  args + "))"+sep+"'";
61401             }
61402             // raw.. - :raw modifier..
61403             return "'"+ sep + udef_st  + name + ")"+sep+"'";
61404             
61405         };
61406         var body;
61407         // branched to use + in gecko and [].join() in others
61408         if(Roo.isGecko){
61409             body = "tpl.compiled = function(values, parent){  with(values) { return '" +
61410                    tpl.body.replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
61411                     "';};};";
61412         }else{
61413             body = ["tpl.compiled = function(values, parent){  with (values) { return ['"];
61414             body.push(tpl.body.replace(/(\r\n|\n)/g,
61415                             '\\n').replace(/'/g, "\\'").replace(this.re, fn));
61416             body.push("'].join('');};};");
61417             body = body.join('');
61418         }
61419         
61420         Roo.debug && Roo.log(body.replace(/\\n/,'\n'));
61421        
61422         /** eval:var:tpl eval:var:fm eval:var:useF eval:var:undef  */
61423         eval(body);
61424         
61425         return this;
61426     },
61427
61428     applyTemplate : function(values){
61429         return this.master.compiled.call(this, values, {});
61430         //var s = this.subs;
61431     },
61432
61433     apply : function(){
61434         return this.applyTemplate.apply(this, arguments);
61435     }
61436
61437  });
61438
61439 Roo.XTemplate.from = function(el){
61440     el = Roo.getDom(el);
61441     return new Roo.XTemplate(el.value || el.innerHTML);
61442 };