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                 fields : config.fields
23587             },
23588             Roo.data.Record.create(config.fields)
23589         ),
23590         proxy : new Roo.data.MemoryProxy(config.data)
23591     });
23592     this.load();
23593 };
23594 Roo.extend(Roo.data.SimpleStore, Roo.data.Store);/*
23595  * Based on:
23596  * Ext JS Library 1.1.1
23597  * Copyright(c) 2006-2007, Ext JS, LLC.
23598  *
23599  * Originally Released Under LGPL - original licence link has changed is not relivant.
23600  *
23601  * Fork - LGPL
23602  * <script type="text/javascript">
23603  */
23604
23605 /**
23606 /**
23607  * @extends Roo.data.Store
23608  * @class Roo.data.JsonStore
23609  * Small helper class to make creating Stores for JSON data easier. <br/>
23610 <pre><code>
23611 var store = new Roo.data.JsonStore({
23612     url: 'get-images.php',
23613     root: 'images',
23614     fields: ['name', 'url', {name:'size', type: 'float'}, {name:'lastmod', type:'date'}]
23615 });
23616 </code></pre>
23617  * <b>Note: Although they are not listed, this class inherits all of the config options of Store,
23618  * JsonReader and HttpProxy (unless inline data is provided).</b>
23619  * @cfg {Array} fields An array of field definition objects, or field name strings.
23620  * @constructor
23621  * @param {Object} config
23622  */
23623 Roo.data.JsonStore = function(c){
23624     Roo.data.JsonStore.superclass.constructor.call(this, Roo.apply(c, {
23625         proxy: !c.data ? new Roo.data.HttpProxy({url: c.url}) : undefined,
23626         reader: new Roo.data.JsonReader(c, c.fields)
23627     }));
23628 };
23629 Roo.extend(Roo.data.JsonStore, Roo.data.Store);/*
23630  * Based on:
23631  * Ext JS Library 1.1.1
23632  * Copyright(c) 2006-2007, Ext JS, LLC.
23633  *
23634  * Originally Released Under LGPL - original licence link has changed is not relivant.
23635  *
23636  * Fork - LGPL
23637  * <script type="text/javascript">
23638  */
23639
23640  
23641 Roo.data.Field = function(config){
23642     if(typeof config == "string"){
23643         config = {name: config};
23644     }
23645     Roo.apply(this, config);
23646     
23647     if(!this.type){
23648         this.type = "auto";
23649     }
23650     
23651     var st = Roo.data.SortTypes;
23652     // named sortTypes are supported, here we look them up
23653     if(typeof this.sortType == "string"){
23654         this.sortType = st[this.sortType];
23655     }
23656     
23657     // set default sortType for strings and dates
23658     if(!this.sortType){
23659         switch(this.type){
23660             case "string":
23661                 this.sortType = st.asUCString;
23662                 break;
23663             case "date":
23664                 this.sortType = st.asDate;
23665                 break;
23666             default:
23667                 this.sortType = st.none;
23668         }
23669     }
23670
23671     // define once
23672     var stripRe = /[\$,%]/g;
23673
23674     // prebuilt conversion function for this field, instead of
23675     // switching every time we're reading a value
23676     if(!this.convert){
23677         var cv, dateFormat = this.dateFormat;
23678         switch(this.type){
23679             case "":
23680             case "auto":
23681             case undefined:
23682                 cv = function(v){ return v; };
23683                 break;
23684             case "string":
23685                 cv = function(v){ return (v === undefined || v === null) ? '' : String(v); };
23686                 break;
23687             case "int":
23688                 cv = function(v){
23689                     return v !== undefined && v !== null && v !== '' ?
23690                            parseInt(String(v).replace(stripRe, ""), 10) : '';
23691                     };
23692                 break;
23693             case "float":
23694                 cv = function(v){
23695                     return v !== undefined && v !== null && v !== '' ?
23696                            parseFloat(String(v).replace(stripRe, ""), 10) : ''; 
23697                     };
23698                 break;
23699             case "bool":
23700             case "boolean":
23701                 cv = function(v){ return v === true || v === "true" || v == 1; };
23702                 break;
23703             case "date":
23704                 cv = function(v){
23705                     if(!v){
23706                         return '';
23707                     }
23708                     if(v instanceof Date){
23709                         return v;
23710                     }
23711                     if(dateFormat){
23712                         if(dateFormat == "timestamp"){
23713                             return new Date(v*1000);
23714                         }
23715                         return Date.parseDate(v, dateFormat);
23716                     }
23717                     var parsed = Date.parse(v);
23718                     return parsed ? new Date(parsed) : null;
23719                 };
23720              break;
23721             
23722         }
23723         this.convert = cv;
23724     }
23725 };
23726
23727 Roo.data.Field.prototype = {
23728     dateFormat: null,
23729     defaultValue: "",
23730     mapping: null,
23731     sortType : null,
23732     sortDir : "ASC"
23733 };/*
23734  * Based on:
23735  * Ext JS Library 1.1.1
23736  * Copyright(c) 2006-2007, Ext JS, LLC.
23737  *
23738  * Originally Released Under LGPL - original licence link has changed is not relivant.
23739  *
23740  * Fork - LGPL
23741  * <script type="text/javascript">
23742  */
23743  
23744 // Base class for reading structured data from a data source.  This class is intended to be
23745 // extended (see ArrayReader, JsonReader and XmlReader) and should not be created directly.
23746
23747 /**
23748  * @class Roo.data.DataReader
23749  * Base class for reading structured data from a data source.  This class is intended to be
23750  * extended (see {Roo.data.ArrayReader}, {Roo.data.JsonReader} and {Roo.data.XmlReader}) and should not be created directly.
23751  */
23752
23753 Roo.data.DataReader = function(meta, recordType){
23754     
23755     this.meta = meta;
23756     
23757     this.recordType = recordType instanceof Array ? 
23758         Roo.data.Record.create(recordType) : recordType;
23759 };
23760
23761 Roo.data.DataReader.prototype = {
23762      /**
23763      * Create an empty record
23764      * @param {Object} data (optional) - overlay some values
23765      * @return {Roo.data.Record} record created.
23766      */
23767     newRow :  function(d) {
23768         var da =  {};
23769         this.recordType.prototype.fields.each(function(c) {
23770             switch( c.type) {
23771                 case 'int' : da[c.name] = 0; break;
23772                 case 'date' : da[c.name] = new Date(); break;
23773                 case 'float' : da[c.name] = 0.0; break;
23774                 case 'boolean' : da[c.name] = false; break;
23775                 default : da[c.name] = ""; break;
23776             }
23777             
23778         });
23779         return new this.recordType(Roo.apply(da, d));
23780     }
23781     
23782     
23783 };/*
23784  * Based on:
23785  * Ext JS Library 1.1.1
23786  * Copyright(c) 2006-2007, Ext JS, LLC.
23787  *
23788  * Originally Released Under LGPL - original licence link has changed is not relivant.
23789  *
23790  * Fork - LGPL
23791  * <script type="text/javascript">
23792  */
23793
23794 /**
23795  * @class Roo.data.DataProxy
23796  * @extends Roo.data.Observable
23797  * This class is an abstract base class for implementations which provide retrieval of
23798  * unformatted data objects.<br>
23799  * <p>
23800  * DataProxy implementations are usually used in conjunction with an implementation of Roo.data.DataReader
23801  * (of the appropriate type which knows how to parse the data object) to provide a block of
23802  * {@link Roo.data.Records} to an {@link Roo.data.Store}.<br>
23803  * <p>
23804  * Custom implementations must implement the load method as described in
23805  * {@link Roo.data.HttpProxy#load}.
23806  */
23807 Roo.data.DataProxy = function(){
23808     this.addEvents({
23809         /**
23810          * @event beforeload
23811          * Fires before a network request is made to retrieve a data object.
23812          * @param {Object} This DataProxy object.
23813          * @param {Object} params The params parameter to the load function.
23814          */
23815         beforeload : true,
23816         /**
23817          * @event load
23818          * Fires before the load method's callback is called.
23819          * @param {Object} This DataProxy object.
23820          * @param {Object} o The data object.
23821          * @param {Object} arg The callback argument object passed to the load function.
23822          */
23823         load : true,
23824         /**
23825          * @event loadexception
23826          * Fires if an Exception occurs during data retrieval.
23827          * @param {Object} This DataProxy object.
23828          * @param {Object} o The data object.
23829          * @param {Object} arg The callback argument object passed to the load function.
23830          * @param {Object} e The Exception.
23831          */
23832         loadexception : true
23833     });
23834     Roo.data.DataProxy.superclass.constructor.call(this);
23835 };
23836
23837 Roo.extend(Roo.data.DataProxy, Roo.util.Observable);
23838
23839     /**
23840      * @cfg {void} listeners (Not available) Constructor blocks listeners from being set
23841      */
23842 /*
23843  * Based on:
23844  * Ext JS Library 1.1.1
23845  * Copyright(c) 2006-2007, Ext JS, LLC.
23846  *
23847  * Originally Released Under LGPL - original licence link has changed is not relivant.
23848  *
23849  * Fork - LGPL
23850  * <script type="text/javascript">
23851  */
23852 /**
23853  * @class Roo.data.MemoryProxy
23854  * An implementation of Roo.data.DataProxy that simply passes the data specified in its constructor
23855  * to the Reader when its load method is called.
23856  * @constructor
23857  * @param {Object} data The data object which the Reader uses to construct a block of Roo.data.Records.
23858  */
23859 Roo.data.MemoryProxy = function(data){
23860     if (data.data) {
23861         data = data.data;
23862     }
23863     Roo.data.MemoryProxy.superclass.constructor.call(this);
23864     this.data = data;
23865 };
23866
23867 Roo.extend(Roo.data.MemoryProxy, Roo.data.DataProxy, {
23868     
23869     /**
23870      * Load data from the requested source (in this case an in-memory
23871      * data object passed to the constructor), read the data object into
23872      * a block of Roo.data.Records using the passed Roo.data.DataReader implementation, and
23873      * process that block using the passed callback.
23874      * @param {Object} params This parameter is not used by the MemoryProxy class.
23875      * @param {Roo.data.DataReader} reader The Reader object which converts the data
23876      * object into a block of Roo.data.Records.
23877      * @param {Function} callback The function into which to pass the block of Roo.data.records.
23878      * The function must be passed <ul>
23879      * <li>The Record block object</li>
23880      * <li>The "arg" argument from the load function</li>
23881      * <li>A boolean success indicator</li>
23882      * </ul>
23883      * @param {Object} scope The scope in which to call the callback
23884      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
23885      */
23886     load : function(params, reader, callback, scope, arg){
23887         params = params || {};
23888         var result;
23889         try {
23890             result = reader.readRecords(params.data ? params.data :this.data);
23891         }catch(e){
23892             this.fireEvent("loadexception", this, arg, null, e);
23893             callback.call(scope, null, arg, false);
23894             return;
23895         }
23896         callback.call(scope, result, arg, true);
23897     },
23898     
23899     // private
23900     update : function(params, records){
23901         
23902     }
23903 });/*
23904  * Based on:
23905  * Ext JS Library 1.1.1
23906  * Copyright(c) 2006-2007, Ext JS, LLC.
23907  *
23908  * Originally Released Under LGPL - original licence link has changed is not relivant.
23909  *
23910  * Fork - LGPL
23911  * <script type="text/javascript">
23912  */
23913 /**
23914  * @class Roo.data.HttpProxy
23915  * @extends Roo.data.DataProxy
23916  * An implementation of {@link Roo.data.DataProxy} that reads a data object from an {@link Roo.data.Connection} object
23917  * configured to reference a certain URL.<br><br>
23918  * <p>
23919  * <em>Note that this class cannot be used to retrieve data from a domain other than the domain
23920  * from which the running page was served.<br><br>
23921  * <p>
23922  * For cross-domain access to remote data, use an {@link Roo.data.ScriptTagProxy}.</em><br><br>
23923  * <p>
23924  * Be aware that to enable the browser to parse an XML document, the server must set
23925  * the Content-Type header in the HTTP response to "text/xml".
23926  * @constructor
23927  * @param {Object} conn Connection config options to add to each request (e.g. {url: 'foo.php'} or
23928  * an {@link Roo.data.Connection} object.  If a Connection config is passed, the singleton {@link Roo.Ajax} object
23929  * will be used to make the request.
23930  */
23931 Roo.data.HttpProxy = function(conn){
23932     Roo.data.HttpProxy.superclass.constructor.call(this);
23933     // is conn a conn config or a real conn?
23934     this.conn = conn;
23935     this.useAjax = !conn || !conn.events;
23936   
23937 };
23938
23939 Roo.extend(Roo.data.HttpProxy, Roo.data.DataProxy, {
23940     // thse are take from connection...
23941     
23942     /**
23943      * @cfg {String} url (Optional) The default URL to be used for requests to the server. (defaults to undefined)
23944      */
23945     /**
23946      * @cfg {Object} extraParams (Optional) An object containing properties which are used as
23947      * extra parameters to each request made by this object. (defaults to undefined)
23948      */
23949     /**
23950      * @cfg {Object} defaultHeaders (Optional) An object containing request headers which are added
23951      *  to each request made by this object. (defaults to undefined)
23952      */
23953     /**
23954      * @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)
23955      */
23956     /**
23957      * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
23958      */
23959      /**
23960      * @cfg {Boolean} autoAbort (Optional) Whether this request should abort any pending requests. (defaults to false)
23961      * @type Boolean
23962      */
23963   
23964
23965     /**
23966      * @cfg {Boolean} disableCaching (Optional) True to add a unique cache-buster param to GET requests. (defaults to true)
23967      * @type Boolean
23968      */
23969     /**
23970      * Return the {@link Roo.data.Connection} object being used by this Proxy.
23971      * @return {Connection} The Connection object. This object may be used to subscribe to events on
23972      * a finer-grained basis than the DataProxy events.
23973      */
23974     getConnection : function(){
23975         return this.useAjax ? Roo.Ajax : this.conn;
23976     },
23977
23978     /**
23979      * Load data from the configured {@link Roo.data.Connection}, read the data object into
23980      * a block of Roo.data.Records using the passed {@link Roo.data.DataReader} implementation, and
23981      * process that block using the passed callback.
23982      * @param {Object} params An object containing properties which are to be used as HTTP parameters
23983      * for the request to the remote server.
23984      * @param {Roo.data.DataReader} reader The Reader object which converts the data
23985      * object into a block of Roo.data.Records.
23986      * @param {Function} callback The function into which to pass the block of Roo.data.Records.
23987      * The function must be passed <ul>
23988      * <li>The Record block object</li>
23989      * <li>The "arg" argument from the load function</li>
23990      * <li>A boolean success indicator</li>
23991      * </ul>
23992      * @param {Object} scope The scope in which to call the callback
23993      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
23994      */
23995     load : function(params, reader, callback, scope, arg){
23996         if(this.fireEvent("beforeload", this, params) !== false){
23997             var  o = {
23998                 params : params || {},
23999                 request: {
24000                     callback : callback,
24001                     scope : scope,
24002                     arg : arg
24003                 },
24004                 reader: reader,
24005                 callback : this.loadResponse,
24006                 scope: this
24007             };
24008             if(this.useAjax){
24009                 Roo.applyIf(o, this.conn);
24010                 if(this.activeRequest){
24011                     Roo.Ajax.abort(this.activeRequest);
24012                 }
24013                 this.activeRequest = Roo.Ajax.request(o);
24014             }else{
24015                 this.conn.request(o);
24016             }
24017         }else{
24018             callback.call(scope||this, null, arg, false);
24019         }
24020     },
24021
24022     // private
24023     loadResponse : function(o, success, response){
24024         delete this.activeRequest;
24025         if(!success){
24026             this.fireEvent("loadexception", this, o, response);
24027             o.request.callback.call(o.request.scope, null, o.request.arg, false);
24028             return;
24029         }
24030         var result;
24031         try {
24032             result = o.reader.read(response);
24033         }catch(e){
24034             this.fireEvent("loadexception", this, o, response, e);
24035             o.request.callback.call(o.request.scope, null, o.request.arg, false);
24036             return;
24037         }
24038         
24039         this.fireEvent("load", this, o, o.request.arg);
24040         o.request.callback.call(o.request.scope, result, o.request.arg, true);
24041     },
24042
24043     // private
24044     update : function(dataSet){
24045
24046     },
24047
24048     // private
24049     updateResponse : function(dataSet){
24050
24051     }
24052 });/*
24053  * Based on:
24054  * Ext JS Library 1.1.1
24055  * Copyright(c) 2006-2007, Ext JS, LLC.
24056  *
24057  * Originally Released Under LGPL - original licence link has changed is not relivant.
24058  *
24059  * Fork - LGPL
24060  * <script type="text/javascript">
24061  */
24062
24063 /**
24064  * @class Roo.data.ScriptTagProxy
24065  * An implementation of Roo.data.DataProxy that reads a data object from a URL which may be in a domain
24066  * other than the originating domain of the running page.<br><br>
24067  * <p>
24068  * <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
24069  * of the running page, you must use this class, rather than DataProxy.</em><br><br>
24070  * <p>
24071  * The content passed back from a server resource requested by a ScriptTagProxy is executable JavaScript
24072  * source code that is used as the source inside a &lt;script> tag.<br><br>
24073  * <p>
24074  * In order for the browser to process the returned data, the server must wrap the data object
24075  * with a call to a callback function, the name of which is passed as a parameter by the ScriptTagProxy.
24076  * Below is a Java example for a servlet which returns data for either a ScriptTagProxy, or an HttpProxy
24077  * depending on whether the callback name was passed:
24078  * <p>
24079  * <pre><code>
24080 boolean scriptTag = false;
24081 String cb = request.getParameter("callback");
24082 if (cb != null) {
24083     scriptTag = true;
24084     response.setContentType("text/javascript");
24085 } else {
24086     response.setContentType("application/x-json");
24087 }
24088 Writer out = response.getWriter();
24089 if (scriptTag) {
24090     out.write(cb + "(");
24091 }
24092 out.print(dataBlock.toJsonString());
24093 if (scriptTag) {
24094     out.write(");");
24095 }
24096 </pre></code>
24097  *
24098  * @constructor
24099  * @param {Object} config A configuration object.
24100  */
24101 Roo.data.ScriptTagProxy = function(config){
24102     Roo.data.ScriptTagProxy.superclass.constructor.call(this);
24103     Roo.apply(this, config);
24104     this.head = document.getElementsByTagName("head")[0];
24105 };
24106
24107 Roo.data.ScriptTagProxy.TRANS_ID = 1000;
24108
24109 Roo.extend(Roo.data.ScriptTagProxy, Roo.data.DataProxy, {
24110     /**
24111      * @cfg {String} url The URL from which to request the data object.
24112      */
24113     /**
24114      * @cfg {Number} timeout (Optional) The number of milliseconds to wait for a response. Defaults to 30 seconds.
24115      */
24116     timeout : 30000,
24117     /**
24118      * @cfg {String} callbackParam (Optional) The name of the parameter to pass to the server which tells
24119      * the server the name of the callback function set up by the load call to process the returned data object.
24120      * Defaults to "callback".<p>The server-side processing must read this parameter value, and generate
24121      * javascript output which calls this named function passing the data object as its only parameter.
24122      */
24123     callbackParam : "callback",
24124     /**
24125      *  @cfg {Boolean} nocache (Optional) Defaults to true. Disable cacheing by adding a unique parameter
24126      * name to the request.
24127      */
24128     nocache : true,
24129
24130     /**
24131      * Load data from the configured URL, read the data object into
24132      * a block of Roo.data.Records using the passed Roo.data.DataReader implementation, and
24133      * process that block using the passed callback.
24134      * @param {Object} params An object containing properties which are to be used as HTTP parameters
24135      * for the request to the remote server.
24136      * @param {Roo.data.DataReader} reader The Reader object which converts the data
24137      * object into a block of Roo.data.Records.
24138      * @param {Function} callback The function into which to pass the block of Roo.data.Records.
24139      * The function must be passed <ul>
24140      * <li>The Record block object</li>
24141      * <li>The "arg" argument from the load function</li>
24142      * <li>A boolean success indicator</li>
24143      * </ul>
24144      * @param {Object} scope The scope in which to call the callback
24145      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
24146      */
24147     load : function(params, reader, callback, scope, arg){
24148         if(this.fireEvent("beforeload", this, params) !== false){
24149
24150             var p = Roo.urlEncode(Roo.apply(params, this.extraParams));
24151
24152             var url = this.url;
24153             url += (url.indexOf("?") != -1 ? "&" : "?") + p;
24154             if(this.nocache){
24155                 url += "&_dc=" + (new Date().getTime());
24156             }
24157             var transId = ++Roo.data.ScriptTagProxy.TRANS_ID;
24158             var trans = {
24159                 id : transId,
24160                 cb : "stcCallback"+transId,
24161                 scriptId : "stcScript"+transId,
24162                 params : params,
24163                 arg : arg,
24164                 url : url,
24165                 callback : callback,
24166                 scope : scope,
24167                 reader : reader
24168             };
24169             var conn = this;
24170
24171             window[trans.cb] = function(o){
24172                 conn.handleResponse(o, trans);
24173             };
24174
24175             url += String.format("&{0}={1}", this.callbackParam, trans.cb);
24176
24177             if(this.autoAbort !== false){
24178                 this.abort();
24179             }
24180
24181             trans.timeoutId = this.handleFailure.defer(this.timeout, this, [trans]);
24182
24183             var script = document.createElement("script");
24184             script.setAttribute("src", url);
24185             script.setAttribute("type", "text/javascript");
24186             script.setAttribute("id", trans.scriptId);
24187             this.head.appendChild(script);
24188
24189             this.trans = trans;
24190         }else{
24191             callback.call(scope||this, null, arg, false);
24192         }
24193     },
24194
24195     // private
24196     isLoading : function(){
24197         return this.trans ? true : false;
24198     },
24199
24200     /**
24201      * Abort the current server request.
24202      */
24203     abort : function(){
24204         if(this.isLoading()){
24205             this.destroyTrans(this.trans);
24206         }
24207     },
24208
24209     // private
24210     destroyTrans : function(trans, isLoaded){
24211         this.head.removeChild(document.getElementById(trans.scriptId));
24212         clearTimeout(trans.timeoutId);
24213         if(isLoaded){
24214             window[trans.cb] = undefined;
24215             try{
24216                 delete window[trans.cb];
24217             }catch(e){}
24218         }else{
24219             // if hasn't been loaded, wait for load to remove it to prevent script error
24220             window[trans.cb] = function(){
24221                 window[trans.cb] = undefined;
24222                 try{
24223                     delete window[trans.cb];
24224                 }catch(e){}
24225             };
24226         }
24227     },
24228
24229     // private
24230     handleResponse : function(o, trans){
24231         this.trans = false;
24232         this.destroyTrans(trans, true);
24233         var result;
24234         try {
24235             result = trans.reader.readRecords(o);
24236         }catch(e){
24237             this.fireEvent("loadexception", this, o, trans.arg, e);
24238             trans.callback.call(trans.scope||window, null, trans.arg, false);
24239             return;
24240         }
24241         this.fireEvent("load", this, o, trans.arg);
24242         trans.callback.call(trans.scope||window, result, trans.arg, true);
24243     },
24244
24245     // private
24246     handleFailure : function(trans){
24247         this.trans = false;
24248         this.destroyTrans(trans, false);
24249         this.fireEvent("loadexception", this, null, trans.arg);
24250         trans.callback.call(trans.scope||window, null, trans.arg, false);
24251     }
24252 });/*
24253  * Based on:
24254  * Ext JS Library 1.1.1
24255  * Copyright(c) 2006-2007, Ext JS, LLC.
24256  *
24257  * Originally Released Under LGPL - original licence link has changed is not relivant.
24258  *
24259  * Fork - LGPL
24260  * <script type="text/javascript">
24261  */
24262
24263 /**
24264  * @class Roo.data.JsonReader
24265  * @extends Roo.data.DataReader
24266  * Data reader class to create an Array of Roo.data.Record objects from a JSON response
24267  * based on mappings in a provided Roo.data.Record constructor.
24268  * 
24269  * The default behaviour of a store is to send ?_requestMeta=1, unless the class has recieved 'metaData' property
24270  * in the reply previously. 
24271  * 
24272  * <p>
24273  * Example code:
24274  * <pre><code>
24275 var RecordDef = Roo.data.Record.create([
24276     {name: 'name', mapping: 'name'},     // "mapping" property not needed if it's the same as "name"
24277     {name: 'occupation'}                 // This field will use "occupation" as the mapping.
24278 ]);
24279 var myReader = new Roo.data.JsonReader({
24280     totalProperty: "results",    // The property which contains the total dataset size (optional)
24281     root: "rows",                // The property which contains an Array of row objects
24282     id: "id"                     // The property within each row object that provides an ID for the record (optional)
24283 }, RecordDef);
24284 </code></pre>
24285  * <p>
24286  * This would consume a JSON file like this:
24287  * <pre><code>
24288 { 'results': 2, 'rows': [
24289     { 'id': 1, 'name': 'Bill', occupation: 'Gardener' },
24290     { 'id': 2, 'name': 'Ben', occupation: 'Horticulturalist' } ]
24291 }
24292 </code></pre>
24293  * @cfg {String} totalProperty Name of the property from which to retrieve the total number of records
24294  * in the dataset. This is only needed if the whole dataset is not passed in one go, but is being
24295  * paged from the remote server.
24296  * @cfg {String} successProperty Name of the property from which to retrieve the success attribute used by forms.
24297  * @cfg {String} root name of the property which contains the Array of row objects.
24298  * @cfg {String} id Name of the property within a row object that contains a record identifier value.
24299  * @cfg {Array} fields Array of field definition objects
24300  * @constructor
24301  * Create a new JsonReader
24302  * @param {Object} meta Metadata configuration options
24303  * @param {Object} recordType Either an Array of field definition objects,
24304  * or an {@link Roo.data.Record} object created using {@link Roo.data.Record#create}.
24305  */
24306 Roo.data.JsonReader = function(meta, recordType){
24307     
24308     meta = meta || {};
24309     // set some defaults:
24310     Roo.applyIf(meta, {
24311         totalProperty: 'total',
24312         successProperty : 'success',
24313         root : 'data',
24314         id : 'id'
24315     });
24316     
24317     Roo.data.JsonReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24318 };
24319 Roo.extend(Roo.data.JsonReader, Roo.data.DataReader, {
24320     
24321     /**
24322      * @prop {Boolean} metaFromRemote  - if the meta data was loaded from the remote source.
24323      * Used by Store query builder to append _requestMeta to params.
24324      * 
24325      */
24326     metaFromRemote : false,
24327     /**
24328      * This method is only used by a DataProxy which has retrieved data from a remote server.
24329      * @param {Object} response The XHR object which contains the JSON data in its responseText.
24330      * @return {Object} data A data block which is used by an Roo.data.Store object as
24331      * a cache of Roo.data.Records.
24332      */
24333     read : function(response){
24334         var json = response.responseText;
24335        
24336         var o = /* eval:var:o */ eval("("+json+")");
24337         if(!o) {
24338             throw {message: "JsonReader.read: Json object not found"};
24339         }
24340         
24341         if(o.metaData){
24342             
24343             delete this.ef;
24344             this.metaFromRemote = true;
24345             this.meta = o.metaData;
24346             this.recordType = Roo.data.Record.create(o.metaData.fields);
24347             this.onMetaChange(this.meta, this.recordType, o);
24348         }
24349         return this.readRecords(o);
24350     },
24351
24352     // private function a store will implement
24353     onMetaChange : function(meta, recordType, o){
24354
24355     },
24356
24357     /**
24358          * @ignore
24359          */
24360     simpleAccess: function(obj, subsc) {
24361         return obj[subsc];
24362     },
24363
24364         /**
24365          * @ignore
24366          */
24367     getJsonAccessor: function(){
24368         var re = /[\[\.]/;
24369         return function(expr) {
24370             try {
24371                 return(re.test(expr))
24372                     ? new Function("obj", "return obj." + expr)
24373                     : function(obj){
24374                         return obj[expr];
24375                     };
24376             } catch(e){}
24377             return Roo.emptyFn;
24378         };
24379     }(),
24380
24381     /**
24382      * Create a data block containing Roo.data.Records from an XML document.
24383      * @param {Object} o An object which contains an Array of row objects in the property specified
24384      * in the config as 'root, and optionally a property, specified in the config as 'totalProperty'
24385      * which contains the total size of the dataset.
24386      * @return {Object} data A data block which is used by an Roo.data.Store object as
24387      * a cache of Roo.data.Records.
24388      */
24389     readRecords : function(o){
24390         /**
24391          * After any data loads, the raw JSON data is available for further custom processing.
24392          * @type Object
24393          */
24394         this.o = o;
24395         var s = this.meta, Record = this.recordType,
24396             f = Record ? Record.prototype.fields : null, fi = f ? f.items : [], fl = f ? f.length : 0;
24397
24398 //      Generate extraction functions for the totalProperty, the root, the id, and for each field
24399         if (!this.ef) {
24400             if(s.totalProperty) {
24401                     this.getTotal = this.getJsonAccessor(s.totalProperty);
24402                 }
24403                 if(s.successProperty) {
24404                     this.getSuccess = this.getJsonAccessor(s.successProperty);
24405                 }
24406                 this.getRoot = s.root ? this.getJsonAccessor(s.root) : function(p){return p;};
24407                 if (s.id) {
24408                         var g = this.getJsonAccessor(s.id);
24409                         this.getId = function(rec) {
24410                                 var r = g(rec);  
24411                                 return (r === undefined || r === "") ? null : r;
24412                         };
24413                 } else {
24414                         this.getId = function(){return null;};
24415                 }
24416             this.ef = [];
24417             for(var jj = 0; jj < fl; jj++){
24418                 f = fi[jj];
24419                 var map = (f.mapping !== undefined && f.mapping !== null) ? f.mapping : f.name;
24420                 this.ef[jj] = this.getJsonAccessor(map);
24421             }
24422         }
24423
24424         var root = this.getRoot(o), c = root.length, totalRecords = c, success = true;
24425         if(s.totalProperty){
24426             var vt = parseInt(this.getTotal(o), 10);
24427             if(!isNaN(vt)){
24428                 totalRecords = vt;
24429             }
24430         }
24431         if(s.successProperty){
24432             var vs = this.getSuccess(o);
24433             if(vs === false || vs === 'false'){
24434                 success = false;
24435             }
24436         }
24437         var records = [];
24438         for(var i = 0; i < c; i++){
24439                 var n = root[i];
24440             var values = {};
24441             var id = this.getId(n);
24442             for(var j = 0; j < fl; j++){
24443                 f = fi[j];
24444             var v = this.ef[j](n);
24445             if (!f.convert) {
24446                 Roo.log('missing convert for ' + f.name);
24447                 Roo.log(f);
24448                 continue;
24449             }
24450             values[f.name] = f.convert((v !== undefined) ? v : f.defaultValue);
24451             }
24452             var record = new Record(values, id);
24453             record.json = n;
24454             records[i] = record;
24455         }
24456         return {
24457             raw : o,
24458             success : success,
24459             records : records,
24460             totalRecords : totalRecords
24461         };
24462     }
24463 });/*
24464  * Based on:
24465  * Ext JS Library 1.1.1
24466  * Copyright(c) 2006-2007, Ext JS, LLC.
24467  *
24468  * Originally Released Under LGPL - original licence link has changed is not relivant.
24469  *
24470  * Fork - LGPL
24471  * <script type="text/javascript">
24472  */
24473
24474 /**
24475  * @class Roo.data.XmlReader
24476  * @extends Roo.data.DataReader
24477  * Data reader class to create an Array of {@link Roo.data.Record} objects from an XML document
24478  * based on mappings in a provided Roo.data.Record constructor.<br><br>
24479  * <p>
24480  * <em>Note that in order for the browser to parse a returned XML document, the Content-Type
24481  * header in the HTTP response must be set to "text/xml".</em>
24482  * <p>
24483  * Example code:
24484  * <pre><code>
24485 var RecordDef = Roo.data.Record.create([
24486    {name: 'name', mapping: 'name'},     // "mapping" property not needed if it's the same as "name"
24487    {name: 'occupation'}                 // This field will use "occupation" as the mapping.
24488 ]);
24489 var myReader = new Roo.data.XmlReader({
24490    totalRecords: "results", // The element which contains the total dataset size (optional)
24491    record: "row",           // The repeated element which contains row information
24492    id: "id"                 // The element within the row that provides an ID for the record (optional)
24493 }, RecordDef);
24494 </code></pre>
24495  * <p>
24496  * This would consume an XML file like this:
24497  * <pre><code>
24498 &lt;?xml?>
24499 &lt;dataset>
24500  &lt;results>2&lt;/results>
24501  &lt;row>
24502    &lt;id>1&lt;/id>
24503    &lt;name>Bill&lt;/name>
24504    &lt;occupation>Gardener&lt;/occupation>
24505  &lt;/row>
24506  &lt;row>
24507    &lt;id>2&lt;/id>
24508    &lt;name>Ben&lt;/name>
24509    &lt;occupation>Horticulturalist&lt;/occupation>
24510  &lt;/row>
24511 &lt;/dataset>
24512 </code></pre>
24513  * @cfg {String} totalRecords The DomQuery path from which to retrieve the total number of records
24514  * in the dataset. This is only needed if the whole dataset is not passed in one go, but is being
24515  * paged from the remote server.
24516  * @cfg {String} record The DomQuery path to the repeated element which contains record information.
24517  * @cfg {String} success The DomQuery path to the success attribute used by forms.
24518  * @cfg {String} id The DomQuery path relative from the record element to the element that contains
24519  * a record identifier value.
24520  * @constructor
24521  * Create a new XmlReader
24522  * @param {Object} meta Metadata configuration options
24523  * @param {Mixed} recordType The definition of the data record type to produce.  This can be either a valid
24524  * Record subclass created with {@link Roo.data.Record#create}, or an array of objects with which to call
24525  * Roo.data.Record.create.  See the {@link Roo.data.Record} class for more details.
24526  */
24527 Roo.data.XmlReader = function(meta, recordType){
24528     meta = meta || {};
24529     Roo.data.XmlReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24530 };
24531 Roo.extend(Roo.data.XmlReader, Roo.data.DataReader, {
24532     /**
24533      * This method is only used by a DataProxy which has retrieved data from a remote server.
24534          * @param {Object} response The XHR object which contains the parsed XML document.  The response is expected
24535          * to contain a method called 'responseXML' that returns an XML document object.
24536      * @return {Object} records A data block which is used by an {@link Roo.data.Store} as
24537      * a cache of Roo.data.Records.
24538      */
24539     read : function(response){
24540         var doc = response.responseXML;
24541         if(!doc) {
24542             throw {message: "XmlReader.read: XML Document not available"};
24543         }
24544         return this.readRecords(doc);
24545     },
24546
24547     /**
24548      * Create a data block containing Roo.data.Records from an XML document.
24549          * @param {Object} doc A parsed XML document.
24550      * @return {Object} records A data block which is used by an {@link Roo.data.Store} as
24551      * a cache of Roo.data.Records.
24552      */
24553     readRecords : function(doc){
24554         /**
24555          * After any data loads/reads, the raw XML Document is available for further custom processing.
24556          * @type XMLDocument
24557          */
24558         this.xmlData = doc;
24559         var root = doc.documentElement || doc;
24560         var q = Roo.DomQuery;
24561         var recordType = this.recordType, fields = recordType.prototype.fields;
24562         var sid = this.meta.id;
24563         var totalRecords = 0, success = true;
24564         if(this.meta.totalRecords){
24565             totalRecords = q.selectNumber(this.meta.totalRecords, root, 0);
24566         }
24567         
24568         if(this.meta.success){
24569             var sv = q.selectValue(this.meta.success, root, true);
24570             success = sv !== false && sv !== 'false';
24571         }
24572         var records = [];
24573         var ns = q.select(this.meta.record, root);
24574         for(var i = 0, len = ns.length; i < len; i++) {
24575                 var n = ns[i];
24576                 var values = {};
24577                 var id = sid ? q.selectValue(sid, n) : undefined;
24578                 for(var j = 0, jlen = fields.length; j < jlen; j++){
24579                     var f = fields.items[j];
24580                 var v = q.selectValue(f.mapping || f.name, n, f.defaultValue);
24581                     v = f.convert(v);
24582                     values[f.name] = v;
24583                 }
24584                 var record = new recordType(values, id);
24585                 record.node = n;
24586                 records[records.length] = record;
24587             }
24588
24589             return {
24590                 success : success,
24591                 records : records,
24592                 totalRecords : totalRecords || records.length
24593             };
24594     }
24595 });/*
24596  * Based on:
24597  * Ext JS Library 1.1.1
24598  * Copyright(c) 2006-2007, Ext JS, LLC.
24599  *
24600  * Originally Released Under LGPL - original licence link has changed is not relivant.
24601  *
24602  * Fork - LGPL
24603  * <script type="text/javascript">
24604  */
24605
24606 /**
24607  * @class Roo.data.ArrayReader
24608  * @extends Roo.data.DataReader
24609  * Data reader class to create an Array of Roo.data.Record objects from an Array.
24610  * Each element of that Array represents a row of data fields. The
24611  * fields are pulled into a Record object using as a subscript, the <em>mapping</em> property
24612  * of the field definition if it exists, or the field's ordinal position in the definition.<br>
24613  * <p>
24614  * Example code:.
24615  * <pre><code>
24616 var RecordDef = Roo.data.Record.create([
24617     {name: 'name', mapping: 1},         // "mapping" only needed if an "id" field is present which
24618     {name: 'occupation', mapping: 2}    // precludes using the ordinal position as the index.
24619 ]);
24620 var myReader = new Roo.data.ArrayReader({
24621     id: 0                     // The subscript within row Array that provides an ID for the Record (optional)
24622 }, RecordDef);
24623 </code></pre>
24624  * <p>
24625  * This would consume an Array like this:
24626  * <pre><code>
24627 [ [1, 'Bill', 'Gardener'], [2, 'Ben', 'Horticulturalist'] ]
24628   </code></pre>
24629  
24630  * @constructor
24631  * Create a new JsonReader
24632  * @param {Object} meta Metadata configuration options.
24633  * @param {Object|Array} recordType Either an Array of field definition objects
24634  * 
24635  * @cfg {Array} fields Array of field definition objects
24636  * @cfg {String} id Name of the property within a row object that contains a record identifier value.
24637  * as specified to {@link Roo.data.Record#create},
24638  * or an {@link Roo.data.Record} object
24639  *
24640  * 
24641  * created using {@link Roo.data.Record#create}.
24642  */
24643 Roo.data.ArrayReader = function(meta, recordType)
24644 {    
24645     Roo.data.ArrayReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24646 };
24647
24648 Roo.extend(Roo.data.ArrayReader, Roo.data.JsonReader, {
24649     /**
24650      * Create a data block containing Roo.data.Records from an XML document.
24651      * @param {Object} o An Array of row objects which represents the dataset.
24652      * @return {Object} A data block which is used by an {@link Roo.data.Store} object as
24653      * a cache of Roo.data.Records.
24654      */
24655     readRecords : function(o)
24656     {
24657         var sid = this.meta ? this.meta.id : null;
24658         var recordType = this.recordType, fields = recordType.prototype.fields;
24659         var records = [];
24660         var root = o;
24661         for(var i = 0; i < root.length; i++){
24662                 var n = root[i];
24663             var values = {};
24664             var id = ((sid || sid === 0) && n[sid] !== undefined && n[sid] !== "" ? n[sid] : null);
24665             for(var j = 0, jlen = fields.length; j < jlen; j++){
24666                 var f = fields.items[j];
24667                 var k = f.mapping !== undefined && f.mapping !== null ? f.mapping : j;
24668                 var v = n[k] !== undefined ? n[k] : f.defaultValue;
24669                 v = f.convert(v);
24670                 values[f.name] = v;
24671             }
24672             var record = new recordType(values, id);
24673             record.json = n;
24674             records[records.length] = record;
24675         }
24676         return {
24677             records : records,
24678             totalRecords : records.length
24679         };
24680     }
24681 });/*
24682  * Based on:
24683  * Ext JS Library 1.1.1
24684  * Copyright(c) 2006-2007, Ext JS, LLC.
24685  *
24686  * Originally Released Under LGPL - original licence link has changed is not relivant.
24687  *
24688  * Fork - LGPL
24689  * <script type="text/javascript">
24690  */
24691
24692
24693 /**
24694  * @class Roo.data.Tree
24695  * @extends Roo.util.Observable
24696  * Represents a tree data structure and bubbles all the events for its nodes. The nodes
24697  * in the tree have most standard DOM functionality.
24698  * @constructor
24699  * @param {Node} root (optional) The root node
24700  */
24701 Roo.data.Tree = function(root){
24702    this.nodeHash = {};
24703    /**
24704     * The root node for this tree
24705     * @type Node
24706     */
24707    this.root = null;
24708    if(root){
24709        this.setRootNode(root);
24710    }
24711    this.addEvents({
24712        /**
24713         * @event append
24714         * Fires when a new child node is appended to a node in this tree.
24715         * @param {Tree} tree The owner tree
24716         * @param {Node} parent The parent node
24717         * @param {Node} node The newly appended node
24718         * @param {Number} index The index of the newly appended node
24719         */
24720        "append" : true,
24721        /**
24722         * @event remove
24723         * Fires when a child node is removed from a node in this tree.
24724         * @param {Tree} tree The owner tree
24725         * @param {Node} parent The parent node
24726         * @param {Node} node The child node removed
24727         */
24728        "remove" : true,
24729        /**
24730         * @event move
24731         * Fires when a node is moved to a new location in the tree
24732         * @param {Tree} tree The owner tree
24733         * @param {Node} node The node moved
24734         * @param {Node} oldParent The old parent of this node
24735         * @param {Node} newParent The new parent of this node
24736         * @param {Number} index The index it was moved to
24737         */
24738        "move" : true,
24739        /**
24740         * @event insert
24741         * Fires when a new child node is inserted in a node in this tree.
24742         * @param {Tree} tree The owner tree
24743         * @param {Node} parent The parent node
24744         * @param {Node} node The child node inserted
24745         * @param {Node} refNode The child node the node was inserted before
24746         */
24747        "insert" : true,
24748        /**
24749         * @event beforeappend
24750         * Fires before a new child is appended to a node in this tree, return false to cancel the append.
24751         * @param {Tree} tree The owner tree
24752         * @param {Node} parent The parent node
24753         * @param {Node} node The child node to be appended
24754         */
24755        "beforeappend" : true,
24756        /**
24757         * @event beforeremove
24758         * Fires before a child is removed from a node in this tree, return false to cancel the remove.
24759         * @param {Tree} tree The owner tree
24760         * @param {Node} parent The parent node
24761         * @param {Node} node The child node to be removed
24762         */
24763        "beforeremove" : true,
24764        /**
24765         * @event beforemove
24766         * Fires before a node is moved to a new location in the tree. Return false to cancel the move.
24767         * @param {Tree} tree The owner tree
24768         * @param {Node} node The node being moved
24769         * @param {Node} oldParent The parent of the node
24770         * @param {Node} newParent The new parent the node is moving to
24771         * @param {Number} index The index it is being moved to
24772         */
24773        "beforemove" : true,
24774        /**
24775         * @event beforeinsert
24776         * Fires before a new child is inserted in a node in this tree, return false to cancel the insert.
24777         * @param {Tree} tree The owner tree
24778         * @param {Node} parent The parent node
24779         * @param {Node} node The child node to be inserted
24780         * @param {Node} refNode The child node the node is being inserted before
24781         */
24782        "beforeinsert" : true
24783    });
24784
24785     Roo.data.Tree.superclass.constructor.call(this);
24786 };
24787
24788 Roo.extend(Roo.data.Tree, Roo.util.Observable, {
24789     pathSeparator: "/",
24790
24791     proxyNodeEvent : function(){
24792         return this.fireEvent.apply(this, arguments);
24793     },
24794
24795     /**
24796      * Returns the root node for this tree.
24797      * @return {Node}
24798      */
24799     getRootNode : function(){
24800         return this.root;
24801     },
24802
24803     /**
24804      * Sets the root node for this tree.
24805      * @param {Node} node
24806      * @return {Node}
24807      */
24808     setRootNode : function(node){
24809         this.root = node;
24810         node.ownerTree = this;
24811         node.isRoot = true;
24812         this.registerNode(node);
24813         return node;
24814     },
24815
24816     /**
24817      * Gets a node in this tree by its id.
24818      * @param {String} id
24819      * @return {Node}
24820      */
24821     getNodeById : function(id){
24822         return this.nodeHash[id];
24823     },
24824
24825     registerNode : function(node){
24826         this.nodeHash[node.id] = node;
24827     },
24828
24829     unregisterNode : function(node){
24830         delete this.nodeHash[node.id];
24831     },
24832
24833     toString : function(){
24834         return "[Tree"+(this.id?" "+this.id:"")+"]";
24835     }
24836 });
24837
24838 /**
24839  * @class Roo.data.Node
24840  * @extends Roo.util.Observable
24841  * @cfg {Boolean} leaf true if this node is a leaf and does not have children
24842  * @cfg {String} id The id for this node. If one is not specified, one is generated.
24843  * @constructor
24844  * @param {Object} attributes The attributes/config for the node
24845  */
24846 Roo.data.Node = function(attributes){
24847     /**
24848      * The attributes supplied for the node. You can use this property to access any custom attributes you supplied.
24849      * @type {Object}
24850      */
24851     this.attributes = attributes || {};
24852     this.leaf = this.attributes.leaf;
24853     /**
24854      * The node id. @type String
24855      */
24856     this.id = this.attributes.id;
24857     if(!this.id){
24858         this.id = Roo.id(null, "ynode-");
24859         this.attributes.id = this.id;
24860     }
24861      
24862     
24863     /**
24864      * All child nodes of this node. @type Array
24865      */
24866     this.childNodes = [];
24867     if(!this.childNodes.indexOf){ // indexOf is a must
24868         this.childNodes.indexOf = function(o){
24869             for(var i = 0, len = this.length; i < len; i++){
24870                 if(this[i] == o) {
24871                     return i;
24872                 }
24873             }
24874             return -1;
24875         };
24876     }
24877     /**
24878      * The parent node for this node. @type Node
24879      */
24880     this.parentNode = null;
24881     /**
24882      * The first direct child node of this node, or null if this node has no child nodes. @type Node
24883      */
24884     this.firstChild = null;
24885     /**
24886      * The last direct child node of this node, or null if this node has no child nodes. @type Node
24887      */
24888     this.lastChild = null;
24889     /**
24890      * The node immediately preceding this node in the tree, or null if there is no sibling node. @type Node
24891      */
24892     this.previousSibling = null;
24893     /**
24894      * The node immediately following this node in the tree, or null if there is no sibling node. @type Node
24895      */
24896     this.nextSibling = null;
24897
24898     this.addEvents({
24899        /**
24900         * @event append
24901         * Fires when a new child node is appended
24902         * @param {Tree} tree The owner tree
24903         * @param {Node} this This node
24904         * @param {Node} node The newly appended node
24905         * @param {Number} index The index of the newly appended node
24906         */
24907        "append" : true,
24908        /**
24909         * @event remove
24910         * Fires when a child node is removed
24911         * @param {Tree} tree The owner tree
24912         * @param {Node} this This node
24913         * @param {Node} node The removed node
24914         */
24915        "remove" : true,
24916        /**
24917         * @event move
24918         * Fires when this node is moved to a new location in the tree
24919         * @param {Tree} tree The owner tree
24920         * @param {Node} this This node
24921         * @param {Node} oldParent The old parent of this node
24922         * @param {Node} newParent The new parent of this node
24923         * @param {Number} index The index it was moved to
24924         */
24925        "move" : true,
24926        /**
24927         * @event insert
24928         * Fires when a new child node is inserted.
24929         * @param {Tree} tree The owner tree
24930         * @param {Node} this This node
24931         * @param {Node} node The child node inserted
24932         * @param {Node} refNode The child node the node was inserted before
24933         */
24934        "insert" : true,
24935        /**
24936         * @event beforeappend
24937         * Fires before a new child is appended, return false to cancel the append.
24938         * @param {Tree} tree The owner tree
24939         * @param {Node} this This node
24940         * @param {Node} node The child node to be appended
24941         */
24942        "beforeappend" : true,
24943        /**
24944         * @event beforeremove
24945         * Fires before a child is removed, return false to cancel the remove.
24946         * @param {Tree} tree The owner tree
24947         * @param {Node} this This node
24948         * @param {Node} node The child node to be removed
24949         */
24950        "beforeremove" : true,
24951        /**
24952         * @event beforemove
24953         * Fires before this node is moved to a new location in the tree. Return false to cancel the move.
24954         * @param {Tree} tree The owner tree
24955         * @param {Node} this This node
24956         * @param {Node} oldParent The parent of this node
24957         * @param {Node} newParent The new parent this node is moving to
24958         * @param {Number} index The index it is being moved to
24959         */
24960        "beforemove" : true,
24961        /**
24962         * @event beforeinsert
24963         * Fires before a new child is inserted, return false to cancel the insert.
24964         * @param {Tree} tree The owner tree
24965         * @param {Node} this This node
24966         * @param {Node} node The child node to be inserted
24967         * @param {Node} refNode The child node the node is being inserted before
24968         */
24969        "beforeinsert" : true
24970    });
24971     this.listeners = this.attributes.listeners;
24972     Roo.data.Node.superclass.constructor.call(this);
24973 };
24974
24975 Roo.extend(Roo.data.Node, Roo.util.Observable, {
24976     fireEvent : function(evtName){
24977         // first do standard event for this node
24978         if(Roo.data.Node.superclass.fireEvent.apply(this, arguments) === false){
24979             return false;
24980         }
24981         // then bubble it up to the tree if the event wasn't cancelled
24982         var ot = this.getOwnerTree();
24983         if(ot){
24984             if(ot.proxyNodeEvent.apply(ot, arguments) === false){
24985                 return false;
24986             }
24987         }
24988         return true;
24989     },
24990
24991     /**
24992      * Returns true if this node is a leaf
24993      * @return {Boolean}
24994      */
24995     isLeaf : function(){
24996         return this.leaf === true;
24997     },
24998
24999     // private
25000     setFirstChild : function(node){
25001         this.firstChild = node;
25002     },
25003
25004     //private
25005     setLastChild : function(node){
25006         this.lastChild = node;
25007     },
25008
25009
25010     /**
25011      * Returns true if this node is the last child of its parent
25012      * @return {Boolean}
25013      */
25014     isLast : function(){
25015        return (!this.parentNode ? true : this.parentNode.lastChild == this);
25016     },
25017
25018     /**
25019      * Returns true if this node is the first child of its parent
25020      * @return {Boolean}
25021      */
25022     isFirst : function(){
25023        return (!this.parentNode ? true : this.parentNode.firstChild == this);
25024     },
25025
25026     hasChildNodes : function(){
25027         return !this.isLeaf() && this.childNodes.length > 0;
25028     },
25029
25030     /**
25031      * Insert node(s) as the last child node of this node.
25032      * @param {Node/Array} node The node or Array of nodes to append
25033      * @return {Node} The appended node if single append, or null if an array was passed
25034      */
25035     appendChild : function(node){
25036         var multi = false;
25037         if(node instanceof Array){
25038             multi = node;
25039         }else if(arguments.length > 1){
25040             multi = arguments;
25041         }
25042         
25043         // if passed an array or multiple args do them one by one
25044         if(multi){
25045             for(var i = 0, len = multi.length; i < len; i++) {
25046                 this.appendChild(multi[i]);
25047             }
25048         }else{
25049             if(this.fireEvent("beforeappend", this.ownerTree, this, node) === false){
25050                 return false;
25051             }
25052             var index = this.childNodes.length;
25053             var oldParent = node.parentNode;
25054             // it's a move, make sure we move it cleanly
25055             if(oldParent){
25056                 if(node.fireEvent("beforemove", node.getOwnerTree(), node, oldParent, this, index) === false){
25057                     return false;
25058                 }
25059                 oldParent.removeChild(node);
25060             }
25061             
25062             index = this.childNodes.length;
25063             if(index == 0){
25064                 this.setFirstChild(node);
25065             }
25066             this.childNodes.push(node);
25067             node.parentNode = this;
25068             var ps = this.childNodes[index-1];
25069             if(ps){
25070                 node.previousSibling = ps;
25071                 ps.nextSibling = node;
25072             }else{
25073                 node.previousSibling = null;
25074             }
25075             node.nextSibling = null;
25076             this.setLastChild(node);
25077             node.setOwnerTree(this.getOwnerTree());
25078             this.fireEvent("append", this.ownerTree, this, node, index);
25079             if(this.ownerTree) {
25080                 this.ownerTree.fireEvent("appendnode", this, node, index);
25081             }
25082             if(oldParent){
25083                 node.fireEvent("move", this.ownerTree, node, oldParent, this, index);
25084             }
25085             return node;
25086         }
25087     },
25088
25089     /**
25090      * Removes a child node from this node.
25091      * @param {Node} node The node to remove
25092      * @return {Node} The removed node
25093      */
25094     removeChild : function(node){
25095         var index = this.childNodes.indexOf(node);
25096         if(index == -1){
25097             return false;
25098         }
25099         if(this.fireEvent("beforeremove", this.ownerTree, this, node) === false){
25100             return false;
25101         }
25102
25103         // remove it from childNodes collection
25104         this.childNodes.splice(index, 1);
25105
25106         // update siblings
25107         if(node.previousSibling){
25108             node.previousSibling.nextSibling = node.nextSibling;
25109         }
25110         if(node.nextSibling){
25111             node.nextSibling.previousSibling = node.previousSibling;
25112         }
25113
25114         // update child refs
25115         if(this.firstChild == node){
25116             this.setFirstChild(node.nextSibling);
25117         }
25118         if(this.lastChild == node){
25119             this.setLastChild(node.previousSibling);
25120         }
25121
25122         node.setOwnerTree(null);
25123         // clear any references from the node
25124         node.parentNode = null;
25125         node.previousSibling = null;
25126         node.nextSibling = null;
25127         this.fireEvent("remove", this.ownerTree, this, node);
25128         return node;
25129     },
25130
25131     /**
25132      * Inserts the first node before the second node in this nodes childNodes collection.
25133      * @param {Node} node The node to insert
25134      * @param {Node} refNode The node to insert before (if null the node is appended)
25135      * @return {Node} The inserted node
25136      */
25137     insertBefore : function(node, refNode){
25138         if(!refNode){ // like standard Dom, refNode can be null for append
25139             return this.appendChild(node);
25140         }
25141         // nothing to do
25142         if(node == refNode){
25143             return false;
25144         }
25145
25146         if(this.fireEvent("beforeinsert", this.ownerTree, this, node, refNode) === false){
25147             return false;
25148         }
25149         var index = this.childNodes.indexOf(refNode);
25150         var oldParent = node.parentNode;
25151         var refIndex = index;
25152
25153         // when moving internally, indexes will change after remove
25154         if(oldParent == this && this.childNodes.indexOf(node) < index){
25155             refIndex--;
25156         }
25157
25158         // it's a move, make sure we move it cleanly
25159         if(oldParent){
25160             if(node.fireEvent("beforemove", node.getOwnerTree(), node, oldParent, this, index, refNode) === false){
25161                 return false;
25162             }
25163             oldParent.removeChild(node);
25164         }
25165         if(refIndex == 0){
25166             this.setFirstChild(node);
25167         }
25168         this.childNodes.splice(refIndex, 0, node);
25169         node.parentNode = this;
25170         var ps = this.childNodes[refIndex-1];
25171         if(ps){
25172             node.previousSibling = ps;
25173             ps.nextSibling = node;
25174         }else{
25175             node.previousSibling = null;
25176         }
25177         node.nextSibling = refNode;
25178         refNode.previousSibling = node;
25179         node.setOwnerTree(this.getOwnerTree());
25180         this.fireEvent("insert", this.ownerTree, this, node, refNode);
25181         if(oldParent){
25182             node.fireEvent("move", this.ownerTree, node, oldParent, this, refIndex, refNode);
25183         }
25184         return node;
25185     },
25186
25187     /**
25188      * Returns the child node at the specified index.
25189      * @param {Number} index
25190      * @return {Node}
25191      */
25192     item : function(index){
25193         return this.childNodes[index];
25194     },
25195
25196     /**
25197      * Replaces one child node in this node with another.
25198      * @param {Node} newChild The replacement node
25199      * @param {Node} oldChild The node to replace
25200      * @return {Node} The replaced node
25201      */
25202     replaceChild : function(newChild, oldChild){
25203         this.insertBefore(newChild, oldChild);
25204         this.removeChild(oldChild);
25205         return oldChild;
25206     },
25207
25208     /**
25209      * Returns the index of a child node
25210      * @param {Node} node
25211      * @return {Number} The index of the node or -1 if it was not found
25212      */
25213     indexOf : function(child){
25214         return this.childNodes.indexOf(child);
25215     },
25216
25217     /**
25218      * Returns the tree this node is in.
25219      * @return {Tree}
25220      */
25221     getOwnerTree : function(){
25222         // if it doesn't have one, look for one
25223         if(!this.ownerTree){
25224             var p = this;
25225             while(p){
25226                 if(p.ownerTree){
25227                     this.ownerTree = p.ownerTree;
25228                     break;
25229                 }
25230                 p = p.parentNode;
25231             }
25232         }
25233         return this.ownerTree;
25234     },
25235
25236     /**
25237      * Returns depth of this node (the root node has a depth of 0)
25238      * @return {Number}
25239      */
25240     getDepth : function(){
25241         var depth = 0;
25242         var p = this;
25243         while(p.parentNode){
25244             ++depth;
25245             p = p.parentNode;
25246         }
25247         return depth;
25248     },
25249
25250     // private
25251     setOwnerTree : function(tree){
25252         // if it's move, we need to update everyone
25253         if(tree != this.ownerTree){
25254             if(this.ownerTree){
25255                 this.ownerTree.unregisterNode(this);
25256             }
25257             this.ownerTree = tree;
25258             var cs = this.childNodes;
25259             for(var i = 0, len = cs.length; i < len; i++) {
25260                 cs[i].setOwnerTree(tree);
25261             }
25262             if(tree){
25263                 tree.registerNode(this);
25264             }
25265         }
25266     },
25267
25268     /**
25269      * Returns the path for this node. The path can be used to expand or select this node programmatically.
25270      * @param {String} attr (optional) The attr to use for the path (defaults to the node's id)
25271      * @return {String} The path
25272      */
25273     getPath : function(attr){
25274         attr = attr || "id";
25275         var p = this.parentNode;
25276         var b = [this.attributes[attr]];
25277         while(p){
25278             b.unshift(p.attributes[attr]);
25279             p = p.parentNode;
25280         }
25281         var sep = this.getOwnerTree().pathSeparator;
25282         return sep + b.join(sep);
25283     },
25284
25285     /**
25286      * Bubbles up the tree from this node, calling the specified function with each node. The scope (<i>this</i>) of
25287      * function call will be the scope provided or the current node. The arguments to the function
25288      * will be the args provided or the current node. If the function returns false at any point,
25289      * the bubble is stopped.
25290      * @param {Function} fn The function to call
25291      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25292      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25293      */
25294     bubble : function(fn, scope, args){
25295         var p = this;
25296         while(p){
25297             if(fn.call(scope || p, args || p) === false){
25298                 break;
25299             }
25300             p = p.parentNode;
25301         }
25302     },
25303
25304     /**
25305      * Cascades down the tree from this node, calling the specified function with each node. The scope (<i>this</i>) of
25306      * function call will be the scope provided or the current node. The arguments to the function
25307      * will be the args provided or the current node. If the function returns false at any point,
25308      * the cascade is stopped on that branch.
25309      * @param {Function} fn The function to call
25310      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25311      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25312      */
25313     cascade : function(fn, scope, args){
25314         if(fn.call(scope || this, args || this) !== false){
25315             var cs = this.childNodes;
25316             for(var i = 0, len = cs.length; i < len; i++) {
25317                 cs[i].cascade(fn, scope, args);
25318             }
25319         }
25320     },
25321
25322     /**
25323      * Interates the child nodes of this node, calling the specified function with each node. The scope (<i>this</i>) of
25324      * function call will be the scope provided or the current node. The arguments to the function
25325      * will be the args provided or the current node. If the function returns false at any point,
25326      * the iteration stops.
25327      * @param {Function} fn The function to call
25328      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25329      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25330      */
25331     eachChild : function(fn, scope, args){
25332         var cs = this.childNodes;
25333         for(var i = 0, len = cs.length; i < len; i++) {
25334                 if(fn.call(scope || this, args || cs[i]) === false){
25335                     break;
25336                 }
25337         }
25338     },
25339
25340     /**
25341      * Finds the first child that has the attribute with the specified value.
25342      * @param {String} attribute The attribute name
25343      * @param {Mixed} value The value to search for
25344      * @return {Node} The found child or null if none was found
25345      */
25346     findChild : function(attribute, value){
25347         var cs = this.childNodes;
25348         for(var i = 0, len = cs.length; i < len; i++) {
25349                 if(cs[i].attributes[attribute] == value){
25350                     return cs[i];
25351                 }
25352         }
25353         return null;
25354     },
25355
25356     /**
25357      * Finds the first child by a custom function. The child matches if the function passed
25358      * returns true.
25359      * @param {Function} fn
25360      * @param {Object} scope (optional)
25361      * @return {Node} The found child or null if none was found
25362      */
25363     findChildBy : function(fn, scope){
25364         var cs = this.childNodes;
25365         for(var i = 0, len = cs.length; i < len; i++) {
25366                 if(fn.call(scope||cs[i], cs[i]) === true){
25367                     return cs[i];
25368                 }
25369         }
25370         return null;
25371     },
25372
25373     /**
25374      * Sorts this nodes children using the supplied sort function
25375      * @param {Function} fn
25376      * @param {Object} scope (optional)
25377      */
25378     sort : function(fn, scope){
25379         var cs = this.childNodes;
25380         var len = cs.length;
25381         if(len > 0){
25382             var sortFn = scope ? function(){fn.apply(scope, arguments);} : fn;
25383             cs.sort(sortFn);
25384             for(var i = 0; i < len; i++){
25385                 var n = cs[i];
25386                 n.previousSibling = cs[i-1];
25387                 n.nextSibling = cs[i+1];
25388                 if(i == 0){
25389                     this.setFirstChild(n);
25390                 }
25391                 if(i == len-1){
25392                     this.setLastChild(n);
25393                 }
25394             }
25395         }
25396     },
25397
25398     /**
25399      * Returns true if this node is an ancestor (at any point) of the passed node.
25400      * @param {Node} node
25401      * @return {Boolean}
25402      */
25403     contains : function(node){
25404         return node.isAncestor(this);
25405     },
25406
25407     /**
25408      * Returns true if the passed node is an ancestor (at any point) of this node.
25409      * @param {Node} node
25410      * @return {Boolean}
25411      */
25412     isAncestor : function(node){
25413         var p = this.parentNode;
25414         while(p){
25415             if(p == node){
25416                 return true;
25417             }
25418             p = p.parentNode;
25419         }
25420         return false;
25421     },
25422
25423     toString : function(){
25424         return "[Node"+(this.id?" "+this.id:"")+"]";
25425     }
25426 });/*
25427  * Based on:
25428  * Ext JS Library 1.1.1
25429  * Copyright(c) 2006-2007, Ext JS, LLC.
25430  *
25431  * Originally Released Under LGPL - original licence link has changed is not relivant.
25432  *
25433  * Fork - LGPL
25434  * <script type="text/javascript">
25435  */
25436  (function(){ 
25437 /**
25438  * @class Roo.Layer
25439  * @extends Roo.Element
25440  * An extended {@link Roo.Element} object that supports a shadow and shim, constrain to viewport and
25441  * automatic maintaining of shadow/shim positions.
25442  * @cfg {Boolean} shim False to disable the iframe shim in browsers which need one (defaults to true)
25443  * @cfg {String/Boolean} shadow True to create a shadow element with default class "x-layer-shadow", or
25444  * you can pass a string with a CSS class name. False turns off the shadow.
25445  * @cfg {Object} dh DomHelper object config to create element with (defaults to {tag: "div", cls: "x-layer"}).
25446  * @cfg {Boolean} constrain False to disable constrain to viewport (defaults to true)
25447  * @cfg {String} cls CSS class to add to the element
25448  * @cfg {Number} zindex Starting z-index (defaults to 11000)
25449  * @cfg {Number} shadowOffset Number of pixels to offset the shadow (defaults to 3)
25450  * @constructor
25451  * @param {Object} config An object with config options.
25452  * @param {String/HTMLElement} existingEl (optional) Uses an existing DOM element. If the element is not found it creates it.
25453  */
25454
25455 Roo.Layer = function(config, existingEl){
25456     config = config || {};
25457     var dh = Roo.DomHelper;
25458     var cp = config.parentEl, pel = cp ? Roo.getDom(cp) : document.body;
25459     if(existingEl){
25460         this.dom = Roo.getDom(existingEl);
25461     }
25462     if(!this.dom){
25463         var o = config.dh || {tag: "div", cls: "x-layer"};
25464         this.dom = dh.append(pel, o);
25465     }
25466     if(config.cls){
25467         this.addClass(config.cls);
25468     }
25469     this.constrain = config.constrain !== false;
25470     this.visibilityMode = Roo.Element.VISIBILITY;
25471     if(config.id){
25472         this.id = this.dom.id = config.id;
25473     }else{
25474         this.id = Roo.id(this.dom);
25475     }
25476     this.zindex = config.zindex || this.getZIndex();
25477     this.position("absolute", this.zindex);
25478     if(config.shadow){
25479         this.shadowOffset = config.shadowOffset || 4;
25480         this.shadow = new Roo.Shadow({
25481             offset : this.shadowOffset,
25482             mode : config.shadow
25483         });
25484     }else{
25485         this.shadowOffset = 0;
25486     }
25487     this.useShim = config.shim !== false && Roo.useShims;
25488     this.useDisplay = config.useDisplay;
25489     this.hide();
25490 };
25491
25492 var supr = Roo.Element.prototype;
25493
25494 // shims are shared among layer to keep from having 100 iframes
25495 var shims = [];
25496
25497 Roo.extend(Roo.Layer, Roo.Element, {
25498
25499     getZIndex : function(){
25500         return this.zindex || parseInt(this.getStyle("z-index"), 10) || 11000;
25501     },
25502
25503     getShim : function(){
25504         if(!this.useShim){
25505             return null;
25506         }
25507         if(this.shim){
25508             return this.shim;
25509         }
25510         var shim = shims.shift();
25511         if(!shim){
25512             shim = this.createShim();
25513             shim.enableDisplayMode('block');
25514             shim.dom.style.display = 'none';
25515             shim.dom.style.visibility = 'visible';
25516         }
25517         var pn = this.dom.parentNode;
25518         if(shim.dom.parentNode != pn){
25519             pn.insertBefore(shim.dom, this.dom);
25520         }
25521         shim.setStyle('z-index', this.getZIndex()-2);
25522         this.shim = shim;
25523         return shim;
25524     },
25525
25526     hideShim : function(){
25527         if(this.shim){
25528             this.shim.setDisplayed(false);
25529             shims.push(this.shim);
25530             delete this.shim;
25531         }
25532     },
25533
25534     disableShadow : function(){
25535         if(this.shadow){
25536             this.shadowDisabled = true;
25537             this.shadow.hide();
25538             this.lastShadowOffset = this.shadowOffset;
25539             this.shadowOffset = 0;
25540         }
25541     },
25542
25543     enableShadow : function(show){
25544         if(this.shadow){
25545             this.shadowDisabled = false;
25546             this.shadowOffset = this.lastShadowOffset;
25547             delete this.lastShadowOffset;
25548             if(show){
25549                 this.sync(true);
25550             }
25551         }
25552     },
25553
25554     // private
25555     // this code can execute repeatedly in milliseconds (i.e. during a drag) so
25556     // code size was sacrificed for effeciency (e.g. no getBox/setBox, no XY calls)
25557     sync : function(doShow){
25558         var sw = this.shadow;
25559         if(!this.updating && this.isVisible() && (sw || this.useShim)){
25560             var sh = this.getShim();
25561
25562             var w = this.getWidth(),
25563                 h = this.getHeight();
25564
25565             var l = this.getLeft(true),
25566                 t = this.getTop(true);
25567
25568             if(sw && !this.shadowDisabled){
25569                 if(doShow && !sw.isVisible()){
25570                     sw.show(this);
25571                 }else{
25572                     sw.realign(l, t, w, h);
25573                 }
25574                 if(sh){
25575                     if(doShow){
25576                        sh.show();
25577                     }
25578                     // fit the shim behind the shadow, so it is shimmed too
25579                     var a = sw.adjusts, s = sh.dom.style;
25580                     s.left = (Math.min(l, l+a.l))+"px";
25581                     s.top = (Math.min(t, t+a.t))+"px";
25582                     s.width = (w+a.w)+"px";
25583                     s.height = (h+a.h)+"px";
25584                 }
25585             }else if(sh){
25586                 if(doShow){
25587                    sh.show();
25588                 }
25589                 sh.setSize(w, h);
25590                 sh.setLeftTop(l, t);
25591             }
25592             
25593         }
25594     },
25595
25596     // private
25597     destroy : function(){
25598         this.hideShim();
25599         if(this.shadow){
25600             this.shadow.hide();
25601         }
25602         this.removeAllListeners();
25603         var pn = this.dom.parentNode;
25604         if(pn){
25605             pn.removeChild(this.dom);
25606         }
25607         Roo.Element.uncache(this.id);
25608     },
25609
25610     remove : function(){
25611         this.destroy();
25612     },
25613
25614     // private
25615     beginUpdate : function(){
25616         this.updating = true;
25617     },
25618
25619     // private
25620     endUpdate : function(){
25621         this.updating = false;
25622         this.sync(true);
25623     },
25624
25625     // private
25626     hideUnders : function(negOffset){
25627         if(this.shadow){
25628             this.shadow.hide();
25629         }
25630         this.hideShim();
25631     },
25632
25633     // private
25634     constrainXY : function(){
25635         if(this.constrain){
25636             var vw = Roo.lib.Dom.getViewWidth(),
25637                 vh = Roo.lib.Dom.getViewHeight();
25638             var s = Roo.get(document).getScroll();
25639
25640             var xy = this.getXY();
25641             var x = xy[0], y = xy[1];   
25642             var w = this.dom.offsetWidth+this.shadowOffset, h = this.dom.offsetHeight+this.shadowOffset;
25643             // only move it if it needs it
25644             var moved = false;
25645             // first validate right/bottom
25646             if((x + w) > vw+s.left){
25647                 x = vw - w - this.shadowOffset;
25648                 moved = true;
25649             }
25650             if((y + h) > vh+s.top){
25651                 y = vh - h - this.shadowOffset;
25652                 moved = true;
25653             }
25654             // then make sure top/left isn't negative
25655             if(x < s.left){
25656                 x = s.left;
25657                 moved = true;
25658             }
25659             if(y < s.top){
25660                 y = s.top;
25661                 moved = true;
25662             }
25663             if(moved){
25664                 if(this.avoidY){
25665                     var ay = this.avoidY;
25666                     if(y <= ay && (y+h) >= ay){
25667                         y = ay-h-5;   
25668                     }
25669                 }
25670                 xy = [x, y];
25671                 this.storeXY(xy);
25672                 supr.setXY.call(this, xy);
25673                 this.sync();
25674             }
25675         }
25676     },
25677
25678     isVisible : function(){
25679         return this.visible;    
25680     },
25681
25682     // private
25683     showAction : function(){
25684         this.visible = true; // track visibility to prevent getStyle calls
25685         if(this.useDisplay === true){
25686             this.setDisplayed("");
25687         }else if(this.lastXY){
25688             supr.setXY.call(this, this.lastXY);
25689         }else if(this.lastLT){
25690             supr.setLeftTop.call(this, this.lastLT[0], this.lastLT[1]);
25691         }
25692     },
25693
25694     // private
25695     hideAction : function(){
25696         this.visible = false;
25697         if(this.useDisplay === true){
25698             this.setDisplayed(false);
25699         }else{
25700             this.setLeftTop(-10000,-10000);
25701         }
25702     },
25703
25704     // overridden Element method
25705     setVisible : function(v, a, d, c, e){
25706         if(v){
25707             this.showAction();
25708         }
25709         if(a && v){
25710             var cb = function(){
25711                 this.sync(true);
25712                 if(c){
25713                     c();
25714                 }
25715             }.createDelegate(this);
25716             supr.setVisible.call(this, true, true, d, cb, e);
25717         }else{
25718             if(!v){
25719                 this.hideUnders(true);
25720             }
25721             var cb = c;
25722             if(a){
25723                 cb = function(){
25724                     this.hideAction();
25725                     if(c){
25726                         c();
25727                     }
25728                 }.createDelegate(this);
25729             }
25730             supr.setVisible.call(this, v, a, d, cb, e);
25731             if(v){
25732                 this.sync(true);
25733             }else if(!a){
25734                 this.hideAction();
25735             }
25736         }
25737     },
25738
25739     storeXY : function(xy){
25740         delete this.lastLT;
25741         this.lastXY = xy;
25742     },
25743
25744     storeLeftTop : function(left, top){
25745         delete this.lastXY;
25746         this.lastLT = [left, top];
25747     },
25748
25749     // private
25750     beforeFx : function(){
25751         this.beforeAction();
25752         return Roo.Layer.superclass.beforeFx.apply(this, arguments);
25753     },
25754
25755     // private
25756     afterFx : function(){
25757         Roo.Layer.superclass.afterFx.apply(this, arguments);
25758         this.sync(this.isVisible());
25759     },
25760
25761     // private
25762     beforeAction : function(){
25763         if(!this.updating && this.shadow){
25764             this.shadow.hide();
25765         }
25766     },
25767
25768     // overridden Element method
25769     setLeft : function(left){
25770         this.storeLeftTop(left, this.getTop(true));
25771         supr.setLeft.apply(this, arguments);
25772         this.sync();
25773     },
25774
25775     setTop : function(top){
25776         this.storeLeftTop(this.getLeft(true), top);
25777         supr.setTop.apply(this, arguments);
25778         this.sync();
25779     },
25780
25781     setLeftTop : function(left, top){
25782         this.storeLeftTop(left, top);
25783         supr.setLeftTop.apply(this, arguments);
25784         this.sync();
25785     },
25786
25787     setXY : function(xy, a, d, c, e){
25788         this.fixDisplay();
25789         this.beforeAction();
25790         this.storeXY(xy);
25791         var cb = this.createCB(c);
25792         supr.setXY.call(this, xy, a, d, cb, e);
25793         if(!a){
25794             cb();
25795         }
25796     },
25797
25798     // private
25799     createCB : function(c){
25800         var el = this;
25801         return function(){
25802             el.constrainXY();
25803             el.sync(true);
25804             if(c){
25805                 c();
25806             }
25807         };
25808     },
25809
25810     // overridden Element method
25811     setX : function(x, a, d, c, e){
25812         this.setXY([x, this.getY()], a, d, c, e);
25813     },
25814
25815     // overridden Element method
25816     setY : function(y, a, d, c, e){
25817         this.setXY([this.getX(), y], a, d, c, e);
25818     },
25819
25820     // overridden Element method
25821     setSize : function(w, h, a, d, c, e){
25822         this.beforeAction();
25823         var cb = this.createCB(c);
25824         supr.setSize.call(this, w, h, a, d, cb, e);
25825         if(!a){
25826             cb();
25827         }
25828     },
25829
25830     // overridden Element method
25831     setWidth : function(w, a, d, c, e){
25832         this.beforeAction();
25833         var cb = this.createCB(c);
25834         supr.setWidth.call(this, w, a, d, cb, e);
25835         if(!a){
25836             cb();
25837         }
25838     },
25839
25840     // overridden Element method
25841     setHeight : function(h, a, d, c, e){
25842         this.beforeAction();
25843         var cb = this.createCB(c);
25844         supr.setHeight.call(this, h, a, d, cb, e);
25845         if(!a){
25846             cb();
25847         }
25848     },
25849
25850     // overridden Element method
25851     setBounds : function(x, y, w, h, a, d, c, e){
25852         this.beforeAction();
25853         var cb = this.createCB(c);
25854         if(!a){
25855             this.storeXY([x, y]);
25856             supr.setXY.call(this, [x, y]);
25857             supr.setSize.call(this, w, h, a, d, cb, e);
25858             cb();
25859         }else{
25860             supr.setBounds.call(this, x, y, w, h, a, d, cb, e);
25861         }
25862         return this;
25863     },
25864     
25865     /**
25866      * Sets the z-index of this layer and adjusts any shadow and shim z-indexes. The layer z-index is automatically
25867      * incremented by two more than the value passed in so that it always shows above any shadow or shim (the shadow
25868      * element, if any, will be assigned z-index + 1, and the shim element, if any, will be assigned the unmodified z-index).
25869      * @param {Number} zindex The new z-index to set
25870      * @return {this} The Layer
25871      */
25872     setZIndex : function(zindex){
25873         this.zindex = zindex;
25874         this.setStyle("z-index", zindex + 2);
25875         if(this.shadow){
25876             this.shadow.setZIndex(zindex + 1);
25877         }
25878         if(this.shim){
25879             this.shim.setStyle("z-index", zindex);
25880         }
25881     }
25882 });
25883 })();/*
25884  * Based on:
25885  * Ext JS Library 1.1.1
25886  * Copyright(c) 2006-2007, Ext JS, LLC.
25887  *
25888  * Originally Released Under LGPL - original licence link has changed is not relivant.
25889  *
25890  * Fork - LGPL
25891  * <script type="text/javascript">
25892  */
25893
25894
25895 /**
25896  * @class Roo.Shadow
25897  * Simple class that can provide a shadow effect for any element.  Note that the element MUST be absolutely positioned,
25898  * and the shadow does not provide any shimming.  This should be used only in simple cases -- for more advanced
25899  * functionality that can also provide the same shadow effect, see the {@link Roo.Layer} class.
25900  * @constructor
25901  * Create a new Shadow
25902  * @param {Object} config The config object
25903  */
25904 Roo.Shadow = function(config){
25905     Roo.apply(this, config);
25906     if(typeof this.mode != "string"){
25907         this.mode = this.defaultMode;
25908     }
25909     var o = this.offset, a = {h: 0};
25910     var rad = Math.floor(this.offset/2);
25911     switch(this.mode.toLowerCase()){ // all this hideous nonsense calculates the various offsets for shadows
25912         case "drop":
25913             a.w = 0;
25914             a.l = a.t = o;
25915             a.t -= 1;
25916             if(Roo.isIE){
25917                 a.l -= this.offset + rad;
25918                 a.t -= this.offset + rad;
25919                 a.w -= rad;
25920                 a.h -= rad;
25921                 a.t += 1;
25922             }
25923         break;
25924         case "sides":
25925             a.w = (o*2);
25926             a.l = -o;
25927             a.t = o-1;
25928             if(Roo.isIE){
25929                 a.l -= (this.offset - rad);
25930                 a.t -= this.offset + rad;
25931                 a.l += 1;
25932                 a.w -= (this.offset - rad)*2;
25933                 a.w -= rad + 1;
25934                 a.h -= 1;
25935             }
25936         break;
25937         case "frame":
25938             a.w = a.h = (o*2);
25939             a.l = a.t = -o;
25940             a.t += 1;
25941             a.h -= 2;
25942             if(Roo.isIE){
25943                 a.l -= (this.offset - rad);
25944                 a.t -= (this.offset - rad);
25945                 a.l += 1;
25946                 a.w -= (this.offset + rad + 1);
25947                 a.h -= (this.offset + rad);
25948                 a.h += 1;
25949             }
25950         break;
25951     };
25952
25953     this.adjusts = a;
25954 };
25955
25956 Roo.Shadow.prototype = {
25957     /**
25958      * @cfg {String} mode
25959      * The shadow display mode.  Supports the following options:<br />
25960      * sides: Shadow displays on both sides and bottom only<br />
25961      * frame: Shadow displays equally on all four sides<br />
25962      * drop: Traditional bottom-right drop shadow (default)
25963      */
25964     /**
25965      * @cfg {String} offset
25966      * The number of pixels to offset the shadow from the element (defaults to 4)
25967      */
25968     offset: 4,
25969
25970     // private
25971     defaultMode: "drop",
25972
25973     /**
25974      * Displays the shadow under the target element
25975      * @param {String/HTMLElement/Element} targetEl The id or element under which the shadow should display
25976      */
25977     show : function(target){
25978         target = Roo.get(target);
25979         if(!this.el){
25980             this.el = Roo.Shadow.Pool.pull();
25981             if(this.el.dom.nextSibling != target.dom){
25982                 this.el.insertBefore(target);
25983             }
25984         }
25985         this.el.setStyle("z-index", this.zIndex || parseInt(target.getStyle("z-index"), 10)-1);
25986         if(Roo.isIE){
25987             this.el.dom.style.filter="progid:DXImageTransform.Microsoft.alpha(opacity=50) progid:DXImageTransform.Microsoft.Blur(pixelradius="+(this.offset)+")";
25988         }
25989         this.realign(
25990             target.getLeft(true),
25991             target.getTop(true),
25992             target.getWidth(),
25993             target.getHeight()
25994         );
25995         this.el.dom.style.display = "block";
25996     },
25997
25998     /**
25999      * Returns true if the shadow is visible, else false
26000      */
26001     isVisible : function(){
26002         return this.el ? true : false;  
26003     },
26004
26005     /**
26006      * Direct alignment when values are already available. Show must be called at least once before
26007      * calling this method to ensure it is initialized.
26008      * @param {Number} left The target element left position
26009      * @param {Number} top The target element top position
26010      * @param {Number} width The target element width
26011      * @param {Number} height The target element height
26012      */
26013     realign : function(l, t, w, h){
26014         if(!this.el){
26015             return;
26016         }
26017         var a = this.adjusts, d = this.el.dom, s = d.style;
26018         var iea = 0;
26019         s.left = (l+a.l)+"px";
26020         s.top = (t+a.t)+"px";
26021         var sw = (w+a.w), sh = (h+a.h), sws = sw +"px", shs = sh + "px";
26022  
26023         if(s.width != sws || s.height != shs){
26024             s.width = sws;
26025             s.height = shs;
26026             if(!Roo.isIE){
26027                 var cn = d.childNodes;
26028                 var sww = Math.max(0, (sw-12))+"px";
26029                 cn[0].childNodes[1].style.width = sww;
26030                 cn[1].childNodes[1].style.width = sww;
26031                 cn[2].childNodes[1].style.width = sww;
26032                 cn[1].style.height = Math.max(0, (sh-12))+"px";
26033             }
26034         }
26035     },
26036
26037     /**
26038      * Hides this shadow
26039      */
26040     hide : function(){
26041         if(this.el){
26042             this.el.dom.style.display = "none";
26043             Roo.Shadow.Pool.push(this.el);
26044             delete this.el;
26045         }
26046     },
26047
26048     /**
26049      * Adjust the z-index of this shadow
26050      * @param {Number} zindex The new z-index
26051      */
26052     setZIndex : function(z){
26053         this.zIndex = z;
26054         if(this.el){
26055             this.el.setStyle("z-index", z);
26056         }
26057     }
26058 };
26059
26060 // Private utility class that manages the internal Shadow cache
26061 Roo.Shadow.Pool = function(){
26062     var p = [];
26063     var markup = Roo.isIE ?
26064                  '<div class="x-ie-shadow"></div>' :
26065                  '<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>';
26066     return {
26067         pull : function(){
26068             var sh = p.shift();
26069             if(!sh){
26070                 sh = Roo.get(Roo.DomHelper.insertHtml("beforeBegin", document.body.firstChild, markup));
26071                 sh.autoBoxAdjust = false;
26072             }
26073             return sh;
26074         },
26075
26076         push : function(sh){
26077             p.push(sh);
26078         }
26079     };
26080 }();/*
26081  * Based on:
26082  * Ext JS Library 1.1.1
26083  * Copyright(c) 2006-2007, Ext JS, LLC.
26084  *
26085  * Originally Released Under LGPL - original licence link has changed is not relivant.
26086  *
26087  * Fork - LGPL
26088  * <script type="text/javascript">
26089  */
26090
26091
26092 /**
26093  * @class Roo.SplitBar
26094  * @extends Roo.util.Observable
26095  * Creates draggable splitter bar functionality from two elements (element to be dragged and element to be resized).
26096  * <br><br>
26097  * Usage:
26098  * <pre><code>
26099 var split = new Roo.SplitBar("elementToDrag", "elementToSize",
26100                    Roo.SplitBar.HORIZONTAL, Roo.SplitBar.LEFT);
26101 split.setAdapter(new Roo.SplitBar.AbsoluteLayoutAdapter("container"));
26102 split.minSize = 100;
26103 split.maxSize = 600;
26104 split.animate = true;
26105 split.on('moved', splitterMoved);
26106 </code></pre>
26107  * @constructor
26108  * Create a new SplitBar
26109  * @param {String/HTMLElement/Roo.Element} dragElement The element to be dragged and act as the SplitBar. 
26110  * @param {String/HTMLElement/Roo.Element} resizingElement The element to be resized based on where the SplitBar element is dragged 
26111  * @param {Number} orientation (optional) Either Roo.SplitBar.HORIZONTAL or Roo.SplitBar.VERTICAL. (Defaults to HORIZONTAL)
26112  * @param {Number} placement (optional) Either Roo.SplitBar.LEFT or Roo.SplitBar.RIGHT for horizontal or  
26113                         Roo.SplitBar.TOP or Roo.SplitBar.BOTTOM for vertical. (By default, this is determined automatically by the initial
26114                         position of the SplitBar).
26115  */
26116 Roo.SplitBar = function(dragElement, resizingElement, orientation, placement, existingProxy){
26117     
26118     /** @private */
26119     this.el = Roo.get(dragElement, true);
26120     this.el.dom.unselectable = "on";
26121     /** @private */
26122     this.resizingEl = Roo.get(resizingElement, true);
26123
26124     /**
26125      * @private
26126      * The orientation of the split. Either Roo.SplitBar.HORIZONTAL or Roo.SplitBar.VERTICAL. (Defaults to HORIZONTAL)
26127      * Note: If this is changed after creating the SplitBar, the placement property must be manually updated
26128      * @type Number
26129      */
26130     this.orientation = orientation || Roo.SplitBar.HORIZONTAL;
26131     
26132     /**
26133      * The minimum size of the resizing element. (Defaults to 0)
26134      * @type Number
26135      */
26136     this.minSize = 0;
26137     
26138     /**
26139      * The maximum size of the resizing element. (Defaults to 2000)
26140      * @type Number
26141      */
26142     this.maxSize = 2000;
26143     
26144     /**
26145      * Whether to animate the transition to the new size
26146      * @type Boolean
26147      */
26148     this.animate = false;
26149     
26150     /**
26151      * Whether to create a transparent shim that overlays the page when dragging, enables dragging across iframes.
26152      * @type Boolean
26153      */
26154     this.useShim = false;
26155     
26156     /** @private */
26157     this.shim = null;
26158     
26159     if(!existingProxy){
26160         /** @private */
26161         this.proxy = Roo.SplitBar.createProxy(this.orientation);
26162     }else{
26163         this.proxy = Roo.get(existingProxy).dom;
26164     }
26165     /** @private */
26166     this.dd = new Roo.dd.DDProxy(this.el.dom.id, "XSplitBars", {dragElId : this.proxy.id});
26167     
26168     /** @private */
26169     this.dd.b4StartDrag = this.onStartProxyDrag.createDelegate(this);
26170     
26171     /** @private */
26172     this.dd.endDrag = this.onEndProxyDrag.createDelegate(this);
26173     
26174     /** @private */
26175     this.dragSpecs = {};
26176     
26177     /**
26178      * @private The adapter to use to positon and resize elements
26179      */
26180     this.adapter = new Roo.SplitBar.BasicLayoutAdapter();
26181     this.adapter.init(this);
26182     
26183     if(this.orientation == Roo.SplitBar.HORIZONTAL){
26184         /** @private */
26185         this.placement = placement || (this.el.getX() > this.resizingEl.getX() ? Roo.SplitBar.LEFT : Roo.SplitBar.RIGHT);
26186         this.el.addClass("x-splitbar-h");
26187     }else{
26188         /** @private */
26189         this.placement = placement || (this.el.getY() > this.resizingEl.getY() ? Roo.SplitBar.TOP : Roo.SplitBar.BOTTOM);
26190         this.el.addClass("x-splitbar-v");
26191     }
26192     
26193     this.addEvents({
26194         /**
26195          * @event resize
26196          * Fires when the splitter is moved (alias for {@link #event-moved})
26197          * @param {Roo.SplitBar} this
26198          * @param {Number} newSize the new width or height
26199          */
26200         "resize" : true,
26201         /**
26202          * @event moved
26203          * Fires when the splitter is moved
26204          * @param {Roo.SplitBar} this
26205          * @param {Number} newSize the new width or height
26206          */
26207         "moved" : true,
26208         /**
26209          * @event beforeresize
26210          * Fires before the splitter is dragged
26211          * @param {Roo.SplitBar} this
26212          */
26213         "beforeresize" : true,
26214
26215         "beforeapply" : true
26216     });
26217
26218     Roo.util.Observable.call(this);
26219 };
26220
26221 Roo.extend(Roo.SplitBar, Roo.util.Observable, {
26222     onStartProxyDrag : function(x, y){
26223         this.fireEvent("beforeresize", this);
26224         if(!this.overlay){
26225             var o = Roo.DomHelper.insertFirst(document.body,  {cls: "x-drag-overlay", html: "&#160;"}, true);
26226             o.unselectable();
26227             o.enableDisplayMode("block");
26228             // all splitbars share the same overlay
26229             Roo.SplitBar.prototype.overlay = o;
26230         }
26231         this.overlay.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
26232         this.overlay.show();
26233         Roo.get(this.proxy).setDisplayed("block");
26234         var size = this.adapter.getElementSize(this);
26235         this.activeMinSize = this.getMinimumSize();;
26236         this.activeMaxSize = this.getMaximumSize();;
26237         var c1 = size - this.activeMinSize;
26238         var c2 = Math.max(this.activeMaxSize - size, 0);
26239         if(this.orientation == Roo.SplitBar.HORIZONTAL){
26240             this.dd.resetConstraints();
26241             this.dd.setXConstraint(
26242                 this.placement == Roo.SplitBar.LEFT ? c1 : c2, 
26243                 this.placement == Roo.SplitBar.LEFT ? c2 : c1
26244             );
26245             this.dd.setYConstraint(0, 0);
26246         }else{
26247             this.dd.resetConstraints();
26248             this.dd.setXConstraint(0, 0);
26249             this.dd.setYConstraint(
26250                 this.placement == Roo.SplitBar.TOP ? c1 : c2, 
26251                 this.placement == Roo.SplitBar.TOP ? c2 : c1
26252             );
26253          }
26254         this.dragSpecs.startSize = size;
26255         this.dragSpecs.startPoint = [x, y];
26256         Roo.dd.DDProxy.prototype.b4StartDrag.call(this.dd, x, y);
26257     },
26258     
26259     /** 
26260      * @private Called after the drag operation by the DDProxy
26261      */
26262     onEndProxyDrag : function(e){
26263         Roo.get(this.proxy).setDisplayed(false);
26264         var endPoint = Roo.lib.Event.getXY(e);
26265         if(this.overlay){
26266             this.overlay.hide();
26267         }
26268         var newSize;
26269         if(this.orientation == Roo.SplitBar.HORIZONTAL){
26270             newSize = this.dragSpecs.startSize + 
26271                 (this.placement == Roo.SplitBar.LEFT ?
26272                     endPoint[0] - this.dragSpecs.startPoint[0] :
26273                     this.dragSpecs.startPoint[0] - endPoint[0]
26274                 );
26275         }else{
26276             newSize = this.dragSpecs.startSize + 
26277                 (this.placement == Roo.SplitBar.TOP ?
26278                     endPoint[1] - this.dragSpecs.startPoint[1] :
26279                     this.dragSpecs.startPoint[1] - endPoint[1]
26280                 );
26281         }
26282         newSize = Math.min(Math.max(newSize, this.activeMinSize), this.activeMaxSize);
26283         if(newSize != this.dragSpecs.startSize){
26284             if(this.fireEvent('beforeapply', this, newSize) !== false){
26285                 this.adapter.setElementSize(this, newSize);
26286                 this.fireEvent("moved", this, newSize);
26287                 this.fireEvent("resize", this, newSize);
26288             }
26289         }
26290     },
26291     
26292     /**
26293      * Get the adapter this SplitBar uses
26294      * @return The adapter object
26295      */
26296     getAdapter : function(){
26297         return this.adapter;
26298     },
26299     
26300     /**
26301      * Set the adapter this SplitBar uses
26302      * @param {Object} adapter A SplitBar adapter object
26303      */
26304     setAdapter : function(adapter){
26305         this.adapter = adapter;
26306         this.adapter.init(this);
26307     },
26308     
26309     /**
26310      * Gets the minimum size for the resizing element
26311      * @return {Number} The minimum size
26312      */
26313     getMinimumSize : function(){
26314         return this.minSize;
26315     },
26316     
26317     /**
26318      * Sets the minimum size for the resizing element
26319      * @param {Number} minSize The minimum size
26320      */
26321     setMinimumSize : function(minSize){
26322         this.minSize = minSize;
26323     },
26324     
26325     /**
26326      * Gets the maximum size for the resizing element
26327      * @return {Number} The maximum size
26328      */
26329     getMaximumSize : function(){
26330         return this.maxSize;
26331     },
26332     
26333     /**
26334      * Sets the maximum size for the resizing element
26335      * @param {Number} maxSize The maximum size
26336      */
26337     setMaximumSize : function(maxSize){
26338         this.maxSize = maxSize;
26339     },
26340     
26341     /**
26342      * Sets the initialize size for the resizing element
26343      * @param {Number} size The initial size
26344      */
26345     setCurrentSize : function(size){
26346         var oldAnimate = this.animate;
26347         this.animate = false;
26348         this.adapter.setElementSize(this, size);
26349         this.animate = oldAnimate;
26350     },
26351     
26352     /**
26353      * Destroy this splitbar. 
26354      * @param {Boolean} removeEl True to remove the element
26355      */
26356     destroy : function(removeEl){
26357         if(this.shim){
26358             this.shim.remove();
26359         }
26360         this.dd.unreg();
26361         this.proxy.parentNode.removeChild(this.proxy);
26362         if(removeEl){
26363             this.el.remove();
26364         }
26365     }
26366 });
26367
26368 /**
26369  * @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.
26370  */
26371 Roo.SplitBar.createProxy = function(dir){
26372     var proxy = new Roo.Element(document.createElement("div"));
26373     proxy.unselectable();
26374     var cls = 'x-splitbar-proxy';
26375     proxy.addClass(cls + ' ' + (dir == Roo.SplitBar.HORIZONTAL ? cls +'-h' : cls + '-v'));
26376     document.body.appendChild(proxy.dom);
26377     return proxy.dom;
26378 };
26379
26380 /** 
26381  * @class Roo.SplitBar.BasicLayoutAdapter
26382  * Default Adapter. It assumes the splitter and resizing element are not positioned
26383  * elements and only gets/sets the width of the element. Generally used for table based layouts.
26384  */
26385 Roo.SplitBar.BasicLayoutAdapter = function(){
26386 };
26387
26388 Roo.SplitBar.BasicLayoutAdapter.prototype = {
26389     // do nothing for now
26390     init : function(s){
26391     
26392     },
26393     /**
26394      * Called before drag operations to get the current size of the resizing element. 
26395      * @param {Roo.SplitBar} s The SplitBar using this adapter
26396      */
26397      getElementSize : function(s){
26398         if(s.orientation == Roo.SplitBar.HORIZONTAL){
26399             return s.resizingEl.getWidth();
26400         }else{
26401             return s.resizingEl.getHeight();
26402         }
26403     },
26404     
26405     /**
26406      * Called after drag operations to set the size of the resizing element.
26407      * @param {Roo.SplitBar} s The SplitBar using this adapter
26408      * @param {Number} newSize The new size to set
26409      * @param {Function} onComplete A function to be invoked when resizing is complete
26410      */
26411     setElementSize : function(s, newSize, onComplete){
26412         if(s.orientation == Roo.SplitBar.HORIZONTAL){
26413             if(!s.animate){
26414                 s.resizingEl.setWidth(newSize);
26415                 if(onComplete){
26416                     onComplete(s, newSize);
26417                 }
26418             }else{
26419                 s.resizingEl.setWidth(newSize, true, .1, onComplete, 'easeOut');
26420             }
26421         }else{
26422             
26423             if(!s.animate){
26424                 s.resizingEl.setHeight(newSize);
26425                 if(onComplete){
26426                     onComplete(s, newSize);
26427                 }
26428             }else{
26429                 s.resizingEl.setHeight(newSize, true, .1, onComplete, 'easeOut');
26430             }
26431         }
26432     }
26433 };
26434
26435 /** 
26436  *@class Roo.SplitBar.AbsoluteLayoutAdapter
26437  * @extends Roo.SplitBar.BasicLayoutAdapter
26438  * Adapter that  moves the splitter element to align with the resized sizing element. 
26439  * Used with an absolute positioned SplitBar.
26440  * @param {String/HTMLElement/Roo.Element} container The container that wraps around the absolute positioned content. If it's
26441  * document.body, make sure you assign an id to the body element.
26442  */
26443 Roo.SplitBar.AbsoluteLayoutAdapter = function(container){
26444     this.basic = new Roo.SplitBar.BasicLayoutAdapter();
26445     this.container = Roo.get(container);
26446 };
26447
26448 Roo.SplitBar.AbsoluteLayoutAdapter.prototype = {
26449     init : function(s){
26450         this.basic.init(s);
26451     },
26452     
26453     getElementSize : function(s){
26454         return this.basic.getElementSize(s);
26455     },
26456     
26457     setElementSize : function(s, newSize, onComplete){
26458         this.basic.setElementSize(s, newSize, this.moveSplitter.createDelegate(this, [s]));
26459     },
26460     
26461     moveSplitter : function(s){
26462         var yes = Roo.SplitBar;
26463         switch(s.placement){
26464             case yes.LEFT:
26465                 s.el.setX(s.resizingEl.getRight());
26466                 break;
26467             case yes.RIGHT:
26468                 s.el.setStyle("right", (this.container.getWidth() - s.resizingEl.getLeft()) + "px");
26469                 break;
26470             case yes.TOP:
26471                 s.el.setY(s.resizingEl.getBottom());
26472                 break;
26473             case yes.BOTTOM:
26474                 s.el.setY(s.resizingEl.getTop() - s.el.getHeight());
26475                 break;
26476         }
26477     }
26478 };
26479
26480 /**
26481  * Orientation constant - Create a vertical SplitBar
26482  * @static
26483  * @type Number
26484  */
26485 Roo.SplitBar.VERTICAL = 1;
26486
26487 /**
26488  * Orientation constant - Create a horizontal SplitBar
26489  * @static
26490  * @type Number
26491  */
26492 Roo.SplitBar.HORIZONTAL = 2;
26493
26494 /**
26495  * Placement constant - The resizing element is to the left of the splitter element
26496  * @static
26497  * @type Number
26498  */
26499 Roo.SplitBar.LEFT = 1;
26500
26501 /**
26502  * Placement constant - The resizing element is to the right of the splitter element
26503  * @static
26504  * @type Number
26505  */
26506 Roo.SplitBar.RIGHT = 2;
26507
26508 /**
26509  * Placement constant - The resizing element is positioned above the splitter element
26510  * @static
26511  * @type Number
26512  */
26513 Roo.SplitBar.TOP = 3;
26514
26515 /**
26516  * Placement constant - The resizing element is positioned under splitter element
26517  * @static
26518  * @type Number
26519  */
26520 Roo.SplitBar.BOTTOM = 4;
26521 /*
26522  * Based on:
26523  * Ext JS Library 1.1.1
26524  * Copyright(c) 2006-2007, Ext JS, LLC.
26525  *
26526  * Originally Released Under LGPL - original licence link has changed is not relivant.
26527  *
26528  * Fork - LGPL
26529  * <script type="text/javascript">
26530  */
26531
26532 /**
26533  * @class Roo.View
26534  * @extends Roo.util.Observable
26535  * Create a "View" for an element based on a data model or UpdateManager and the supplied DomHelper template. 
26536  * This class also supports single and multi selection modes. <br>
26537  * Create a data model bound view:
26538  <pre><code>
26539  var store = new Roo.data.Store(...);
26540
26541  var view = new Roo.View({
26542     el : "my-element",
26543     tpl : '&lt;div id="{0}"&gt;{2} - {1}&lt;/div&gt;', // auto create template
26544  
26545     singleSelect: true,
26546     selectedClass: "ydataview-selected",
26547     store: store
26548  });
26549
26550  // listen for node click?
26551  view.on("click", function(vw, index, node, e){
26552  alert('Node "' + node.id + '" at index: ' + index + " was clicked.");
26553  });
26554
26555  // load XML data
26556  dataModel.load("foobar.xml");
26557  </code></pre>
26558  For an example of creating a JSON/UpdateManager view, see {@link Roo.JsonView}.
26559  * <br><br>
26560  * <b>Note: The root of your template must be a single node. Table/row implementations may work but are not supported due to
26561  * IE"s limited insertion support with tables and Opera"s faulty event bubbling.</b>
26562  * 
26563  * Note: old style constructor is still suported (container, template, config)
26564  * 
26565  * @constructor
26566  * Create a new View
26567  * @param {Object} config The config object
26568  * 
26569  */
26570 Roo.View = function(config, depreciated_tpl, depreciated_config){
26571     
26572     this.parent = false;
26573     
26574     if (typeof(depreciated_tpl) == 'undefined') {
26575         // new way.. - universal constructor.
26576         Roo.apply(this, config);
26577         this.el  = Roo.get(this.el);
26578     } else {
26579         // old format..
26580         this.el  = Roo.get(config);
26581         this.tpl = depreciated_tpl;
26582         Roo.apply(this, depreciated_config);
26583     }
26584     this.wrapEl  = this.el.wrap().wrap();
26585     ///this.el = this.wrapEla.appendChild(document.createElement("div"));
26586     
26587     
26588     if(typeof(this.tpl) == "string"){
26589         this.tpl = new Roo.Template(this.tpl);
26590     } else {
26591         // support xtype ctors..
26592         this.tpl = new Roo.factory(this.tpl, Roo);
26593     }
26594     
26595     
26596     this.tpl.compile();
26597     
26598     /** @private */
26599     this.addEvents({
26600         /**
26601          * @event beforeclick
26602          * Fires before a click is processed. Returns false to cancel the default action.
26603          * @param {Roo.View} this
26604          * @param {Number} index The index of the target node
26605          * @param {HTMLElement} node The target node
26606          * @param {Roo.EventObject} e The raw event object
26607          */
26608             "beforeclick" : true,
26609         /**
26610          * @event click
26611          * Fires when a template node is clicked.
26612          * @param {Roo.View} this
26613          * @param {Number} index The index of the target node
26614          * @param {HTMLElement} node The target node
26615          * @param {Roo.EventObject} e The raw event object
26616          */
26617             "click" : true,
26618         /**
26619          * @event dblclick
26620          * Fires when a template node is double clicked.
26621          * @param {Roo.View} this
26622          * @param {Number} index The index of the target node
26623          * @param {HTMLElement} node The target node
26624          * @param {Roo.EventObject} e The raw event object
26625          */
26626             "dblclick" : true,
26627         /**
26628          * @event contextmenu
26629          * Fires when a template node is right clicked.
26630          * @param {Roo.View} this
26631          * @param {Number} index The index of the target node
26632          * @param {HTMLElement} node The target node
26633          * @param {Roo.EventObject} e The raw event object
26634          */
26635             "contextmenu" : true,
26636         /**
26637          * @event selectionchange
26638          * Fires when the selected nodes change.
26639          * @param {Roo.View} this
26640          * @param {Array} selections Array of the selected nodes
26641          */
26642             "selectionchange" : true,
26643     
26644         /**
26645          * @event beforeselect
26646          * Fires before a selection is made. If any handlers return false, the selection is cancelled.
26647          * @param {Roo.View} this
26648          * @param {HTMLElement} node The node to be selected
26649          * @param {Array} selections Array of currently selected nodes
26650          */
26651             "beforeselect" : true,
26652         /**
26653          * @event preparedata
26654          * Fires on every row to render, to allow you to change the data.
26655          * @param {Roo.View} this
26656          * @param {Object} data to be rendered (change this)
26657          */
26658           "preparedata" : true
26659           
26660           
26661         });
26662
26663
26664
26665     this.el.on({
26666         "click": this.onClick,
26667         "dblclick": this.onDblClick,
26668         "contextmenu": this.onContextMenu,
26669         scope:this
26670     });
26671
26672     this.selections = [];
26673     this.nodes = [];
26674     this.cmp = new Roo.CompositeElementLite([]);
26675     if(this.store){
26676         this.store = Roo.factory(this.store, Roo.data);
26677         this.setStore(this.store, true);
26678     }
26679     
26680     if ( this.footer && this.footer.xtype) {
26681            
26682          var fctr = this.wrapEl.appendChild(document.createElement("div"));
26683         
26684         this.footer.dataSource = this.store;
26685         this.footer.container = fctr;
26686         this.footer = Roo.factory(this.footer, Roo);
26687         fctr.insertFirst(this.el);
26688         
26689         // this is a bit insane - as the paging toolbar seems to detach the el..
26690 //        dom.parentNode.parentNode.parentNode
26691          // they get detached?
26692     }
26693     
26694     
26695     Roo.View.superclass.constructor.call(this);
26696     
26697     
26698 };
26699
26700 Roo.extend(Roo.View, Roo.util.Observable, {
26701     
26702      /**
26703      * @cfg {Roo.data.Store} store Data store to load data from.
26704      */
26705     store : false,
26706     
26707     /**
26708      * @cfg {String|Roo.Element} el The container element.
26709      */
26710     el : '',
26711     
26712     /**
26713      * @cfg {String|Roo.Template} tpl The template used by this View 
26714      */
26715     tpl : false,
26716     /**
26717      * @cfg {String} dataName the named area of the template to use as the data area
26718      *                          Works with domtemplates roo-name="name"
26719      */
26720     dataName: false,
26721     /**
26722      * @cfg {String} selectedClass The css class to add to selected nodes
26723      */
26724     selectedClass : "x-view-selected",
26725      /**
26726      * @cfg {String} emptyText The empty text to show when nothing is loaded.
26727      */
26728     emptyText : "",
26729     
26730     /**
26731      * @cfg {String} text to display on mask (default Loading)
26732      */
26733     mask : false,
26734     /**
26735      * @cfg {Boolean} multiSelect Allow multiple selection
26736      */
26737     multiSelect : false,
26738     /**
26739      * @cfg {Boolean} singleSelect Allow single selection
26740      */
26741     singleSelect:  false,
26742     
26743     /**
26744      * @cfg {Boolean} toggleSelect - selecting 
26745      */
26746     toggleSelect : false,
26747     
26748     /**
26749      * @cfg {Boolean} tickable - selecting 
26750      */
26751     tickable : false,
26752     
26753     /**
26754      * Returns the element this view is bound to.
26755      * @return {Roo.Element}
26756      */
26757     getEl : function(){
26758         return this.wrapEl;
26759     },
26760     
26761     
26762
26763     /**
26764      * Refreshes the view. - called by datachanged on the store. - do not call directly.
26765      */
26766     refresh : function(){
26767         //Roo.log('refresh');
26768         var t = this.tpl;
26769         
26770         // if we are using something like 'domtemplate', then
26771         // the what gets used is:
26772         // t.applySubtemplate(NAME, data, wrapping data..)
26773         // the outer template then get' applied with
26774         //     the store 'extra data'
26775         // and the body get's added to the
26776         //      roo-name="data" node?
26777         //      <span class='roo-tpl-{name}'></span> ?????
26778         
26779         
26780         
26781         this.clearSelections();
26782         this.el.update("");
26783         var html = [];
26784         var records = this.store.getRange();
26785         if(records.length < 1) {
26786             
26787             // is this valid??  = should it render a template??
26788             
26789             this.el.update(this.emptyText);
26790             return;
26791         }
26792         var el = this.el;
26793         if (this.dataName) {
26794             this.el.update(t.apply(this.store.meta)); //????
26795             el = this.el.child('.roo-tpl-' + this.dataName);
26796         }
26797         
26798         for(var i = 0, len = records.length; i < len; i++){
26799             var data = this.prepareData(records[i].data, i, records[i]);
26800             this.fireEvent("preparedata", this, data, i, records[i]);
26801             
26802             var d = Roo.apply({}, data);
26803             
26804             if(this.tickable){
26805                 Roo.apply(d, {'roo-id' : Roo.id()});
26806                 
26807                 var _this = this;
26808             
26809                 Roo.each(this.parent.item, function(item){
26810                     if(item[_this.parent.valueField] != data[_this.parent.valueField]){
26811                         return;
26812                     }
26813                     Roo.apply(d, {'roo-data-checked' : 'checked'});
26814                 });
26815             }
26816             
26817             html[html.length] = Roo.util.Format.trim(
26818                 this.dataName ?
26819                     t.applySubtemplate(this.dataName, d, this.store.meta) :
26820                     t.apply(d)
26821             );
26822         }
26823         
26824         
26825         
26826         el.update(html.join(""));
26827         this.nodes = el.dom.childNodes;
26828         this.updateIndexes(0);
26829     },
26830     
26831
26832     /**
26833      * Function to override to reformat the data that is sent to
26834      * the template for each node.
26835      * DEPRICATED - use the preparedata event handler.
26836      * @param {Array/Object} data The raw data (array of colData for a data model bound view or
26837      * a JSON object for an UpdateManager bound view).
26838      */
26839     prepareData : function(data, index, record)
26840     {
26841         this.fireEvent("preparedata", this, data, index, record);
26842         return data;
26843     },
26844
26845     onUpdate : function(ds, record){
26846         // Roo.log('on update');   
26847         this.clearSelections();
26848         var index = this.store.indexOf(record);
26849         var n = this.nodes[index];
26850         this.tpl.insertBefore(n, this.prepareData(record.data, index, record));
26851         n.parentNode.removeChild(n);
26852         this.updateIndexes(index, index);
26853     },
26854
26855     
26856     
26857 // --------- FIXME     
26858     onAdd : function(ds, records, index)
26859     {
26860         //Roo.log(['on Add', ds, records, index] );        
26861         this.clearSelections();
26862         if(this.nodes.length == 0){
26863             this.refresh();
26864             return;
26865         }
26866         var n = this.nodes[index];
26867         for(var i = 0, len = records.length; i < len; i++){
26868             var d = this.prepareData(records[i].data, i, records[i]);
26869             if(n){
26870                 this.tpl.insertBefore(n, d);
26871             }else{
26872                 
26873                 this.tpl.append(this.el, d);
26874             }
26875         }
26876         this.updateIndexes(index);
26877     },
26878
26879     onRemove : function(ds, record, index){
26880        // Roo.log('onRemove');
26881         this.clearSelections();
26882         var el = this.dataName  ?
26883             this.el.child('.roo-tpl-' + this.dataName) :
26884             this.el; 
26885         
26886         el.dom.removeChild(this.nodes[index]);
26887         this.updateIndexes(index);
26888     },
26889
26890     /**
26891      * Refresh an individual node.
26892      * @param {Number} index
26893      */
26894     refreshNode : function(index){
26895         this.onUpdate(this.store, this.store.getAt(index));
26896     },
26897
26898     updateIndexes : function(startIndex, endIndex){
26899         var ns = this.nodes;
26900         startIndex = startIndex || 0;
26901         endIndex = endIndex || ns.length - 1;
26902         for(var i = startIndex; i <= endIndex; i++){
26903             ns[i].nodeIndex = i;
26904         }
26905     },
26906
26907     /**
26908      * Changes the data store this view uses and refresh the view.
26909      * @param {Store} store
26910      */
26911     setStore : function(store, initial){
26912         if(!initial && this.store){
26913             this.store.un("datachanged", this.refresh);
26914             this.store.un("add", this.onAdd);
26915             this.store.un("remove", this.onRemove);
26916             this.store.un("update", this.onUpdate);
26917             this.store.un("clear", this.refresh);
26918             this.store.un("beforeload", this.onBeforeLoad);
26919             this.store.un("load", this.onLoad);
26920             this.store.un("loadexception", this.onLoad);
26921         }
26922         if(store){
26923           
26924             store.on("datachanged", this.refresh, this);
26925             store.on("add", this.onAdd, this);
26926             store.on("remove", this.onRemove, this);
26927             store.on("update", this.onUpdate, this);
26928             store.on("clear", this.refresh, this);
26929             store.on("beforeload", this.onBeforeLoad, this);
26930             store.on("load", this.onLoad, this);
26931             store.on("loadexception", this.onLoad, this);
26932         }
26933         
26934         if(store){
26935             this.refresh();
26936         }
26937     },
26938     /**
26939      * onbeforeLoad - masks the loading area.
26940      *
26941      */
26942     onBeforeLoad : function(store,opts)
26943     {
26944          //Roo.log('onBeforeLoad');   
26945         if (!opts.add) {
26946             this.el.update("");
26947         }
26948         this.el.mask(this.mask ? this.mask : "Loading" ); 
26949     },
26950     onLoad : function ()
26951     {
26952         this.el.unmask();
26953     },
26954     
26955
26956     /**
26957      * Returns the template node the passed child belongs to or null if it doesn't belong to one.
26958      * @param {HTMLElement} node
26959      * @return {HTMLElement} The template node
26960      */
26961     findItemFromChild : function(node){
26962         var el = this.dataName  ?
26963             this.el.child('.roo-tpl-' + this.dataName,true) :
26964             this.el.dom; 
26965         
26966         if(!node || node.parentNode == el){
26967                     return node;
26968             }
26969             var p = node.parentNode;
26970             while(p && p != el){
26971             if(p.parentNode == el){
26972                 return p;
26973             }
26974             p = p.parentNode;
26975         }
26976             return null;
26977     },
26978
26979     /** @ignore */
26980     onClick : function(e){
26981         var item = this.findItemFromChild(e.getTarget());
26982         if(item){
26983             var index = this.indexOf(item);
26984             if(this.onItemClick(item, index, e) !== false){
26985                 this.fireEvent("click", this, index, item, e);
26986             }
26987         }else{
26988             this.clearSelections();
26989         }
26990     },
26991
26992     /** @ignore */
26993     onContextMenu : function(e){
26994         var item = this.findItemFromChild(e.getTarget());
26995         if(item){
26996             this.fireEvent("contextmenu", this, this.indexOf(item), item, e);
26997         }
26998     },
26999
27000     /** @ignore */
27001     onDblClick : function(e){
27002         var item = this.findItemFromChild(e.getTarget());
27003         if(item){
27004             this.fireEvent("dblclick", this, this.indexOf(item), item, e);
27005         }
27006     },
27007
27008     onItemClick : function(item, index, e)
27009     {
27010         if(this.fireEvent("beforeclick", this, index, item, e) === false){
27011             return false;
27012         }
27013         if (this.toggleSelect) {
27014             var m = this.isSelected(item) ? 'unselect' : 'select';
27015             //Roo.log(m);
27016             var _t = this;
27017             _t[m](item, true, false);
27018             return true;
27019         }
27020         if(this.multiSelect || this.singleSelect){
27021             if(this.multiSelect && e.shiftKey && this.lastSelection){
27022                 this.select(this.getNodes(this.indexOf(this.lastSelection), index), false);
27023             }else{
27024                 this.select(item, this.multiSelect && e.ctrlKey);
27025                 this.lastSelection = item;
27026             }
27027             
27028             if(!this.tickable){
27029                 e.preventDefault();
27030             }
27031             
27032         }
27033         return true;
27034     },
27035
27036     /**
27037      * Get the number of selected nodes.
27038      * @return {Number}
27039      */
27040     getSelectionCount : function(){
27041         return this.selections.length;
27042     },
27043
27044     /**
27045      * Get the currently selected nodes.
27046      * @return {Array} An array of HTMLElements
27047      */
27048     getSelectedNodes : function(){
27049         return this.selections;
27050     },
27051
27052     /**
27053      * Get the indexes of the selected nodes.
27054      * @return {Array}
27055      */
27056     getSelectedIndexes : function(){
27057         var indexes = [], s = this.selections;
27058         for(var i = 0, len = s.length; i < len; i++){
27059             indexes.push(s[i].nodeIndex);
27060         }
27061         return indexes;
27062     },
27063
27064     /**
27065      * Clear all selections
27066      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange event
27067      */
27068     clearSelections : function(suppressEvent){
27069         if(this.nodes && (this.multiSelect || this.singleSelect) && this.selections.length > 0){
27070             this.cmp.elements = this.selections;
27071             this.cmp.removeClass(this.selectedClass);
27072             this.selections = [];
27073             if(!suppressEvent){
27074                 this.fireEvent("selectionchange", this, this.selections);
27075             }
27076         }
27077     },
27078
27079     /**
27080      * Returns true if the passed node is selected
27081      * @param {HTMLElement/Number} node The node or node index
27082      * @return {Boolean}
27083      */
27084     isSelected : function(node){
27085         var s = this.selections;
27086         if(s.length < 1){
27087             return false;
27088         }
27089         node = this.getNode(node);
27090         return s.indexOf(node) !== -1;
27091     },
27092
27093     /**
27094      * Selects nodes.
27095      * @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
27096      * @param {Boolean} keepExisting (optional) true to keep existing selections
27097      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange vent
27098      */
27099     select : function(nodeInfo, keepExisting, suppressEvent){
27100         if(nodeInfo instanceof Array){
27101             if(!keepExisting){
27102                 this.clearSelections(true);
27103             }
27104             for(var i = 0, len = nodeInfo.length; i < len; i++){
27105                 this.select(nodeInfo[i], true, true);
27106             }
27107             return;
27108         } 
27109         var node = this.getNode(nodeInfo);
27110         if(!node || this.isSelected(node)){
27111             return; // already selected.
27112         }
27113         if(!keepExisting){
27114             this.clearSelections(true);
27115         }
27116         
27117         if(this.fireEvent("beforeselect", this, node, this.selections) !== false){
27118             Roo.fly(node).addClass(this.selectedClass);
27119             this.selections.push(node);
27120             if(!suppressEvent){
27121                 this.fireEvent("selectionchange", this, this.selections);
27122             }
27123         }
27124         
27125         
27126     },
27127       /**
27128      * Unselects nodes.
27129      * @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
27130      * @param {Boolean} keepExisting (optional) true IGNORED (for campatibility with select)
27131      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange vent
27132      */
27133     unselect : function(nodeInfo, keepExisting, suppressEvent)
27134     {
27135         if(nodeInfo instanceof Array){
27136             Roo.each(this.selections, function(s) {
27137                 this.unselect(s, nodeInfo);
27138             }, this);
27139             return;
27140         }
27141         var node = this.getNode(nodeInfo);
27142         if(!node || !this.isSelected(node)){
27143             //Roo.log("not selected");
27144             return; // not selected.
27145         }
27146         // fireevent???
27147         var ns = [];
27148         Roo.each(this.selections, function(s) {
27149             if (s == node ) {
27150                 Roo.fly(node).removeClass(this.selectedClass);
27151
27152                 return;
27153             }
27154             ns.push(s);
27155         },this);
27156         
27157         this.selections= ns;
27158         this.fireEvent("selectionchange", this, this.selections);
27159     },
27160
27161     /**
27162      * Gets a template node.
27163      * @param {HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node or the id of a template node
27164      * @return {HTMLElement} The node or null if it wasn't found
27165      */
27166     getNode : function(nodeInfo){
27167         if(typeof nodeInfo == "string"){
27168             return document.getElementById(nodeInfo);
27169         }else if(typeof nodeInfo == "number"){
27170             return this.nodes[nodeInfo];
27171         }
27172         return nodeInfo;
27173     },
27174
27175     /**
27176      * Gets a range template nodes.
27177      * @param {Number} startIndex
27178      * @param {Number} endIndex
27179      * @return {Array} An array of nodes
27180      */
27181     getNodes : function(start, end){
27182         var ns = this.nodes;
27183         start = start || 0;
27184         end = typeof end == "undefined" ? ns.length - 1 : end;
27185         var nodes = [];
27186         if(start <= end){
27187             for(var i = start; i <= end; i++){
27188                 nodes.push(ns[i]);
27189             }
27190         } else{
27191             for(var i = start; i >= end; i--){
27192                 nodes.push(ns[i]);
27193             }
27194         }
27195         return nodes;
27196     },
27197
27198     /**
27199      * Finds the index of the passed node
27200      * @param {HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node or the id of a template node
27201      * @return {Number} The index of the node or -1
27202      */
27203     indexOf : function(node){
27204         node = this.getNode(node);
27205         if(typeof node.nodeIndex == "number"){
27206             return node.nodeIndex;
27207         }
27208         var ns = this.nodes;
27209         for(var i = 0, len = ns.length; i < len; i++){
27210             if(ns[i] == node){
27211                 return i;
27212             }
27213         }
27214         return -1;
27215     }
27216 });
27217 /*
27218  * Based on:
27219  * Ext JS Library 1.1.1
27220  * Copyright(c) 2006-2007, Ext JS, LLC.
27221  *
27222  * Originally Released Under LGPL - original licence link has changed is not relivant.
27223  *
27224  * Fork - LGPL
27225  * <script type="text/javascript">
27226  */
27227
27228 /**
27229  * @class Roo.JsonView
27230  * @extends Roo.View
27231  * Shortcut class to create a JSON + {@link Roo.UpdateManager} template view. Usage:
27232 <pre><code>
27233 var view = new Roo.JsonView({
27234     container: "my-element",
27235     tpl: '&lt;div id="{id}"&gt;{foo} - {bar}&lt;/div&gt;', // auto create template
27236     multiSelect: true, 
27237     jsonRoot: "data" 
27238 });
27239
27240 // listen for node click?
27241 view.on("click", function(vw, index, node, e){
27242     alert('Node "' + node.id + '" at index: ' + index + " was clicked.");
27243 });
27244
27245 // direct load of JSON data
27246 view.load("foobar.php");
27247
27248 // Example from my blog list
27249 var tpl = new Roo.Template(
27250     '&lt;div class="entry"&gt;' +
27251     '&lt;a class="entry-title" href="{link}"&gt;{title}&lt;/a&gt;' +
27252     "&lt;h4&gt;{date} by {author} | {comments} Comments&lt;/h4&gt;{description}" +
27253     "&lt;/div&gt;&lt;hr /&gt;"
27254 );
27255
27256 var moreView = new Roo.JsonView({
27257     container :  "entry-list", 
27258     template : tpl,
27259     jsonRoot: "posts"
27260 });
27261 moreView.on("beforerender", this.sortEntries, this);
27262 moreView.load({
27263     url: "/blog/get-posts.php",
27264     params: "allposts=true",
27265     text: "Loading Blog Entries..."
27266 });
27267 </code></pre>
27268
27269 * Note: old code is supported with arguments : (container, template, config)
27270
27271
27272  * @constructor
27273  * Create a new JsonView
27274  * 
27275  * @param {Object} config The config object
27276  * 
27277  */
27278 Roo.JsonView = function(config, depreciated_tpl, depreciated_config){
27279     
27280     
27281     Roo.JsonView.superclass.constructor.call(this, config, depreciated_tpl, depreciated_config);
27282
27283     var um = this.el.getUpdateManager();
27284     um.setRenderer(this);
27285     um.on("update", this.onLoad, this);
27286     um.on("failure", this.onLoadException, this);
27287
27288     /**
27289      * @event beforerender
27290      * Fires before rendering of the downloaded JSON data.
27291      * @param {Roo.JsonView} this
27292      * @param {Object} data The JSON data loaded
27293      */
27294     /**
27295      * @event load
27296      * Fires when data is loaded.
27297      * @param {Roo.JsonView} this
27298      * @param {Object} data The JSON data loaded
27299      * @param {Object} response The raw Connect response object
27300      */
27301     /**
27302      * @event loadexception
27303      * Fires when loading fails.
27304      * @param {Roo.JsonView} this
27305      * @param {Object} response The raw Connect response object
27306      */
27307     this.addEvents({
27308         'beforerender' : true,
27309         'load' : true,
27310         'loadexception' : true
27311     });
27312 };
27313 Roo.extend(Roo.JsonView, Roo.View, {
27314     /**
27315      * @type {String} The root property in the loaded JSON object that contains the data
27316      */
27317     jsonRoot : "",
27318
27319     /**
27320      * Refreshes the view.
27321      */
27322     refresh : function(){
27323         this.clearSelections();
27324         this.el.update("");
27325         var html = [];
27326         var o = this.jsonData;
27327         if(o && o.length > 0){
27328             for(var i = 0, len = o.length; i < len; i++){
27329                 var data = this.prepareData(o[i], i, o);
27330                 html[html.length] = this.tpl.apply(data);
27331             }
27332         }else{
27333             html.push(this.emptyText);
27334         }
27335         this.el.update(html.join(""));
27336         this.nodes = this.el.dom.childNodes;
27337         this.updateIndexes(0);
27338     },
27339
27340     /**
27341      * 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.
27342      * @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:
27343      <pre><code>
27344      view.load({
27345          url: "your-url.php",
27346          params: {param1: "foo", param2: "bar"}, // or a URL encoded string
27347          callback: yourFunction,
27348          scope: yourObject, //(optional scope)
27349          discardUrl: false,
27350          nocache: false,
27351          text: "Loading...",
27352          timeout: 30,
27353          scripts: false
27354      });
27355      </code></pre>
27356      * The only required property is <i>url</i>. The optional properties <i>nocache</i>, <i>text</i> and <i>scripts</i>
27357      * 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.
27358      * @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}
27359      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
27360      * @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.
27361      */
27362     load : function(){
27363         var um = this.el.getUpdateManager();
27364         um.update.apply(um, arguments);
27365     },
27366
27367     // note - render is a standard framework call...
27368     // using it for the response is really flaky... - it's called by UpdateManager normally, except when called by the XComponent/addXtype.
27369     render : function(el, response){
27370         
27371         this.clearSelections();
27372         this.el.update("");
27373         var o;
27374         try{
27375             if (response != '') {
27376                 o = Roo.util.JSON.decode(response.responseText);
27377                 if(this.jsonRoot){
27378                     
27379                     o = o[this.jsonRoot];
27380                 }
27381             }
27382         } catch(e){
27383         }
27384         /**
27385          * The current JSON data or null
27386          */
27387         this.jsonData = o;
27388         this.beforeRender();
27389         this.refresh();
27390     },
27391
27392 /**
27393  * Get the number of records in the current JSON dataset
27394  * @return {Number}
27395  */
27396     getCount : function(){
27397         return this.jsonData ? this.jsonData.length : 0;
27398     },
27399
27400 /**
27401  * Returns the JSON object for the specified node(s)
27402  * @param {HTMLElement/Array} node The node or an array of nodes
27403  * @return {Object/Array} If you pass in an array, you get an array back, otherwise
27404  * you get the JSON object for the node
27405  */
27406     getNodeData : function(node){
27407         if(node instanceof Array){
27408             var data = [];
27409             for(var i = 0, len = node.length; i < len; i++){
27410                 data.push(this.getNodeData(node[i]));
27411             }
27412             return data;
27413         }
27414         return this.jsonData[this.indexOf(node)] || null;
27415     },
27416
27417     beforeRender : function(){
27418         this.snapshot = this.jsonData;
27419         if(this.sortInfo){
27420             this.sort.apply(this, this.sortInfo);
27421         }
27422         this.fireEvent("beforerender", this, this.jsonData);
27423     },
27424
27425     onLoad : function(el, o){
27426         this.fireEvent("load", this, this.jsonData, o);
27427     },
27428
27429     onLoadException : function(el, o){
27430         this.fireEvent("loadexception", this, o);
27431     },
27432
27433 /**
27434  * Filter the data by a specific property.
27435  * @param {String} property A property on your JSON objects
27436  * @param {String/RegExp} value Either string that the property values
27437  * should start with, or a RegExp to test against the property
27438  */
27439     filter : function(property, value){
27440         if(this.jsonData){
27441             var data = [];
27442             var ss = this.snapshot;
27443             if(typeof value == "string"){
27444                 var vlen = value.length;
27445                 if(vlen == 0){
27446                     this.clearFilter();
27447                     return;
27448                 }
27449                 value = value.toLowerCase();
27450                 for(var i = 0, len = ss.length; i < len; i++){
27451                     var o = ss[i];
27452                     if(o[property].substr(0, vlen).toLowerCase() == value){
27453                         data.push(o);
27454                     }
27455                 }
27456             } else if(value.exec){ // regex?
27457                 for(var i = 0, len = ss.length; i < len; i++){
27458                     var o = ss[i];
27459                     if(value.test(o[property])){
27460                         data.push(o);
27461                     }
27462                 }
27463             } else{
27464                 return;
27465             }
27466             this.jsonData = data;
27467             this.refresh();
27468         }
27469     },
27470
27471 /**
27472  * Filter by a function. The passed function will be called with each
27473  * object in the current dataset. If the function returns true the value is kept,
27474  * otherwise it is filtered.
27475  * @param {Function} fn
27476  * @param {Object} scope (optional) The scope of the function (defaults to this JsonView)
27477  */
27478     filterBy : function(fn, scope){
27479         if(this.jsonData){
27480             var data = [];
27481             var ss = this.snapshot;
27482             for(var i = 0, len = ss.length; i < len; i++){
27483                 var o = ss[i];
27484                 if(fn.call(scope || this, o)){
27485                     data.push(o);
27486                 }
27487             }
27488             this.jsonData = data;
27489             this.refresh();
27490         }
27491     },
27492
27493 /**
27494  * Clears the current filter.
27495  */
27496     clearFilter : function(){
27497         if(this.snapshot && this.jsonData != this.snapshot){
27498             this.jsonData = this.snapshot;
27499             this.refresh();
27500         }
27501     },
27502
27503
27504 /**
27505  * Sorts the data for this view and refreshes it.
27506  * @param {String} property A property on your JSON objects to sort on
27507  * @param {String} direction (optional) "desc" or "asc" (defaults to "asc")
27508  * @param {Function} sortType (optional) A function to call to convert the data to a sortable value.
27509  */
27510     sort : function(property, dir, sortType){
27511         this.sortInfo = Array.prototype.slice.call(arguments, 0);
27512         if(this.jsonData){
27513             var p = property;
27514             var dsc = dir && dir.toLowerCase() == "desc";
27515             var f = function(o1, o2){
27516                 var v1 = sortType ? sortType(o1[p]) : o1[p];
27517                 var v2 = sortType ? sortType(o2[p]) : o2[p];
27518                 ;
27519                 if(v1 < v2){
27520                     return dsc ? +1 : -1;
27521                 } else if(v1 > v2){
27522                     return dsc ? -1 : +1;
27523                 } else{
27524                     return 0;
27525                 }
27526             };
27527             this.jsonData.sort(f);
27528             this.refresh();
27529             if(this.jsonData != this.snapshot){
27530                 this.snapshot.sort(f);
27531             }
27532         }
27533     }
27534 });/*
27535  * Based on:
27536  * Ext JS Library 1.1.1
27537  * Copyright(c) 2006-2007, Ext JS, LLC.
27538  *
27539  * Originally Released Under LGPL - original licence link has changed is not relivant.
27540  *
27541  * Fork - LGPL
27542  * <script type="text/javascript">
27543  */
27544  
27545
27546 /**
27547  * @class Roo.ColorPalette
27548  * @extends Roo.Component
27549  * Simple color palette class for choosing colors.  The palette can be rendered to any container.<br />
27550  * Here's an example of typical usage:
27551  * <pre><code>
27552 var cp = new Roo.ColorPalette({value:'993300'});  // initial selected color
27553 cp.render('my-div');
27554
27555 cp.on('select', function(palette, selColor){
27556     // do something with selColor
27557 });
27558 </code></pre>
27559  * @constructor
27560  * Create a new ColorPalette
27561  * @param {Object} config The config object
27562  */
27563 Roo.ColorPalette = function(config){
27564     Roo.ColorPalette.superclass.constructor.call(this, config);
27565     this.addEvents({
27566         /**
27567              * @event select
27568              * Fires when a color is selected
27569              * @param {ColorPalette} this
27570              * @param {String} color The 6-digit color hex code (without the # symbol)
27571              */
27572         select: true
27573     });
27574
27575     if(this.handler){
27576         this.on("select", this.handler, this.scope, true);
27577     }
27578 };
27579 Roo.extend(Roo.ColorPalette, Roo.Component, {
27580     /**
27581      * @cfg {String} itemCls
27582      * The CSS class to apply to the containing element (defaults to "x-color-palette")
27583      */
27584     itemCls : "x-color-palette",
27585     /**
27586      * @cfg {String} value
27587      * The initial color to highlight (should be a valid 6-digit color hex code without the # symbol).  Note that
27588      * the hex codes are case-sensitive.
27589      */
27590     value : null,
27591     clickEvent:'click',
27592     // private
27593     ctype: "Roo.ColorPalette",
27594
27595     /**
27596      * @cfg {Boolean} allowReselect If set to true then reselecting a color that is already selected fires the selection event
27597      */
27598     allowReselect : false,
27599
27600     /**
27601      * <p>An array of 6-digit color hex code strings (without the # symbol).  This array can contain any number
27602      * of colors, and each hex code should be unique.  The width of the palette is controlled via CSS by adjusting
27603      * the width property of the 'x-color-palette' class (or assigning a custom class), so you can balance the number
27604      * of colors with the width setting until the box is symmetrical.</p>
27605      * <p>You can override individual colors if needed:</p>
27606      * <pre><code>
27607 var cp = new Roo.ColorPalette();
27608 cp.colors[0] = "FF0000";  // change the first box to red
27609 </code></pre>
27610
27611 Or you can provide a custom array of your own for complete control:
27612 <pre><code>
27613 var cp = new Roo.ColorPalette();
27614 cp.colors = ["000000", "993300", "333300"];
27615 </code></pre>
27616      * @type Array
27617      */
27618     colors : [
27619         "000000", "993300", "333300", "003300", "003366", "000080", "333399", "333333",
27620         "800000", "FF6600", "808000", "008000", "008080", "0000FF", "666699", "808080",
27621         "FF0000", "FF9900", "99CC00", "339966", "33CCCC", "3366FF", "800080", "969696",
27622         "FF00FF", "FFCC00", "FFFF00", "00FF00", "00FFFF", "00CCFF", "993366", "C0C0C0",
27623         "FF99CC", "FFCC99", "FFFF99", "CCFFCC", "CCFFFF", "99CCFF", "CC99FF", "FFFFFF"
27624     ],
27625
27626     // private
27627     onRender : function(container, position){
27628         var t = new Roo.MasterTemplate(
27629             '<tpl><a href="#" class="color-{0}" hidefocus="on"><em><span style="background:#{0}" unselectable="on">&#160;</span></em></a></tpl>'
27630         );
27631         var c = this.colors;
27632         for(var i = 0, len = c.length; i < len; i++){
27633             t.add([c[i]]);
27634         }
27635         var el = document.createElement("div");
27636         el.className = this.itemCls;
27637         t.overwrite(el);
27638         container.dom.insertBefore(el, position);
27639         this.el = Roo.get(el);
27640         this.el.on(this.clickEvent, this.handleClick,  this, {delegate: "a"});
27641         if(this.clickEvent != 'click'){
27642             this.el.on('click', Roo.emptyFn,  this, {delegate: "a", preventDefault:true});
27643         }
27644     },
27645
27646     // private
27647     afterRender : function(){
27648         Roo.ColorPalette.superclass.afterRender.call(this);
27649         if(this.value){
27650             var s = this.value;
27651             this.value = null;
27652             this.select(s);
27653         }
27654     },
27655
27656     // private
27657     handleClick : function(e, t){
27658         e.preventDefault();
27659         if(!this.disabled){
27660             var c = t.className.match(/(?:^|\s)color-(.{6})(?:\s|$)/)[1];
27661             this.select(c.toUpperCase());
27662         }
27663     },
27664
27665     /**
27666      * Selects the specified color in the palette (fires the select event)
27667      * @param {String} color A valid 6-digit color hex code (# will be stripped if included)
27668      */
27669     select : function(color){
27670         color = color.replace("#", "");
27671         if(color != this.value || this.allowReselect){
27672             var el = this.el;
27673             if(this.value){
27674                 el.child("a.color-"+this.value).removeClass("x-color-palette-sel");
27675             }
27676             el.child("a.color-"+color).addClass("x-color-palette-sel");
27677             this.value = color;
27678             this.fireEvent("select", this, color);
27679         }
27680     }
27681 });/*
27682  * Based on:
27683  * Ext JS Library 1.1.1
27684  * Copyright(c) 2006-2007, Ext JS, LLC.
27685  *
27686  * Originally Released Under LGPL - original licence link has changed is not relivant.
27687  *
27688  * Fork - LGPL
27689  * <script type="text/javascript">
27690  */
27691  
27692 /**
27693  * @class Roo.DatePicker
27694  * @extends Roo.Component
27695  * Simple date picker class.
27696  * @constructor
27697  * Create a new DatePicker
27698  * @param {Object} config The config object
27699  */
27700 Roo.DatePicker = function(config){
27701     Roo.DatePicker.superclass.constructor.call(this, config);
27702
27703     this.value = config && config.value ?
27704                  config.value.clearTime() : new Date().clearTime();
27705
27706     this.addEvents({
27707         /**
27708              * @event select
27709              * Fires when a date is selected
27710              * @param {DatePicker} this
27711              * @param {Date} date The selected date
27712              */
27713         'select': true,
27714         /**
27715              * @event monthchange
27716              * Fires when the displayed month changes 
27717              * @param {DatePicker} this
27718              * @param {Date} date The selected month
27719              */
27720         'monthchange': true
27721     });
27722
27723     if(this.handler){
27724         this.on("select", this.handler,  this.scope || this);
27725     }
27726     // build the disabledDatesRE
27727     if(!this.disabledDatesRE && this.disabledDates){
27728         var dd = this.disabledDates;
27729         var re = "(?:";
27730         for(var i = 0; i < dd.length; i++){
27731             re += dd[i];
27732             if(i != dd.length-1) {
27733                 re += "|";
27734             }
27735         }
27736         this.disabledDatesRE = new RegExp(re + ")");
27737     }
27738 };
27739
27740 Roo.extend(Roo.DatePicker, Roo.Component, {
27741     /**
27742      * @cfg {String} todayText
27743      * The text to display on the button that selects the current date (defaults to "Today")
27744      */
27745     todayText : "Today",
27746     /**
27747      * @cfg {String} okText
27748      * The text to display on the ok button
27749      */
27750     okText : "&#160;OK&#160;", // &#160; to give the user extra clicking room
27751     /**
27752      * @cfg {String} cancelText
27753      * The text to display on the cancel button
27754      */
27755     cancelText : "Cancel",
27756     /**
27757      * @cfg {String} todayTip
27758      * The tooltip to display for the button that selects the current date (defaults to "{current date} (Spacebar)")
27759      */
27760     todayTip : "{0} (Spacebar)",
27761     /**
27762      * @cfg {Date} minDate
27763      * Minimum allowable date (JavaScript date object, defaults to null)
27764      */
27765     minDate : null,
27766     /**
27767      * @cfg {Date} maxDate
27768      * Maximum allowable date (JavaScript date object, defaults to null)
27769      */
27770     maxDate : null,
27771     /**
27772      * @cfg {String} minText
27773      * The error text to display if the minDate validation fails (defaults to "This date is before the minimum date")
27774      */
27775     minText : "This date is before the minimum date",
27776     /**
27777      * @cfg {String} maxText
27778      * The error text to display if the maxDate validation fails (defaults to "This date is after the maximum date")
27779      */
27780     maxText : "This date is after the maximum date",
27781     /**
27782      * @cfg {String} format
27783      * The default date format string which can be overriden for localization support.  The format must be
27784      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
27785      */
27786     format : "m/d/y",
27787     /**
27788      * @cfg {Array} disabledDays
27789      * An array of days to disable, 0-based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
27790      */
27791     disabledDays : null,
27792     /**
27793      * @cfg {String} disabledDaysText
27794      * The tooltip to display when the date falls on a disabled day (defaults to "")
27795      */
27796     disabledDaysText : "",
27797     /**
27798      * @cfg {RegExp} disabledDatesRE
27799      * JavaScript regular expression used to disable a pattern of dates (defaults to null)
27800      */
27801     disabledDatesRE : null,
27802     /**
27803      * @cfg {String} disabledDatesText
27804      * The tooltip text to display when the date falls on a disabled date (defaults to "")
27805      */
27806     disabledDatesText : "",
27807     /**
27808      * @cfg {Boolean} constrainToViewport
27809      * True to constrain the date picker to the viewport (defaults to true)
27810      */
27811     constrainToViewport : true,
27812     /**
27813      * @cfg {Array} monthNames
27814      * An array of textual month names which can be overriden for localization support (defaults to Date.monthNames)
27815      */
27816     monthNames : Date.monthNames,
27817     /**
27818      * @cfg {Array} dayNames
27819      * An array of textual day names which can be overriden for localization support (defaults to Date.dayNames)
27820      */
27821     dayNames : Date.dayNames,
27822     /**
27823      * @cfg {String} nextText
27824      * The next month navigation button tooltip (defaults to 'Next Month (Control+Right)')
27825      */
27826     nextText: 'Next Month (Control+Right)',
27827     /**
27828      * @cfg {String} prevText
27829      * The previous month navigation button tooltip (defaults to 'Previous Month (Control+Left)')
27830      */
27831     prevText: 'Previous Month (Control+Left)',
27832     /**
27833      * @cfg {String} monthYearText
27834      * The header month selector tooltip (defaults to 'Choose a month (Control+Up/Down to move years)')
27835      */
27836     monthYearText: 'Choose a month (Control+Up/Down to move years)',
27837     /**
27838      * @cfg {Number} startDay
27839      * Day index at which the week should begin, 0-based (defaults to 0, which is Sunday)
27840      */
27841     startDay : 0,
27842     /**
27843      * @cfg {Bool} showClear
27844      * Show a clear button (usefull for date form elements that can be blank.)
27845      */
27846     
27847     showClear: false,
27848     
27849     /**
27850      * Sets the value of the date field
27851      * @param {Date} value The date to set
27852      */
27853     setValue : function(value){
27854         var old = this.value;
27855         
27856         if (typeof(value) == 'string') {
27857          
27858             value = Date.parseDate(value, this.format);
27859         }
27860         if (!value) {
27861             value = new Date();
27862         }
27863         
27864         this.value = value.clearTime(true);
27865         if(this.el){
27866             this.update(this.value);
27867         }
27868     },
27869
27870     /**
27871      * Gets the current selected value of the date field
27872      * @return {Date} The selected date
27873      */
27874     getValue : function(){
27875         return this.value;
27876     },
27877
27878     // private
27879     focus : function(){
27880         if(this.el){
27881             this.update(this.activeDate);
27882         }
27883     },
27884
27885     // privateval
27886     onRender : function(container, position){
27887         
27888         var m = [
27889              '<table cellspacing="0">',
27890                 '<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>',
27891                 '<tr><td colspan="3"><table class="x-date-inner" cellspacing="0"><thead><tr>'];
27892         var dn = this.dayNames;
27893         for(var i = 0; i < 7; i++){
27894             var d = this.startDay+i;
27895             if(d > 6){
27896                 d = d-7;
27897             }
27898             m.push("<th><span>", dn[d].substr(0,1), "</span></th>");
27899         }
27900         m[m.length] = "</tr></thead><tbody><tr>";
27901         for(var i = 0; i < 42; i++) {
27902             if(i % 7 == 0 && i != 0){
27903                 m[m.length] = "</tr><tr>";
27904             }
27905             m[m.length] = '<td><a href="#" hidefocus="on" class="x-date-date" tabIndex="1"><em><span></span></em></a></td>';
27906         }
27907         m[m.length] = '</tr></tbody></table></td></tr><tr>'+
27908             '<td colspan="3" class="x-date-bottom" align="center"></td></tr></table><div class="x-date-mp"></div>';
27909
27910         var el = document.createElement("div");
27911         el.className = "x-date-picker";
27912         el.innerHTML = m.join("");
27913
27914         container.dom.insertBefore(el, position);
27915
27916         this.el = Roo.get(el);
27917         this.eventEl = Roo.get(el.firstChild);
27918
27919         new Roo.util.ClickRepeater(this.el.child("td.x-date-left a"), {
27920             handler: this.showPrevMonth,
27921             scope: this,
27922             preventDefault:true,
27923             stopDefault:true
27924         });
27925
27926         new Roo.util.ClickRepeater(this.el.child("td.x-date-right a"), {
27927             handler: this.showNextMonth,
27928             scope: this,
27929             preventDefault:true,
27930             stopDefault:true
27931         });
27932
27933         this.eventEl.on("mousewheel", this.handleMouseWheel,  this);
27934
27935         this.monthPicker = this.el.down('div.x-date-mp');
27936         this.monthPicker.enableDisplayMode('block');
27937         
27938         var kn = new Roo.KeyNav(this.eventEl, {
27939             "left" : function(e){
27940                 e.ctrlKey ?
27941                     this.showPrevMonth() :
27942                     this.update(this.activeDate.add("d", -1));
27943             },
27944
27945             "right" : function(e){
27946                 e.ctrlKey ?
27947                     this.showNextMonth() :
27948                     this.update(this.activeDate.add("d", 1));
27949             },
27950
27951             "up" : function(e){
27952                 e.ctrlKey ?
27953                     this.showNextYear() :
27954                     this.update(this.activeDate.add("d", -7));
27955             },
27956
27957             "down" : function(e){
27958                 e.ctrlKey ?
27959                     this.showPrevYear() :
27960                     this.update(this.activeDate.add("d", 7));
27961             },
27962
27963             "pageUp" : function(e){
27964                 this.showNextMonth();
27965             },
27966
27967             "pageDown" : function(e){
27968                 this.showPrevMonth();
27969             },
27970
27971             "enter" : function(e){
27972                 e.stopPropagation();
27973                 return true;
27974             },
27975
27976             scope : this
27977         });
27978
27979         this.eventEl.on("click", this.handleDateClick,  this, {delegate: "a.x-date-date"});
27980
27981         this.eventEl.addKeyListener(Roo.EventObject.SPACE, this.selectToday,  this);
27982
27983         this.el.unselectable();
27984         
27985         this.cells = this.el.select("table.x-date-inner tbody td");
27986         this.textNodes = this.el.query("table.x-date-inner tbody span");
27987
27988         this.mbtn = new Roo.Button(this.el.child("td.x-date-middle", true), {
27989             text: "&#160;",
27990             tooltip: this.monthYearText
27991         });
27992
27993         this.mbtn.on('click', this.showMonthPicker, this);
27994         this.mbtn.el.child(this.mbtn.menuClassTarget).addClass("x-btn-with-menu");
27995
27996
27997         var today = (new Date()).dateFormat(this.format);
27998         
27999         var baseTb = new Roo.Toolbar(this.el.child("td.x-date-bottom", true));
28000         if (this.showClear) {
28001             baseTb.add( new Roo.Toolbar.Fill());
28002         }
28003         baseTb.add({
28004             text: String.format(this.todayText, today),
28005             tooltip: String.format(this.todayTip, today),
28006             handler: this.selectToday,
28007             scope: this
28008         });
28009         
28010         //var todayBtn = new Roo.Button(this.el.child("td.x-date-bottom", true), {
28011             
28012         //});
28013         if (this.showClear) {
28014             
28015             baseTb.add( new Roo.Toolbar.Fill());
28016             baseTb.add({
28017                 text: '&#160;',
28018                 cls: 'x-btn-icon x-btn-clear',
28019                 handler: function() {
28020                     //this.value = '';
28021                     this.fireEvent("select", this, '');
28022                 },
28023                 scope: this
28024             });
28025         }
28026         
28027         
28028         if(Roo.isIE){
28029             this.el.repaint();
28030         }
28031         this.update(this.value);
28032     },
28033
28034     createMonthPicker : function(){
28035         if(!this.monthPicker.dom.firstChild){
28036             var buf = ['<table border="0" cellspacing="0">'];
28037             for(var i = 0; i < 6; i++){
28038                 buf.push(
28039                     '<tr><td class="x-date-mp-month"><a href="#">', this.monthNames[i].substr(0, 3), '</a></td>',
28040                     '<td class="x-date-mp-month x-date-mp-sep"><a href="#">', this.monthNames[i+6].substr(0, 3), '</a></td>',
28041                     i == 0 ?
28042                     '<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>' :
28043                     '<td class="x-date-mp-year"><a href="#"></a></td><td class="x-date-mp-year"><a href="#"></a></td></tr>'
28044                 );
28045             }
28046             buf.push(
28047                 '<tr class="x-date-mp-btns"><td colspan="4"><button type="button" class="x-date-mp-ok">',
28048                     this.okText,
28049                     '</button><button type="button" class="x-date-mp-cancel">',
28050                     this.cancelText,
28051                     '</button></td></tr>',
28052                 '</table>'
28053             );
28054             this.monthPicker.update(buf.join(''));
28055             this.monthPicker.on('click', this.onMonthClick, this);
28056             this.monthPicker.on('dblclick', this.onMonthDblClick, this);
28057
28058             this.mpMonths = this.monthPicker.select('td.x-date-mp-month');
28059             this.mpYears = this.monthPicker.select('td.x-date-mp-year');
28060
28061             this.mpMonths.each(function(m, a, i){
28062                 i += 1;
28063                 if((i%2) == 0){
28064                     m.dom.xmonth = 5 + Math.round(i * .5);
28065                 }else{
28066                     m.dom.xmonth = Math.round((i-1) * .5);
28067                 }
28068             });
28069         }
28070     },
28071
28072     showMonthPicker : function(){
28073         this.createMonthPicker();
28074         var size = this.el.getSize();
28075         this.monthPicker.setSize(size);
28076         this.monthPicker.child('table').setSize(size);
28077
28078         this.mpSelMonth = (this.activeDate || this.value).getMonth();
28079         this.updateMPMonth(this.mpSelMonth);
28080         this.mpSelYear = (this.activeDate || this.value).getFullYear();
28081         this.updateMPYear(this.mpSelYear);
28082
28083         this.monthPicker.slideIn('t', {duration:.2});
28084     },
28085
28086     updateMPYear : function(y){
28087         this.mpyear = y;
28088         var ys = this.mpYears.elements;
28089         for(var i = 1; i <= 10; i++){
28090             var td = ys[i-1], y2;
28091             if((i%2) == 0){
28092                 y2 = y + Math.round(i * .5);
28093                 td.firstChild.innerHTML = y2;
28094                 td.xyear = y2;
28095             }else{
28096                 y2 = y - (5-Math.round(i * .5));
28097                 td.firstChild.innerHTML = y2;
28098                 td.xyear = y2;
28099             }
28100             this.mpYears.item(i-1)[y2 == this.mpSelYear ? 'addClass' : 'removeClass']('x-date-mp-sel');
28101         }
28102     },
28103
28104     updateMPMonth : function(sm){
28105         this.mpMonths.each(function(m, a, i){
28106             m[m.dom.xmonth == sm ? 'addClass' : 'removeClass']('x-date-mp-sel');
28107         });
28108     },
28109
28110     selectMPMonth: function(m){
28111         
28112     },
28113
28114     onMonthClick : function(e, t){
28115         e.stopEvent();
28116         var el = new Roo.Element(t), pn;
28117         if(el.is('button.x-date-mp-cancel')){
28118             this.hideMonthPicker();
28119         }
28120         else if(el.is('button.x-date-mp-ok')){
28121             this.update(new Date(this.mpSelYear, this.mpSelMonth, (this.activeDate || this.value).getDate()));
28122             this.hideMonthPicker();
28123         }
28124         else if(pn = el.up('td.x-date-mp-month', 2)){
28125             this.mpMonths.removeClass('x-date-mp-sel');
28126             pn.addClass('x-date-mp-sel');
28127             this.mpSelMonth = pn.dom.xmonth;
28128         }
28129         else if(pn = el.up('td.x-date-mp-year', 2)){
28130             this.mpYears.removeClass('x-date-mp-sel');
28131             pn.addClass('x-date-mp-sel');
28132             this.mpSelYear = pn.dom.xyear;
28133         }
28134         else if(el.is('a.x-date-mp-prev')){
28135             this.updateMPYear(this.mpyear-10);
28136         }
28137         else if(el.is('a.x-date-mp-next')){
28138             this.updateMPYear(this.mpyear+10);
28139         }
28140     },
28141
28142     onMonthDblClick : function(e, t){
28143         e.stopEvent();
28144         var el = new Roo.Element(t), pn;
28145         if(pn = el.up('td.x-date-mp-month', 2)){
28146             this.update(new Date(this.mpSelYear, pn.dom.xmonth, (this.activeDate || this.value).getDate()));
28147             this.hideMonthPicker();
28148         }
28149         else if(pn = el.up('td.x-date-mp-year', 2)){
28150             this.update(new Date(pn.dom.xyear, this.mpSelMonth, (this.activeDate || this.value).getDate()));
28151             this.hideMonthPicker();
28152         }
28153     },
28154
28155     hideMonthPicker : function(disableAnim){
28156         if(this.monthPicker){
28157             if(disableAnim === true){
28158                 this.monthPicker.hide();
28159             }else{
28160                 this.monthPicker.slideOut('t', {duration:.2});
28161             }
28162         }
28163     },
28164
28165     // private
28166     showPrevMonth : function(e){
28167         this.update(this.activeDate.add("mo", -1));
28168     },
28169
28170     // private
28171     showNextMonth : function(e){
28172         this.update(this.activeDate.add("mo", 1));
28173     },
28174
28175     // private
28176     showPrevYear : function(){
28177         this.update(this.activeDate.add("y", -1));
28178     },
28179
28180     // private
28181     showNextYear : function(){
28182         this.update(this.activeDate.add("y", 1));
28183     },
28184
28185     // private
28186     handleMouseWheel : function(e){
28187         var delta = e.getWheelDelta();
28188         if(delta > 0){
28189             this.showPrevMonth();
28190             e.stopEvent();
28191         } else if(delta < 0){
28192             this.showNextMonth();
28193             e.stopEvent();
28194         }
28195     },
28196
28197     // private
28198     handleDateClick : function(e, t){
28199         e.stopEvent();
28200         if(t.dateValue && !Roo.fly(t.parentNode).hasClass("x-date-disabled")){
28201             this.setValue(new Date(t.dateValue));
28202             this.fireEvent("select", this, this.value);
28203         }
28204     },
28205
28206     // private
28207     selectToday : function(){
28208         this.setValue(new Date().clearTime());
28209         this.fireEvent("select", this, this.value);
28210     },
28211
28212     // private
28213     update : function(date)
28214     {
28215         var vd = this.activeDate;
28216         this.activeDate = date;
28217         if(vd && this.el){
28218             var t = date.getTime();
28219             if(vd.getMonth() == date.getMonth() && vd.getFullYear() == date.getFullYear()){
28220                 this.cells.removeClass("x-date-selected");
28221                 this.cells.each(function(c){
28222                    if(c.dom.firstChild.dateValue == t){
28223                        c.addClass("x-date-selected");
28224                        setTimeout(function(){
28225                             try{c.dom.firstChild.focus();}catch(e){}
28226                        }, 50);
28227                        return false;
28228                    }
28229                 });
28230                 return;
28231             }
28232         }
28233         
28234         var days = date.getDaysInMonth();
28235         var firstOfMonth = date.getFirstDateOfMonth();
28236         var startingPos = firstOfMonth.getDay()-this.startDay;
28237
28238         if(startingPos <= this.startDay){
28239             startingPos += 7;
28240         }
28241
28242         var pm = date.add("mo", -1);
28243         var prevStart = pm.getDaysInMonth()-startingPos;
28244
28245         var cells = this.cells.elements;
28246         var textEls = this.textNodes;
28247         days += startingPos;
28248
28249         // convert everything to numbers so it's fast
28250         var day = 86400000;
28251         var d = (new Date(pm.getFullYear(), pm.getMonth(), prevStart)).clearTime();
28252         var today = new Date().clearTime().getTime();
28253         var sel = date.clearTime().getTime();
28254         var min = this.minDate ? this.minDate.clearTime() : Number.NEGATIVE_INFINITY;
28255         var max = this.maxDate ? this.maxDate.clearTime() : Number.POSITIVE_INFINITY;
28256         var ddMatch = this.disabledDatesRE;
28257         var ddText = this.disabledDatesText;
28258         var ddays = this.disabledDays ? this.disabledDays.join("") : false;
28259         var ddaysText = this.disabledDaysText;
28260         var format = this.format;
28261
28262         var setCellClass = function(cal, cell){
28263             cell.title = "";
28264             var t = d.getTime();
28265             cell.firstChild.dateValue = t;
28266             if(t == today){
28267                 cell.className += " x-date-today";
28268                 cell.title = cal.todayText;
28269             }
28270             if(t == sel){
28271                 cell.className += " x-date-selected";
28272                 setTimeout(function(){
28273                     try{cell.firstChild.focus();}catch(e){}
28274                 }, 50);
28275             }
28276             // disabling
28277             if(t < min) {
28278                 cell.className = " x-date-disabled";
28279                 cell.title = cal.minText;
28280                 return;
28281             }
28282             if(t > max) {
28283                 cell.className = " x-date-disabled";
28284                 cell.title = cal.maxText;
28285                 return;
28286             }
28287             if(ddays){
28288                 if(ddays.indexOf(d.getDay()) != -1){
28289                     cell.title = ddaysText;
28290                     cell.className = " x-date-disabled";
28291                 }
28292             }
28293             if(ddMatch && format){
28294                 var fvalue = d.dateFormat(format);
28295                 if(ddMatch.test(fvalue)){
28296                     cell.title = ddText.replace("%0", fvalue);
28297                     cell.className = " x-date-disabled";
28298                 }
28299             }
28300         };
28301
28302         var i = 0;
28303         for(; i < startingPos; i++) {
28304             textEls[i].innerHTML = (++prevStart);
28305             d.setDate(d.getDate()+1);
28306             cells[i].className = "x-date-prevday";
28307             setCellClass(this, cells[i]);
28308         }
28309         for(; i < days; i++){
28310             intDay = i - startingPos + 1;
28311             textEls[i].innerHTML = (intDay);
28312             d.setDate(d.getDate()+1);
28313             cells[i].className = "x-date-active";
28314             setCellClass(this, cells[i]);
28315         }
28316         var extraDays = 0;
28317         for(; i < 42; i++) {
28318              textEls[i].innerHTML = (++extraDays);
28319              d.setDate(d.getDate()+1);
28320              cells[i].className = "x-date-nextday";
28321              setCellClass(this, cells[i]);
28322         }
28323
28324         this.mbtn.setText(this.monthNames[date.getMonth()] + " " + date.getFullYear());
28325         this.fireEvent('monthchange', this, date);
28326         
28327         if(!this.internalRender){
28328             var main = this.el.dom.firstChild;
28329             var w = main.offsetWidth;
28330             this.el.setWidth(w + this.el.getBorderWidth("lr"));
28331             Roo.fly(main).setWidth(w);
28332             this.internalRender = true;
28333             // opera does not respect the auto grow header center column
28334             // then, after it gets a width opera refuses to recalculate
28335             // without a second pass
28336             if(Roo.isOpera && !this.secondPass){
28337                 main.rows[0].cells[1].style.width = (w - (main.rows[0].cells[0].offsetWidth+main.rows[0].cells[2].offsetWidth)) + "px";
28338                 this.secondPass = true;
28339                 this.update.defer(10, this, [date]);
28340             }
28341         }
28342         
28343         
28344     }
28345 });        /*
28346  * Based on:
28347  * Ext JS Library 1.1.1
28348  * Copyright(c) 2006-2007, Ext JS, LLC.
28349  *
28350  * Originally Released Under LGPL - original licence link has changed is not relivant.
28351  *
28352  * Fork - LGPL
28353  * <script type="text/javascript">
28354  */
28355 /**
28356  * @class Roo.TabPanel
28357  * @extends Roo.util.Observable
28358  * A lightweight tab container.
28359  * <br><br>
28360  * Usage:
28361  * <pre><code>
28362 // basic tabs 1, built from existing content
28363 var tabs = new Roo.TabPanel("tabs1");
28364 tabs.addTab("script", "View Script");
28365 tabs.addTab("markup", "View Markup");
28366 tabs.activate("script");
28367
28368 // more advanced tabs, built from javascript
28369 var jtabs = new Roo.TabPanel("jtabs");
28370 jtabs.addTab("jtabs-1", "Normal Tab", "My content was added during construction.");
28371
28372 // set up the UpdateManager
28373 var tab2 = jtabs.addTab("jtabs-2", "Ajax Tab 1");
28374 var updater = tab2.getUpdateManager();
28375 updater.setDefaultUrl("ajax1.htm");
28376 tab2.on('activate', updater.refresh, updater, true);
28377
28378 // Use setUrl for Ajax loading
28379 var tab3 = jtabs.addTab("jtabs-3", "Ajax Tab 2");
28380 tab3.setUrl("ajax2.htm", null, true);
28381
28382 // Disabled tab
28383 var tab4 = jtabs.addTab("tabs1-5", "Disabled Tab", "Can't see me cause I'm disabled");
28384 tab4.disable();
28385
28386 jtabs.activate("jtabs-1");
28387  * </code></pre>
28388  * @constructor
28389  * Create a new TabPanel.
28390  * @param {String/HTMLElement/Roo.Element} container The id, DOM element or Roo.Element container where this TabPanel is to be rendered.
28391  * @param {Object/Boolean} config Config object to set any properties for this TabPanel, or true to render the tabs on the bottom.
28392  */
28393 Roo.TabPanel = function(container, config){
28394     /**
28395     * The container element for this TabPanel.
28396     * @type Roo.Element
28397     */
28398     this.el = Roo.get(container, true);
28399     if(config){
28400         if(typeof config == "boolean"){
28401             this.tabPosition = config ? "bottom" : "top";
28402         }else{
28403             Roo.apply(this, config);
28404         }
28405     }
28406     if(this.tabPosition == "bottom"){
28407         this.bodyEl = Roo.get(this.createBody(this.el.dom));
28408         this.el.addClass("x-tabs-bottom");
28409     }
28410     this.stripWrap = Roo.get(this.createStrip(this.el.dom), true);
28411     this.stripEl = Roo.get(this.createStripList(this.stripWrap.dom), true);
28412     this.stripBody = Roo.get(this.stripWrap.dom.firstChild.firstChild, true);
28413     if(Roo.isIE){
28414         Roo.fly(this.stripWrap.dom.firstChild).setStyle("overflow-x", "hidden");
28415     }
28416     if(this.tabPosition != "bottom"){
28417         /** The body element that contains {@link Roo.TabPanelItem} bodies. +
28418          * @type Roo.Element
28419          */
28420         this.bodyEl = Roo.get(this.createBody(this.el.dom));
28421         this.el.addClass("x-tabs-top");
28422     }
28423     this.items = [];
28424
28425     this.bodyEl.setStyle("position", "relative");
28426
28427     this.active = null;
28428     this.activateDelegate = this.activate.createDelegate(this);
28429
28430     this.addEvents({
28431         /**
28432          * @event tabchange
28433          * Fires when the active tab changes
28434          * @param {Roo.TabPanel} this
28435          * @param {Roo.TabPanelItem} activePanel The new active tab
28436          */
28437         "tabchange": true,
28438         /**
28439          * @event beforetabchange
28440          * Fires before the active tab changes, set cancel to true on the "e" parameter to cancel the change
28441          * @param {Roo.TabPanel} this
28442          * @param {Object} e Set cancel to true on this object to cancel the tab change
28443          * @param {Roo.TabPanelItem} tab The tab being changed to
28444          */
28445         "beforetabchange" : true
28446     });
28447
28448     Roo.EventManager.onWindowResize(this.onResize, this);
28449     this.cpad = this.el.getPadding("lr");
28450     this.hiddenCount = 0;
28451
28452
28453     // toolbar on the tabbar support...
28454     if (this.toolbar) {
28455         var tcfg = this.toolbar;
28456         tcfg.container = this.stripEl.child('td.x-tab-strip-toolbar');  
28457         this.toolbar = new Roo.Toolbar(tcfg);
28458         if (Roo.isSafari) {
28459             var tbl = tcfg.container.child('table', true);
28460             tbl.setAttribute('width', '100%');
28461         }
28462         
28463     }
28464    
28465
28466
28467     Roo.TabPanel.superclass.constructor.call(this);
28468 };
28469
28470 Roo.extend(Roo.TabPanel, Roo.util.Observable, {
28471     /*
28472      *@cfg {String} tabPosition "top" or "bottom" (defaults to "top")
28473      */
28474     tabPosition : "top",
28475     /*
28476      *@cfg {Number} currentTabWidth The width of the current tab (defaults to 0)
28477      */
28478     currentTabWidth : 0,
28479     /*
28480      *@cfg {Number} minTabWidth The minimum width of a tab (defaults to 40) (ignored if {@link #resizeTabs} is not true)
28481      */
28482     minTabWidth : 40,
28483     /*
28484      *@cfg {Number} maxTabWidth The maximum width of a tab (defaults to 250) (ignored if {@link #resizeTabs} is not true)
28485      */
28486     maxTabWidth : 250,
28487     /*
28488      *@cfg {Number} preferredTabWidth The preferred (default) width of a tab (defaults to 175) (ignored if {@link #resizeTabs} is not true)
28489      */
28490     preferredTabWidth : 175,
28491     /*
28492      *@cfg {Boolean} resizeTabs True to enable dynamic tab resizing (defaults to false)
28493      */
28494     resizeTabs : false,
28495     /*
28496      *@cfg {Boolean} monitorResize Set this to true to turn on window resize monitoring (ignored if {@link #resizeTabs} is not true) (defaults to true)
28497      */
28498     monitorResize : true,
28499     /*
28500      *@cfg {Object} toolbar xtype description of toolbar to show at the right of the tab bar. 
28501      */
28502     toolbar : false,
28503
28504     /**
28505      * Creates a new {@link Roo.TabPanelItem} by looking for an existing element with the provided id -- if it's not found it creates one.
28506      * @param {String} id The id of the div to use <b>or create</b>
28507      * @param {String} text The text for the tab
28508      * @param {String} content (optional) Content to put in the TabPanelItem body
28509      * @param {Boolean} closable (optional) True to create a close icon on the tab
28510      * @return {Roo.TabPanelItem} The created TabPanelItem
28511      */
28512     addTab : function(id, text, content, closable){
28513         var item = new Roo.TabPanelItem(this, id, text, closable);
28514         this.addTabItem(item);
28515         if(content){
28516             item.setContent(content);
28517         }
28518         return item;
28519     },
28520
28521     /**
28522      * Returns the {@link Roo.TabPanelItem} with the specified id/index
28523      * @param {String/Number} id The id or index of the TabPanelItem to fetch.
28524      * @return {Roo.TabPanelItem}
28525      */
28526     getTab : function(id){
28527         return this.items[id];
28528     },
28529
28530     /**
28531      * Hides the {@link Roo.TabPanelItem} with the specified id/index
28532      * @param {String/Number} id The id or index of the TabPanelItem to hide.
28533      */
28534     hideTab : function(id){
28535         var t = this.items[id];
28536         if(!t.isHidden()){
28537            t.setHidden(true);
28538            this.hiddenCount++;
28539            this.autoSizeTabs();
28540         }
28541     },
28542
28543     /**
28544      * "Unhides" the {@link Roo.TabPanelItem} with the specified id/index.
28545      * @param {String/Number} id The id or index of the TabPanelItem to unhide.
28546      */
28547     unhideTab : function(id){
28548         var t = this.items[id];
28549         if(t.isHidden()){
28550            t.setHidden(false);
28551            this.hiddenCount--;
28552            this.autoSizeTabs();
28553         }
28554     },
28555
28556     /**
28557      * Adds an existing {@link Roo.TabPanelItem}.
28558      * @param {Roo.TabPanelItem} item The TabPanelItem to add
28559      */
28560     addTabItem : function(item){
28561         this.items[item.id] = item;
28562         this.items.push(item);
28563         if(this.resizeTabs){
28564            item.setWidth(this.currentTabWidth || this.preferredTabWidth);
28565            this.autoSizeTabs();
28566         }else{
28567             item.autoSize();
28568         }
28569     },
28570
28571     /**
28572      * Removes a {@link Roo.TabPanelItem}.
28573      * @param {String/Number} id The id or index of the TabPanelItem to remove.
28574      */
28575     removeTab : function(id){
28576         var items = this.items;
28577         var tab = items[id];
28578         if(!tab) { return; }
28579         var index = items.indexOf(tab);
28580         if(this.active == tab && items.length > 1){
28581             var newTab = this.getNextAvailable(index);
28582             if(newTab) {
28583                 newTab.activate();
28584             }
28585         }
28586         this.stripEl.dom.removeChild(tab.pnode.dom);
28587         if(tab.bodyEl.dom.parentNode == this.bodyEl.dom){ // if it was moved already prevent error
28588             this.bodyEl.dom.removeChild(tab.bodyEl.dom);
28589         }
28590         items.splice(index, 1);
28591         delete this.items[tab.id];
28592         tab.fireEvent("close", tab);
28593         tab.purgeListeners();
28594         this.autoSizeTabs();
28595     },
28596
28597     getNextAvailable : function(start){
28598         var items = this.items;
28599         var index = start;
28600         // look for a next tab that will slide over to
28601         // replace the one being removed
28602         while(index < items.length){
28603             var item = items[++index];
28604             if(item && !item.isHidden()){
28605                 return item;
28606             }
28607         }
28608         // if one isn't found select the previous tab (on the left)
28609         index = start;
28610         while(index >= 0){
28611             var item = items[--index];
28612             if(item && !item.isHidden()){
28613                 return item;
28614             }
28615         }
28616         return null;
28617     },
28618
28619     /**
28620      * Disables a {@link Roo.TabPanelItem}. It cannot be the active tab, if it is this call is ignored.
28621      * @param {String/Number} id The id or index of the TabPanelItem to disable.
28622      */
28623     disableTab : function(id){
28624         var tab = this.items[id];
28625         if(tab && this.active != tab){
28626             tab.disable();
28627         }
28628     },
28629
28630     /**
28631      * Enables a {@link Roo.TabPanelItem} that is disabled.
28632      * @param {String/Number} id The id or index of the TabPanelItem to enable.
28633      */
28634     enableTab : function(id){
28635         var tab = this.items[id];
28636         tab.enable();
28637     },
28638
28639     /**
28640      * Activates a {@link Roo.TabPanelItem}. The currently active one will be deactivated.
28641      * @param {String/Number} id The id or index of the TabPanelItem to activate.
28642      * @return {Roo.TabPanelItem} The TabPanelItem.
28643      */
28644     activate : function(id){
28645         var tab = this.items[id];
28646         if(!tab){
28647             return null;
28648         }
28649         if(tab == this.active || tab.disabled){
28650             return tab;
28651         }
28652         var e = {};
28653         this.fireEvent("beforetabchange", this, e, tab);
28654         if(e.cancel !== true && !tab.disabled){
28655             if(this.active){
28656                 this.active.hide();
28657             }
28658             this.active = this.items[id];
28659             this.active.show();
28660             this.fireEvent("tabchange", this, this.active);
28661         }
28662         return tab;
28663     },
28664
28665     /**
28666      * Gets the active {@link Roo.TabPanelItem}.
28667      * @return {Roo.TabPanelItem} The active TabPanelItem or null if none are active.
28668      */
28669     getActiveTab : function(){
28670         return this.active;
28671     },
28672
28673     /**
28674      * Updates the tab body element to fit the height of the container element
28675      * for overflow scrolling
28676      * @param {Number} targetHeight (optional) Override the starting height from the elements height
28677      */
28678     syncHeight : function(targetHeight){
28679         var height = (targetHeight || this.el.getHeight())-this.el.getBorderWidth("tb")-this.el.getPadding("tb");
28680         var bm = this.bodyEl.getMargins();
28681         var newHeight = height-(this.stripWrap.getHeight()||0)-(bm.top+bm.bottom);
28682         this.bodyEl.setHeight(newHeight);
28683         return newHeight;
28684     },
28685
28686     onResize : function(){
28687         if(this.monitorResize){
28688             this.autoSizeTabs();
28689         }
28690     },
28691
28692     /**
28693      * Disables tab resizing while tabs are being added (if {@link #resizeTabs} is false this does nothing)
28694      */
28695     beginUpdate : function(){
28696         this.updating = true;
28697     },
28698
28699     /**
28700      * Stops an update and resizes the tabs (if {@link #resizeTabs} is false this does nothing)
28701      */
28702     endUpdate : function(){
28703         this.updating = false;
28704         this.autoSizeTabs();
28705     },
28706
28707     /**
28708      * Manual call to resize the tabs (if {@link #resizeTabs} is false this does nothing)
28709      */
28710     autoSizeTabs : function(){
28711         var count = this.items.length;
28712         var vcount = count - this.hiddenCount;
28713         if(!this.resizeTabs || count < 1 || vcount < 1 || this.updating) {
28714             return;
28715         }
28716         var w = Math.max(this.el.getWidth() - this.cpad, 10);
28717         var availWidth = Math.floor(w / vcount);
28718         var b = this.stripBody;
28719         if(b.getWidth() > w){
28720             var tabs = this.items;
28721             this.setTabWidth(Math.max(availWidth, this.minTabWidth)-2);
28722             if(availWidth < this.minTabWidth){
28723                 /*if(!this.sleft){    // incomplete scrolling code
28724                     this.createScrollButtons();
28725                 }
28726                 this.showScroll();
28727                 this.stripClip.setWidth(w - (this.sleft.getWidth()+this.sright.getWidth()));*/
28728             }
28729         }else{
28730             if(this.currentTabWidth < this.preferredTabWidth){
28731                 this.setTabWidth(Math.min(availWidth, this.preferredTabWidth)-2);
28732             }
28733         }
28734     },
28735
28736     /**
28737      * Returns the number of tabs in this TabPanel.
28738      * @return {Number}
28739      */
28740      getCount : function(){
28741          return this.items.length;
28742      },
28743
28744     /**
28745      * Resizes all the tabs to the passed width
28746      * @param {Number} The new width
28747      */
28748     setTabWidth : function(width){
28749         this.currentTabWidth = width;
28750         for(var i = 0, len = this.items.length; i < len; i++) {
28751                 if(!this.items[i].isHidden()) {
28752                 this.items[i].setWidth(width);
28753             }
28754         }
28755     },
28756
28757     /**
28758      * Destroys this TabPanel
28759      * @param {Boolean} removeEl (optional) True to remove the element from the DOM as well (defaults to undefined)
28760      */
28761     destroy : function(removeEl){
28762         Roo.EventManager.removeResizeListener(this.onResize, this);
28763         for(var i = 0, len = this.items.length; i < len; i++){
28764             this.items[i].purgeListeners();
28765         }
28766         if(removeEl === true){
28767             this.el.update("");
28768             this.el.remove();
28769         }
28770     }
28771 });
28772
28773 /**
28774  * @class Roo.TabPanelItem
28775  * @extends Roo.util.Observable
28776  * Represents an individual item (tab plus body) in a TabPanel.
28777  * @param {Roo.TabPanel} tabPanel The {@link Roo.TabPanel} this TabPanelItem belongs to
28778  * @param {String} id The id of this TabPanelItem
28779  * @param {String} text The text for the tab of this TabPanelItem
28780  * @param {Boolean} closable True to allow this TabPanelItem to be closable (defaults to false)
28781  */
28782 Roo.TabPanelItem = function(tabPanel, id, text, closable){
28783     /**
28784      * The {@link Roo.TabPanel} this TabPanelItem belongs to
28785      * @type Roo.TabPanel
28786      */
28787     this.tabPanel = tabPanel;
28788     /**
28789      * The id for this TabPanelItem
28790      * @type String
28791      */
28792     this.id = id;
28793     /** @private */
28794     this.disabled = false;
28795     /** @private */
28796     this.text = text;
28797     /** @private */
28798     this.loaded = false;
28799     this.closable = closable;
28800
28801     /**
28802      * The body element for this TabPanelItem.
28803      * @type Roo.Element
28804      */
28805     this.bodyEl = Roo.get(tabPanel.createItemBody(tabPanel.bodyEl.dom, id));
28806     this.bodyEl.setVisibilityMode(Roo.Element.VISIBILITY);
28807     this.bodyEl.setStyle("display", "block");
28808     this.bodyEl.setStyle("zoom", "1");
28809     this.hideAction();
28810
28811     var els = tabPanel.createStripElements(tabPanel.stripEl.dom, text, closable);
28812     /** @private */
28813     this.el = Roo.get(els.el, true);
28814     this.inner = Roo.get(els.inner, true);
28815     this.textEl = Roo.get(this.el.dom.firstChild.firstChild.firstChild, true);
28816     this.pnode = Roo.get(els.el.parentNode, true);
28817     this.el.on("mousedown", this.onTabMouseDown, this);
28818     this.el.on("click", this.onTabClick, this);
28819     /** @private */
28820     if(closable){
28821         var c = Roo.get(els.close, true);
28822         c.dom.title = this.closeText;
28823         c.addClassOnOver("close-over");
28824         c.on("click", this.closeClick, this);
28825      }
28826
28827     this.addEvents({
28828          /**
28829          * @event activate
28830          * Fires when this tab becomes the active tab.
28831          * @param {Roo.TabPanel} tabPanel The parent TabPanel
28832          * @param {Roo.TabPanelItem} this
28833          */
28834         "activate": true,
28835         /**
28836          * @event beforeclose
28837          * Fires before this tab is closed. To cancel the close, set cancel to true on e (e.cancel = true).
28838          * @param {Roo.TabPanelItem} this
28839          * @param {Object} e Set cancel to true on this object to cancel the close.
28840          */
28841         "beforeclose": true,
28842         /**
28843          * @event close
28844          * Fires when this tab is closed.
28845          * @param {Roo.TabPanelItem} this
28846          */
28847          "close": true,
28848         /**
28849          * @event deactivate
28850          * Fires when this tab is no longer the active tab.
28851          * @param {Roo.TabPanel} tabPanel The parent TabPanel
28852          * @param {Roo.TabPanelItem} this
28853          */
28854          "deactivate" : true
28855     });
28856     this.hidden = false;
28857
28858     Roo.TabPanelItem.superclass.constructor.call(this);
28859 };
28860
28861 Roo.extend(Roo.TabPanelItem, Roo.util.Observable, {
28862     purgeListeners : function(){
28863        Roo.util.Observable.prototype.purgeListeners.call(this);
28864        this.el.removeAllListeners();
28865     },
28866     /**
28867      * Shows this TabPanelItem -- this <b>does not</b> deactivate the currently active TabPanelItem.
28868      */
28869     show : function(){
28870         this.pnode.addClass("on");
28871         this.showAction();
28872         if(Roo.isOpera){
28873             this.tabPanel.stripWrap.repaint();
28874         }
28875         this.fireEvent("activate", this.tabPanel, this);
28876     },
28877
28878     /**
28879      * Returns true if this tab is the active tab.
28880      * @return {Boolean}
28881      */
28882     isActive : function(){
28883         return this.tabPanel.getActiveTab() == this;
28884     },
28885
28886     /**
28887      * Hides this TabPanelItem -- if you don't activate another TabPanelItem this could look odd.
28888      */
28889     hide : function(){
28890         this.pnode.removeClass("on");
28891         this.hideAction();
28892         this.fireEvent("deactivate", this.tabPanel, this);
28893     },
28894
28895     hideAction : function(){
28896         this.bodyEl.hide();
28897         this.bodyEl.setStyle("position", "absolute");
28898         this.bodyEl.setLeft("-20000px");
28899         this.bodyEl.setTop("-20000px");
28900     },
28901
28902     showAction : function(){
28903         this.bodyEl.setStyle("position", "relative");
28904         this.bodyEl.setTop("");
28905         this.bodyEl.setLeft("");
28906         this.bodyEl.show();
28907     },
28908
28909     /**
28910      * Set the tooltip for the tab.
28911      * @param {String} tooltip The tab's tooltip
28912      */
28913     setTooltip : function(text){
28914         if(Roo.QuickTips && Roo.QuickTips.isEnabled()){
28915             this.textEl.dom.qtip = text;
28916             this.textEl.dom.removeAttribute('title');
28917         }else{
28918             this.textEl.dom.title = text;
28919         }
28920     },
28921
28922     onTabClick : function(e){
28923         e.preventDefault();
28924         this.tabPanel.activate(this.id);
28925     },
28926
28927     onTabMouseDown : function(e){
28928         e.preventDefault();
28929         this.tabPanel.activate(this.id);
28930     },
28931
28932     getWidth : function(){
28933         return this.inner.getWidth();
28934     },
28935
28936     setWidth : function(width){
28937         var iwidth = width - this.pnode.getPadding("lr");
28938         this.inner.setWidth(iwidth);
28939         this.textEl.setWidth(iwidth-this.inner.getPadding("lr"));
28940         this.pnode.setWidth(width);
28941     },
28942
28943     /**
28944      * Show or hide the tab
28945      * @param {Boolean} hidden True to hide or false to show.
28946      */
28947     setHidden : function(hidden){
28948         this.hidden = hidden;
28949         this.pnode.setStyle("display", hidden ? "none" : "");
28950     },
28951
28952     /**
28953      * Returns true if this tab is "hidden"
28954      * @return {Boolean}
28955      */
28956     isHidden : function(){
28957         return this.hidden;
28958     },
28959
28960     /**
28961      * Returns the text for this tab
28962      * @return {String}
28963      */
28964     getText : function(){
28965         return this.text;
28966     },
28967
28968     autoSize : function(){
28969         //this.el.beginMeasure();
28970         this.textEl.setWidth(1);
28971         /*
28972          *  #2804 [new] Tabs in Roojs
28973          *  increase the width by 2-4 pixels to prevent the ellipssis showing in chrome
28974          */
28975         this.setWidth(this.textEl.dom.scrollWidth+this.pnode.getPadding("lr")+this.inner.getPadding("lr") + 2);
28976         //this.el.endMeasure();
28977     },
28978
28979     /**
28980      * Sets the text for the tab (Note: this also sets the tooltip text)
28981      * @param {String} text The tab's text and tooltip
28982      */
28983     setText : function(text){
28984         this.text = text;
28985         this.textEl.update(text);
28986         this.setTooltip(text);
28987         if(!this.tabPanel.resizeTabs){
28988             this.autoSize();
28989         }
28990     },
28991     /**
28992      * Activates this TabPanelItem -- this <b>does</b> deactivate the currently active TabPanelItem.
28993      */
28994     activate : function(){
28995         this.tabPanel.activate(this.id);
28996     },
28997
28998     /**
28999      * Disables this TabPanelItem -- this does nothing if this is the active TabPanelItem.
29000      */
29001     disable : function(){
29002         if(this.tabPanel.active != this){
29003             this.disabled = true;
29004             this.pnode.addClass("disabled");
29005         }
29006     },
29007
29008     /**
29009      * Enables this TabPanelItem if it was previously disabled.
29010      */
29011     enable : function(){
29012         this.disabled = false;
29013         this.pnode.removeClass("disabled");
29014     },
29015
29016     /**
29017      * Sets the content for this TabPanelItem.
29018      * @param {String} content The content
29019      * @param {Boolean} loadScripts true to look for and load scripts
29020      */
29021     setContent : function(content, loadScripts){
29022         this.bodyEl.update(content, loadScripts);
29023     },
29024
29025     /**
29026      * Gets the {@link Roo.UpdateManager} for the body of this TabPanelItem. Enables you to perform Ajax updates.
29027      * @return {Roo.UpdateManager} The UpdateManager
29028      */
29029     getUpdateManager : function(){
29030         return this.bodyEl.getUpdateManager();
29031     },
29032
29033     /**
29034      * Set a URL to be used to load the content for this TabPanelItem.
29035      * @param {String/Function} url The URL to load the content from, or a function to call to get the URL
29036      * @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)
29037      * @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)
29038      * @return {Roo.UpdateManager} The UpdateManager
29039      */
29040     setUrl : function(url, params, loadOnce){
29041         if(this.refreshDelegate){
29042             this.un('activate', this.refreshDelegate);
29043         }
29044         this.refreshDelegate = this._handleRefresh.createDelegate(this, [url, params, loadOnce]);
29045         this.on("activate", this.refreshDelegate);
29046         return this.bodyEl.getUpdateManager();
29047     },
29048
29049     /** @private */
29050     _handleRefresh : function(url, params, loadOnce){
29051         if(!loadOnce || !this.loaded){
29052             var updater = this.bodyEl.getUpdateManager();
29053             updater.update(url, params, this._setLoaded.createDelegate(this));
29054         }
29055     },
29056
29057     /**
29058      *   Forces a content refresh from the URL specified in the {@link #setUrl} method.
29059      *   Will fail silently if the setUrl method has not been called.
29060      *   This does not activate the panel, just updates its content.
29061      */
29062     refresh : function(){
29063         if(this.refreshDelegate){
29064            this.loaded = false;
29065            this.refreshDelegate();
29066         }
29067     },
29068
29069     /** @private */
29070     _setLoaded : function(){
29071         this.loaded = true;
29072     },
29073
29074     /** @private */
29075     closeClick : function(e){
29076         var o = {};
29077         e.stopEvent();
29078         this.fireEvent("beforeclose", this, o);
29079         if(o.cancel !== true){
29080             this.tabPanel.removeTab(this.id);
29081         }
29082     },
29083     /**
29084      * The text displayed in the tooltip for the close icon.
29085      * @type String
29086      */
29087     closeText : "Close this tab"
29088 });
29089
29090 /** @private */
29091 Roo.TabPanel.prototype.createStrip = function(container){
29092     var strip = document.createElement("div");
29093     strip.className = "x-tabs-wrap";
29094     container.appendChild(strip);
29095     return strip;
29096 };
29097 /** @private */
29098 Roo.TabPanel.prototype.createStripList = function(strip){
29099     // div wrapper for retard IE
29100     // returns the "tr" element.
29101     strip.innerHTML = '<div class="x-tabs-strip-wrap">'+
29102         '<table class="x-tabs-strip" cellspacing="0" cellpadding="0" border="0"><tbody><tr>'+
29103         '<td class="x-tab-strip-toolbar"></td></tr></tbody></table></div>';
29104     return strip.firstChild.firstChild.firstChild.firstChild;
29105 };
29106 /** @private */
29107 Roo.TabPanel.prototype.createBody = function(container){
29108     var body = document.createElement("div");
29109     Roo.id(body, "tab-body");
29110     Roo.fly(body).addClass("x-tabs-body");
29111     container.appendChild(body);
29112     return body;
29113 };
29114 /** @private */
29115 Roo.TabPanel.prototype.createItemBody = function(bodyEl, id){
29116     var body = Roo.getDom(id);
29117     if(!body){
29118         body = document.createElement("div");
29119         body.id = id;
29120     }
29121     Roo.fly(body).addClass("x-tabs-item-body");
29122     bodyEl.insertBefore(body, bodyEl.firstChild);
29123     return body;
29124 };
29125 /** @private */
29126 Roo.TabPanel.prototype.createStripElements = function(stripEl, text, closable){
29127     var td = document.createElement("td");
29128     stripEl.insertBefore(td, stripEl.childNodes[stripEl.childNodes.length-1]);
29129     //stripEl.appendChild(td);
29130     if(closable){
29131         td.className = "x-tabs-closable";
29132         if(!this.closeTpl){
29133             this.closeTpl = new Roo.Template(
29134                '<a href="#" class="x-tabs-right"><span class="x-tabs-left"><em class="x-tabs-inner">' +
29135                '<span unselectable="on"' + (this.disableTooltips ? '' : ' title="{text}"') +' class="x-tabs-text">{text}</span>' +
29136                '<div unselectable="on" class="close-icon">&#160;</div></em></span></a>'
29137             );
29138         }
29139         var el = this.closeTpl.overwrite(td, {"text": text});
29140         var close = el.getElementsByTagName("div")[0];
29141         var inner = el.getElementsByTagName("em")[0];
29142         return {"el": el, "close": close, "inner": inner};
29143     } else {
29144         if(!this.tabTpl){
29145             this.tabTpl = new Roo.Template(
29146                '<a href="#" class="x-tabs-right"><span class="x-tabs-left"><em class="x-tabs-inner">' +
29147                '<span unselectable="on"' + (this.disableTooltips ? '' : ' title="{text}"') +' class="x-tabs-text">{text}</span></em></span></a>'
29148             );
29149         }
29150         var el = this.tabTpl.overwrite(td, {"text": text});
29151         var inner = el.getElementsByTagName("em")[0];
29152         return {"el": el, "inner": inner};
29153     }
29154 };/*
29155  * Based on:
29156  * Ext JS Library 1.1.1
29157  * Copyright(c) 2006-2007, Ext JS, LLC.
29158  *
29159  * Originally Released Under LGPL - original licence link has changed is not relivant.
29160  *
29161  * Fork - LGPL
29162  * <script type="text/javascript">
29163  */
29164
29165 /**
29166  * @class Roo.Button
29167  * @extends Roo.util.Observable
29168  * Simple Button class
29169  * @cfg {String} text The button text
29170  * @cfg {String} icon The path to an image to display in the button (the image will be set as the background-image
29171  * CSS property of the button by default, so if you want a mixed icon/text button, set cls:"x-btn-text-icon")
29172  * @cfg {Function} handler A function called when the button is clicked (can be used instead of click event)
29173  * @cfg {Object} scope The scope of the handler
29174  * @cfg {Number} minWidth The minimum width for this button (used to give a set of buttons a common width)
29175  * @cfg {String/Object} tooltip The tooltip for the button - can be a string or QuickTips config object
29176  * @cfg {Boolean} hidden True to start hidden (defaults to false)
29177  * @cfg {Boolean} disabled True to start disabled (defaults to false)
29178  * @cfg {Boolean} pressed True to start pressed (only if enableToggle = true)
29179  * @cfg {String} toggleGroup The group this toggle button is a member of (only 1 per group can be pressed, only
29180    applies if enableToggle = true)
29181  * @cfg {String/HTMLElement/Element} renderTo The element to append the button to
29182  * @cfg {Boolean/Object} repeat True to repeat fire the click event while the mouse is down. This can also be
29183   an {@link Roo.util.ClickRepeater} config object (defaults to false).
29184  * @constructor
29185  * Create a new button
29186  * @param {Object} config The config object
29187  */
29188 Roo.Button = function(renderTo, config)
29189 {
29190     if (!config) {
29191         config = renderTo;
29192         renderTo = config.renderTo || false;
29193     }
29194     
29195     Roo.apply(this, config);
29196     this.addEvents({
29197         /**
29198              * @event click
29199              * Fires when this button is clicked
29200              * @param {Button} this
29201              * @param {EventObject} e The click event
29202              */
29203             "click" : true,
29204         /**
29205              * @event toggle
29206              * Fires when the "pressed" state of this button changes (only if enableToggle = true)
29207              * @param {Button} this
29208              * @param {Boolean} pressed
29209              */
29210             "toggle" : true,
29211         /**
29212              * @event mouseover
29213              * Fires when the mouse hovers over the button
29214              * @param {Button} this
29215              * @param {Event} e The event object
29216              */
29217         'mouseover' : true,
29218         /**
29219              * @event mouseout
29220              * Fires when the mouse exits the button
29221              * @param {Button} this
29222              * @param {Event} e The event object
29223              */
29224         'mouseout': true,
29225          /**
29226              * @event render
29227              * Fires when the button is rendered
29228              * @param {Button} this
29229              */
29230         'render': true
29231     });
29232     if(this.menu){
29233         this.menu = Roo.menu.MenuMgr.get(this.menu);
29234     }
29235     // register listeners first!!  - so render can be captured..
29236     Roo.util.Observable.call(this);
29237     if(renderTo){
29238         this.render(renderTo);
29239     }
29240     
29241   
29242 };
29243
29244 Roo.extend(Roo.Button, Roo.util.Observable, {
29245     /**
29246      * 
29247      */
29248     
29249     /**
29250      * Read-only. True if this button is hidden
29251      * @type Boolean
29252      */
29253     hidden : false,
29254     /**
29255      * Read-only. True if this button is disabled
29256      * @type Boolean
29257      */
29258     disabled : false,
29259     /**
29260      * Read-only. True if this button is pressed (only if enableToggle = true)
29261      * @type Boolean
29262      */
29263     pressed : false,
29264
29265     /**
29266      * @cfg {Number} tabIndex 
29267      * The DOM tabIndex for this button (defaults to undefined)
29268      */
29269     tabIndex : undefined,
29270
29271     /**
29272      * @cfg {Boolean} enableToggle
29273      * True to enable pressed/not pressed toggling (defaults to false)
29274      */
29275     enableToggle: false,
29276     /**
29277      * @cfg {Mixed} menu
29278      * Standard menu attribute consisting of a reference to a menu object, a menu id or a menu config blob (defaults to undefined).
29279      */
29280     menu : undefined,
29281     /**
29282      * @cfg {String} menuAlign
29283      * The position to align the menu to (see {@link Roo.Element#alignTo} for more details, defaults to 'tl-bl?').
29284      */
29285     menuAlign : "tl-bl?",
29286
29287     /**
29288      * @cfg {String} iconCls
29289      * A css class which sets a background image to be used as the icon for this button (defaults to undefined).
29290      */
29291     iconCls : undefined,
29292     /**
29293      * @cfg {String} type
29294      * The button's type, corresponding to the DOM input element type attribute.  Either "submit," "reset" or "button" (default).
29295      */
29296     type : 'button',
29297
29298     // private
29299     menuClassTarget: 'tr',
29300
29301     /**
29302      * @cfg {String} clickEvent
29303      * The type of event to map to the button's event handler (defaults to 'click')
29304      */
29305     clickEvent : 'click',
29306
29307     /**
29308      * @cfg {Boolean} handleMouseEvents
29309      * False to disable visual cues on mouseover, mouseout and mousedown (defaults to true)
29310      */
29311     handleMouseEvents : true,
29312
29313     /**
29314      * @cfg {String} tooltipType
29315      * The type of tooltip to use. Either "qtip" (default) for QuickTips or "title" for title attribute.
29316      */
29317     tooltipType : 'qtip',
29318
29319     /**
29320      * @cfg {String} cls
29321      * A CSS class to apply to the button's main element.
29322      */
29323     
29324     /**
29325      * @cfg {Roo.Template} template (Optional)
29326      * An {@link Roo.Template} with which to create the Button's main element. This Template must
29327      * contain numeric substitution parameter 0 if it is to display the tRoo property. Changing the template could
29328      * require code modifications if required elements (e.g. a button) aren't present.
29329      */
29330
29331     // private
29332     render : function(renderTo){
29333         var btn;
29334         if(this.hideParent){
29335             this.parentEl = Roo.get(renderTo);
29336         }
29337         if(!this.dhconfig){
29338             if(!this.template){
29339                 if(!Roo.Button.buttonTemplate){
29340                     // hideous table template
29341                     Roo.Button.buttonTemplate = new Roo.Template(
29342                         '<table border="0" cellpadding="0" cellspacing="0" class="x-btn-wrap"><tbody><tr>',
29343                         '<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>',
29344                         "</tr></tbody></table>");
29345                 }
29346                 this.template = Roo.Button.buttonTemplate;
29347             }
29348             btn = this.template.append(renderTo, [this.text || '&#160;', this.type], true);
29349             var btnEl = btn.child("button:first");
29350             btnEl.on('focus', this.onFocus, this);
29351             btnEl.on('blur', this.onBlur, this);
29352             if(this.cls){
29353                 btn.addClass(this.cls);
29354             }
29355             if(this.icon){
29356                 btnEl.setStyle('background-image', 'url(' +this.icon +')');
29357             }
29358             if(this.iconCls){
29359                 btnEl.addClass(this.iconCls);
29360                 if(!this.cls){
29361                     btn.addClass(this.text ? 'x-btn-text-icon' : 'x-btn-icon');
29362                 }
29363             }
29364             if(this.tabIndex !== undefined){
29365                 btnEl.dom.tabIndex = this.tabIndex;
29366             }
29367             if(this.tooltip){
29368                 if(typeof this.tooltip == 'object'){
29369                     Roo.QuickTips.tips(Roo.apply({
29370                           target: btnEl.id
29371                     }, this.tooltip));
29372                 } else {
29373                     btnEl.dom[this.tooltipType] = this.tooltip;
29374                 }
29375             }
29376         }else{
29377             btn = Roo.DomHelper.append(Roo.get(renderTo).dom, this.dhconfig, true);
29378         }
29379         this.el = btn;
29380         if(this.id){
29381             this.el.dom.id = this.el.id = this.id;
29382         }
29383         if(this.menu){
29384             this.el.child(this.menuClassTarget).addClass("x-btn-with-menu");
29385             this.menu.on("show", this.onMenuShow, this);
29386             this.menu.on("hide", this.onMenuHide, this);
29387         }
29388         btn.addClass("x-btn");
29389         if(Roo.isIE && !Roo.isIE7){
29390             this.autoWidth.defer(1, this);
29391         }else{
29392             this.autoWidth();
29393         }
29394         if(this.handleMouseEvents){
29395             btn.on("mouseover", this.onMouseOver, this);
29396             btn.on("mouseout", this.onMouseOut, this);
29397             btn.on("mousedown", this.onMouseDown, this);
29398         }
29399         btn.on(this.clickEvent, this.onClick, this);
29400         //btn.on("mouseup", this.onMouseUp, this);
29401         if(this.hidden){
29402             this.hide();
29403         }
29404         if(this.disabled){
29405             this.disable();
29406         }
29407         Roo.ButtonToggleMgr.register(this);
29408         if(this.pressed){
29409             this.el.addClass("x-btn-pressed");
29410         }
29411         if(this.repeat){
29412             var repeater = new Roo.util.ClickRepeater(btn,
29413                 typeof this.repeat == "object" ? this.repeat : {}
29414             );
29415             repeater.on("click", this.onClick,  this);
29416         }
29417         
29418         this.fireEvent('render', this);
29419         
29420     },
29421     /**
29422      * Returns the button's underlying element
29423      * @return {Roo.Element} The element
29424      */
29425     getEl : function(){
29426         return this.el;  
29427     },
29428     
29429     /**
29430      * Destroys this Button and removes any listeners.
29431      */
29432     destroy : function(){
29433         Roo.ButtonToggleMgr.unregister(this);
29434         this.el.removeAllListeners();
29435         this.purgeListeners();
29436         this.el.remove();
29437     },
29438
29439     // private
29440     autoWidth : function(){
29441         if(this.el){
29442             this.el.setWidth("auto");
29443             if(Roo.isIE7 && Roo.isStrict){
29444                 var ib = this.el.child('button');
29445                 if(ib && ib.getWidth() > 20){
29446                     ib.clip();
29447                     ib.setWidth(Roo.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));
29448                 }
29449             }
29450             if(this.minWidth){
29451                 if(this.hidden){
29452                     this.el.beginMeasure();
29453                 }
29454                 if(this.el.getWidth() < this.minWidth){
29455                     this.el.setWidth(this.minWidth);
29456                 }
29457                 if(this.hidden){
29458                     this.el.endMeasure();
29459                 }
29460             }
29461         }
29462     },
29463
29464     /**
29465      * Assigns this button's click handler
29466      * @param {Function} handler The function to call when the button is clicked
29467      * @param {Object} scope (optional) Scope for the function passed in
29468      */
29469     setHandler : function(handler, scope){
29470         this.handler = handler;
29471         this.scope = scope;  
29472     },
29473     
29474     /**
29475      * Sets this button's text
29476      * @param {String} text The button text
29477      */
29478     setText : function(text){
29479         this.text = text;
29480         if(this.el){
29481             this.el.child("td.x-btn-center button.x-btn-text").update(text);
29482         }
29483         this.autoWidth();
29484     },
29485     
29486     /**
29487      * Gets the text for this button
29488      * @return {String} The button text
29489      */
29490     getText : function(){
29491         return this.text;  
29492     },
29493     
29494     /**
29495      * Show this button
29496      */
29497     show: function(){
29498         this.hidden = false;
29499         if(this.el){
29500             this[this.hideParent? 'parentEl' : 'el'].setStyle("display", "");
29501         }
29502     },
29503     
29504     /**
29505      * Hide this button
29506      */
29507     hide: function(){
29508         this.hidden = true;
29509         if(this.el){
29510             this[this.hideParent? 'parentEl' : 'el'].setStyle("display", "none");
29511         }
29512     },
29513     
29514     /**
29515      * Convenience function for boolean show/hide
29516      * @param {Boolean} visible True to show, false to hide
29517      */
29518     setVisible: function(visible){
29519         if(visible) {
29520             this.show();
29521         }else{
29522             this.hide();
29523         }
29524     },
29525     
29526     /**
29527      * If a state it passed, it becomes the pressed state otherwise the current state is toggled.
29528      * @param {Boolean} state (optional) Force a particular state
29529      */
29530     toggle : function(state){
29531         state = state === undefined ? !this.pressed : state;
29532         if(state != this.pressed){
29533             if(state){
29534                 this.el.addClass("x-btn-pressed");
29535                 this.pressed = true;
29536                 this.fireEvent("toggle", this, true);
29537             }else{
29538                 this.el.removeClass("x-btn-pressed");
29539                 this.pressed = false;
29540                 this.fireEvent("toggle", this, false);
29541             }
29542             if(this.toggleHandler){
29543                 this.toggleHandler.call(this.scope || this, this, state);
29544             }
29545         }
29546     },
29547     
29548     /**
29549      * Focus the button
29550      */
29551     focus : function(){
29552         this.el.child('button:first').focus();
29553     },
29554     
29555     /**
29556      * Disable this button
29557      */
29558     disable : function(){
29559         if(this.el){
29560             this.el.addClass("x-btn-disabled");
29561         }
29562         this.disabled = true;
29563     },
29564     
29565     /**
29566      * Enable this button
29567      */
29568     enable : function(){
29569         if(this.el){
29570             this.el.removeClass("x-btn-disabled");
29571         }
29572         this.disabled = false;
29573     },
29574
29575     /**
29576      * Convenience function for boolean enable/disable
29577      * @param {Boolean} enabled True to enable, false to disable
29578      */
29579     setDisabled : function(v){
29580         this[v !== true ? "enable" : "disable"]();
29581     },
29582
29583     // private
29584     onClick : function(e)
29585     {
29586         if(e){
29587             e.preventDefault();
29588         }
29589         if(e.button != 0){
29590             return;
29591         }
29592         if(!this.disabled){
29593             if(this.enableToggle){
29594                 this.toggle();
29595             }
29596             if(this.menu && !this.menu.isVisible()){
29597                 this.menu.show(this.el, this.menuAlign);
29598             }
29599             this.fireEvent("click", this, e);
29600             if(this.handler){
29601                 this.el.removeClass("x-btn-over");
29602                 this.handler.call(this.scope || this, this, e);
29603             }
29604         }
29605     },
29606     // private
29607     onMouseOver : function(e){
29608         if(!this.disabled){
29609             this.el.addClass("x-btn-over");
29610             this.fireEvent('mouseover', this, e);
29611         }
29612     },
29613     // private
29614     onMouseOut : function(e){
29615         if(!e.within(this.el,  true)){
29616             this.el.removeClass("x-btn-over");
29617             this.fireEvent('mouseout', this, e);
29618         }
29619     },
29620     // private
29621     onFocus : function(e){
29622         if(!this.disabled){
29623             this.el.addClass("x-btn-focus");
29624         }
29625     },
29626     // private
29627     onBlur : function(e){
29628         this.el.removeClass("x-btn-focus");
29629     },
29630     // private
29631     onMouseDown : function(e){
29632         if(!this.disabled && e.button == 0){
29633             this.el.addClass("x-btn-click");
29634             Roo.get(document).on('mouseup', this.onMouseUp, this);
29635         }
29636     },
29637     // private
29638     onMouseUp : function(e){
29639         if(e.button == 0){
29640             this.el.removeClass("x-btn-click");
29641             Roo.get(document).un('mouseup', this.onMouseUp, this);
29642         }
29643     },
29644     // private
29645     onMenuShow : function(e){
29646         this.el.addClass("x-btn-menu-active");
29647     },
29648     // private
29649     onMenuHide : function(e){
29650         this.el.removeClass("x-btn-menu-active");
29651     }   
29652 });
29653
29654 // Private utility class used by Button
29655 Roo.ButtonToggleMgr = function(){
29656    var groups = {};
29657    
29658    function toggleGroup(btn, state){
29659        if(state){
29660            var g = groups[btn.toggleGroup];
29661            for(var i = 0, l = g.length; i < l; i++){
29662                if(g[i] != btn){
29663                    g[i].toggle(false);
29664                }
29665            }
29666        }
29667    }
29668    
29669    return {
29670        register : function(btn){
29671            if(!btn.toggleGroup){
29672                return;
29673            }
29674            var g = groups[btn.toggleGroup];
29675            if(!g){
29676                g = groups[btn.toggleGroup] = [];
29677            }
29678            g.push(btn);
29679            btn.on("toggle", toggleGroup);
29680        },
29681        
29682        unregister : function(btn){
29683            if(!btn.toggleGroup){
29684                return;
29685            }
29686            var g = groups[btn.toggleGroup];
29687            if(g){
29688                g.remove(btn);
29689                btn.un("toggle", toggleGroup);
29690            }
29691        }
29692    };
29693 }();/*
29694  * Based on:
29695  * Ext JS Library 1.1.1
29696  * Copyright(c) 2006-2007, Ext JS, LLC.
29697  *
29698  * Originally Released Under LGPL - original licence link has changed is not relivant.
29699  *
29700  * Fork - LGPL
29701  * <script type="text/javascript">
29702  */
29703  
29704 /**
29705  * @class Roo.SplitButton
29706  * @extends Roo.Button
29707  * A split button that provides a built-in dropdown arrow that can fire an event separately from the default
29708  * click event of the button.  Typically this would be used to display a dropdown menu that provides additional
29709  * options to the primary button action, but any custom handler can provide the arrowclick implementation.
29710  * @cfg {Function} arrowHandler A function called when the arrow button is clicked (can be used instead of click event)
29711  * @cfg {String} arrowTooltip The title attribute of the arrow
29712  * @constructor
29713  * Create a new menu button
29714  * @param {String/HTMLElement/Element} renderTo The element to append the button to
29715  * @param {Object} config The config object
29716  */
29717 Roo.SplitButton = function(renderTo, config){
29718     Roo.SplitButton.superclass.constructor.call(this, renderTo, config);
29719     /**
29720      * @event arrowclick
29721      * Fires when this button's arrow is clicked
29722      * @param {SplitButton} this
29723      * @param {EventObject} e The click event
29724      */
29725     this.addEvents({"arrowclick":true});
29726 };
29727
29728 Roo.extend(Roo.SplitButton, Roo.Button, {
29729     render : function(renderTo){
29730         // this is one sweet looking template!
29731         var tpl = new Roo.Template(
29732             '<table cellspacing="0" class="x-btn-menu-wrap x-btn"><tr><td>',
29733             '<table cellspacing="0" class="x-btn-wrap x-btn-menu-text-wrap"><tbody>',
29734             '<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>',
29735             "</tbody></table></td><td>",
29736             '<table cellspacing="0" class="x-btn-wrap x-btn-menu-arrow-wrap"><tbody>',
29737             '<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>',
29738             "</tbody></table></td></tr></table>"
29739         );
29740         var btn = tpl.append(renderTo, [this.text, this.type], true);
29741         var btnEl = btn.child("button");
29742         if(this.cls){
29743             btn.addClass(this.cls);
29744         }
29745         if(this.icon){
29746             btnEl.setStyle('background-image', 'url(' +this.icon +')');
29747         }
29748         if(this.iconCls){
29749             btnEl.addClass(this.iconCls);
29750             if(!this.cls){
29751                 btn.addClass(this.text ? 'x-btn-text-icon' : 'x-btn-icon');
29752             }
29753         }
29754         this.el = btn;
29755         if(this.handleMouseEvents){
29756             btn.on("mouseover", this.onMouseOver, this);
29757             btn.on("mouseout", this.onMouseOut, this);
29758             btn.on("mousedown", this.onMouseDown, this);
29759             btn.on("mouseup", this.onMouseUp, this);
29760         }
29761         btn.on(this.clickEvent, this.onClick, this);
29762         if(this.tooltip){
29763             if(typeof this.tooltip == 'object'){
29764                 Roo.QuickTips.tips(Roo.apply({
29765                       target: btnEl.id
29766                 }, this.tooltip));
29767             } else {
29768                 btnEl.dom[this.tooltipType] = this.tooltip;
29769             }
29770         }
29771         if(this.arrowTooltip){
29772             btn.child("button:nth(2)").dom[this.tooltipType] = this.arrowTooltip;
29773         }
29774         if(this.hidden){
29775             this.hide();
29776         }
29777         if(this.disabled){
29778             this.disable();
29779         }
29780         if(this.pressed){
29781             this.el.addClass("x-btn-pressed");
29782         }
29783         if(Roo.isIE && !Roo.isIE7){
29784             this.autoWidth.defer(1, this);
29785         }else{
29786             this.autoWidth();
29787         }
29788         if(this.menu){
29789             this.menu.on("show", this.onMenuShow, this);
29790             this.menu.on("hide", this.onMenuHide, this);
29791         }
29792         this.fireEvent('render', this);
29793     },
29794
29795     // private
29796     autoWidth : function(){
29797         if(this.el){
29798             var tbl = this.el.child("table:first");
29799             var tbl2 = this.el.child("table:last");
29800             this.el.setWidth("auto");
29801             tbl.setWidth("auto");
29802             if(Roo.isIE7 && Roo.isStrict){
29803                 var ib = this.el.child('button:first');
29804                 if(ib && ib.getWidth() > 20){
29805                     ib.clip();
29806                     ib.setWidth(Roo.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));
29807                 }
29808             }
29809             if(this.minWidth){
29810                 if(this.hidden){
29811                     this.el.beginMeasure();
29812                 }
29813                 if((tbl.getWidth()+tbl2.getWidth()) < this.minWidth){
29814                     tbl.setWidth(this.minWidth-tbl2.getWidth());
29815                 }
29816                 if(this.hidden){
29817                     this.el.endMeasure();
29818                 }
29819             }
29820             this.el.setWidth(tbl.getWidth()+tbl2.getWidth());
29821         } 
29822     },
29823     /**
29824      * Sets this button's click handler
29825      * @param {Function} handler The function to call when the button is clicked
29826      * @param {Object} scope (optional) Scope for the function passed above
29827      */
29828     setHandler : function(handler, scope){
29829         this.handler = handler;
29830         this.scope = scope;  
29831     },
29832     
29833     /**
29834      * Sets this button's arrow click handler
29835      * @param {Function} handler The function to call when the arrow is clicked
29836      * @param {Object} scope (optional) Scope for the function passed above
29837      */
29838     setArrowHandler : function(handler, scope){
29839         this.arrowHandler = handler;
29840         this.scope = scope;  
29841     },
29842     
29843     /**
29844      * Focus the button
29845      */
29846     focus : function(){
29847         if(this.el){
29848             this.el.child("button:first").focus();
29849         }
29850     },
29851
29852     // private
29853     onClick : function(e){
29854         e.preventDefault();
29855         if(!this.disabled){
29856             if(e.getTarget(".x-btn-menu-arrow-wrap")){
29857                 if(this.menu && !this.menu.isVisible()){
29858                     this.menu.show(this.el, this.menuAlign);
29859                 }
29860                 this.fireEvent("arrowclick", this, e);
29861                 if(this.arrowHandler){
29862                     this.arrowHandler.call(this.scope || this, this, e);
29863                 }
29864             }else{
29865                 this.fireEvent("click", this, e);
29866                 if(this.handler){
29867                     this.handler.call(this.scope || this, this, e);
29868                 }
29869             }
29870         }
29871     },
29872     // private
29873     onMouseDown : function(e){
29874         if(!this.disabled){
29875             Roo.fly(e.getTarget("table")).addClass("x-btn-click");
29876         }
29877     },
29878     // private
29879     onMouseUp : function(e){
29880         Roo.fly(e.getTarget("table")).removeClass("x-btn-click");
29881     }   
29882 });
29883
29884
29885 // backwards compat
29886 Roo.MenuButton = Roo.SplitButton;/*
29887  * Based on:
29888  * Ext JS Library 1.1.1
29889  * Copyright(c) 2006-2007, Ext JS, LLC.
29890  *
29891  * Originally Released Under LGPL - original licence link has changed is not relivant.
29892  *
29893  * Fork - LGPL
29894  * <script type="text/javascript">
29895  */
29896
29897 /**
29898  * @class Roo.Toolbar
29899  * Basic Toolbar class.
29900  * @constructor
29901  * Creates a new Toolbar
29902  * @param {Object} container The config object
29903  */ 
29904 Roo.Toolbar = function(container, buttons, config)
29905 {
29906     /// old consturctor format still supported..
29907     if(container instanceof Array){ // omit the container for later rendering
29908         buttons = container;
29909         config = buttons;
29910         container = null;
29911     }
29912     if (typeof(container) == 'object' && container.xtype) {
29913         config = container;
29914         container = config.container;
29915         buttons = config.buttons || []; // not really - use items!!
29916     }
29917     var xitems = [];
29918     if (config && config.items) {
29919         xitems = config.items;
29920         delete config.items;
29921     }
29922     Roo.apply(this, config);
29923     this.buttons = buttons;
29924     
29925     if(container){
29926         this.render(container);
29927     }
29928     this.xitems = xitems;
29929     Roo.each(xitems, function(b) {
29930         this.add(b);
29931     }, this);
29932     
29933 };
29934
29935 Roo.Toolbar.prototype = {
29936     /**
29937      * @cfg {Array} items
29938      * array of button configs or elements to add (will be converted to a MixedCollection)
29939      */
29940     
29941     /**
29942      * @cfg {String/HTMLElement/Element} container
29943      * The id or element that will contain the toolbar
29944      */
29945     // private
29946     render : function(ct){
29947         this.el = Roo.get(ct);
29948         if(this.cls){
29949             this.el.addClass(this.cls);
29950         }
29951         // using a table allows for vertical alignment
29952         // 100% width is needed by Safari...
29953         this.el.update('<div class="x-toolbar x-small-editor"><table cellspacing="0"><tr></tr></table></div>');
29954         this.tr = this.el.child("tr", true);
29955         var autoId = 0;
29956         this.items = new Roo.util.MixedCollection(false, function(o){
29957             return o.id || ("item" + (++autoId));
29958         });
29959         if(this.buttons){
29960             this.add.apply(this, this.buttons);
29961             delete this.buttons;
29962         }
29963     },
29964
29965     /**
29966      * Adds element(s) to the toolbar -- this function takes a variable number of 
29967      * arguments of mixed type and adds them to the toolbar.
29968      * @param {Mixed} arg1 The following types of arguments are all valid:<br />
29969      * <ul>
29970      * <li>{@link Roo.Toolbar.Button} config: A valid button config object (equivalent to {@link #addButton})</li>
29971      * <li>HtmlElement: Any standard HTML element (equivalent to {@link #addElement})</li>
29972      * <li>Field: Any form field (equivalent to {@link #addField})</li>
29973      * <li>Item: Any subclass of {@link Roo.Toolbar.Item} (equivalent to {@link #addItem})</li>
29974      * <li>String: Any generic string (gets wrapped in a {@link Roo.Toolbar.TextItem}, equivalent to {@link #addText}).
29975      * Note that there are a few special strings that are treated differently as explained nRoo.</li>
29976      * <li>'separator' or '-': Creates a separator element (equivalent to {@link #addSeparator})</li>
29977      * <li>' ': Creates a spacer element (equivalent to {@link #addSpacer})</li>
29978      * <li>'->': Creates a fill element (equivalent to {@link #addFill})</li>
29979      * </ul>
29980      * @param {Mixed} arg2
29981      * @param {Mixed} etc.
29982      */
29983     add : function(){
29984         var a = arguments, l = a.length;
29985         for(var i = 0; i < l; i++){
29986             this._add(a[i]);
29987         }
29988     },
29989     // private..
29990     _add : function(el) {
29991         
29992         if (el.xtype) {
29993             el = Roo.factory(el, typeof(Roo.Toolbar[el.xtype]) == 'undefined' ? Roo.form : Roo.Toolbar);
29994         }
29995         
29996         if (el.applyTo){ // some kind of form field
29997             return this.addField(el);
29998         } 
29999         if (el.render){ // some kind of Toolbar.Item
30000             return this.addItem(el);
30001         }
30002         if (typeof el == "string"){ // string
30003             if(el == "separator" || el == "-"){
30004                 return this.addSeparator();
30005             }
30006             if (el == " "){
30007                 return this.addSpacer();
30008             }
30009             if(el == "->"){
30010                 return this.addFill();
30011             }
30012             return this.addText(el);
30013             
30014         }
30015         if(el.tagName){ // element
30016             return this.addElement(el);
30017         }
30018         if(typeof el == "object"){ // must be button config?
30019             return this.addButton(el);
30020         }
30021         // and now what?!?!
30022         return false;
30023         
30024     },
30025     
30026     /**
30027      * Add an Xtype element
30028      * @param {Object} xtype Xtype Object
30029      * @return {Object} created Object
30030      */
30031     addxtype : function(e){
30032         return this.add(e);  
30033     },
30034     
30035     /**
30036      * Returns the Element for this toolbar.
30037      * @return {Roo.Element}
30038      */
30039     getEl : function(){
30040         return this.el;  
30041     },
30042     
30043     /**
30044      * Adds a separator
30045      * @return {Roo.Toolbar.Item} The separator item
30046      */
30047     addSeparator : function(){
30048         return this.addItem(new Roo.Toolbar.Separator());
30049     },
30050
30051     /**
30052      * Adds a spacer element
30053      * @return {Roo.Toolbar.Spacer} The spacer item
30054      */
30055     addSpacer : function(){
30056         return this.addItem(new Roo.Toolbar.Spacer());
30057     },
30058
30059     /**
30060      * Adds a fill element that forces subsequent additions to the right side of the toolbar
30061      * @return {Roo.Toolbar.Fill} The fill item
30062      */
30063     addFill : function(){
30064         return this.addItem(new Roo.Toolbar.Fill());
30065     },
30066
30067     /**
30068      * Adds any standard HTML element to the toolbar
30069      * @param {String/HTMLElement/Element} el The element or id of the element to add
30070      * @return {Roo.Toolbar.Item} The element's item
30071      */
30072     addElement : function(el){
30073         return this.addItem(new Roo.Toolbar.Item(el));
30074     },
30075     /**
30076      * Collection of items on the toolbar.. (only Toolbar Items, so use fields to retrieve fields)
30077      * @type Roo.util.MixedCollection  
30078      */
30079     items : false,
30080      
30081     /**
30082      * Adds any Toolbar.Item or subclass
30083      * @param {Roo.Toolbar.Item} item
30084      * @return {Roo.Toolbar.Item} The item
30085      */
30086     addItem : function(item){
30087         var td = this.nextBlock();
30088         item.render(td);
30089         this.items.add(item);
30090         return item;
30091     },
30092     
30093     /**
30094      * Adds a button (or buttons). See {@link Roo.Toolbar.Button} for more info on the config.
30095      * @param {Object/Array} config A button config or array of configs
30096      * @return {Roo.Toolbar.Button/Array}
30097      */
30098     addButton : function(config){
30099         if(config instanceof Array){
30100             var buttons = [];
30101             for(var i = 0, len = config.length; i < len; i++) {
30102                 buttons.push(this.addButton(config[i]));
30103             }
30104             return buttons;
30105         }
30106         var b = config;
30107         if(!(config instanceof Roo.Toolbar.Button)){
30108             b = config.split ?
30109                 new Roo.Toolbar.SplitButton(config) :
30110                 new Roo.Toolbar.Button(config);
30111         }
30112         var td = this.nextBlock();
30113         b.render(td);
30114         this.items.add(b);
30115         return b;
30116     },
30117     
30118     /**
30119      * Adds text to the toolbar
30120      * @param {String} text The text to add
30121      * @return {Roo.Toolbar.Item} The element's item
30122      */
30123     addText : function(text){
30124         return this.addItem(new Roo.Toolbar.TextItem(text));
30125     },
30126     
30127     /**
30128      * Inserts any {@link Roo.Toolbar.Item}/{@link Roo.Toolbar.Button} at the specified index.
30129      * @param {Number} index The index where the item is to be inserted
30130      * @param {Object/Roo.Toolbar.Item/Roo.Toolbar.Button (may be Array)} item The button, or button config object to be inserted.
30131      * @return {Roo.Toolbar.Button/Item}
30132      */
30133     insertButton : function(index, item){
30134         if(item instanceof Array){
30135             var buttons = [];
30136             for(var i = 0, len = item.length; i < len; i++) {
30137                buttons.push(this.insertButton(index + i, item[i]));
30138             }
30139             return buttons;
30140         }
30141         if (!(item instanceof Roo.Toolbar.Button)){
30142            item = new Roo.Toolbar.Button(item);
30143         }
30144         var td = document.createElement("td");
30145         this.tr.insertBefore(td, this.tr.childNodes[index]);
30146         item.render(td);
30147         this.items.insert(index, item);
30148         return item;
30149     },
30150     
30151     /**
30152      * Adds a new element to the toolbar from the passed {@link Roo.DomHelper} config.
30153      * @param {Object} config
30154      * @return {Roo.Toolbar.Item} The element's item
30155      */
30156     addDom : function(config, returnEl){
30157         var td = this.nextBlock();
30158         Roo.DomHelper.overwrite(td, config);
30159         var ti = new Roo.Toolbar.Item(td.firstChild);
30160         ti.render(td);
30161         this.items.add(ti);
30162         return ti;
30163     },
30164
30165     /**
30166      * Collection of fields on the toolbar.. usefull for quering (value is false if there are no fields)
30167      * @type Roo.util.MixedCollection  
30168      */
30169     fields : false,
30170     
30171     /**
30172      * Adds a dynamically rendered Roo.form field (TextField, ComboBox, etc).
30173      * Note: the field should not have been rendered yet. For a field that has already been
30174      * rendered, use {@link #addElement}.
30175      * @param {Roo.form.Field} field
30176      * @return {Roo.ToolbarItem}
30177      */
30178      
30179       
30180     addField : function(field) {
30181         if (!this.fields) {
30182             var autoId = 0;
30183             this.fields = new Roo.util.MixedCollection(false, function(o){
30184                 return o.id || ("item" + (++autoId));
30185             });
30186
30187         }
30188         
30189         var td = this.nextBlock();
30190         field.render(td);
30191         var ti = new Roo.Toolbar.Item(td.firstChild);
30192         ti.render(td);
30193         this.items.add(ti);
30194         this.fields.add(field);
30195         return ti;
30196     },
30197     /**
30198      * Hide the toolbar
30199      * @method hide
30200      */
30201      
30202       
30203     hide : function()
30204     {
30205         this.el.child('div').setVisibilityMode(Roo.Element.DISPLAY);
30206         this.el.child('div').hide();
30207     },
30208     /**
30209      * Show the toolbar
30210      * @method show
30211      */
30212     show : function()
30213     {
30214         this.el.child('div').show();
30215     },
30216       
30217     // private
30218     nextBlock : function(){
30219         var td = document.createElement("td");
30220         this.tr.appendChild(td);
30221         return td;
30222     },
30223
30224     // private
30225     destroy : function(){
30226         if(this.items){ // rendered?
30227             Roo.destroy.apply(Roo, this.items.items);
30228         }
30229         if(this.fields){ // rendered?
30230             Roo.destroy.apply(Roo, this.fields.items);
30231         }
30232         Roo.Element.uncache(this.el, this.tr);
30233     }
30234 };
30235
30236 /**
30237  * @class Roo.Toolbar.Item
30238  * The base class that other classes should extend in order to get some basic common toolbar item functionality.
30239  * @constructor
30240  * Creates a new Item
30241  * @param {HTMLElement} el 
30242  */
30243 Roo.Toolbar.Item = function(el){
30244     var cfg = {};
30245     if (typeof (el.xtype) != 'undefined') {
30246         cfg = el;
30247         el = cfg.el;
30248     }
30249     
30250     this.el = Roo.getDom(el);
30251     this.id = Roo.id(this.el);
30252     this.hidden = false;
30253     
30254     this.addEvents({
30255          /**
30256              * @event render
30257              * Fires when the button is rendered
30258              * @param {Button} this
30259              */
30260         'render': true
30261     });
30262     Roo.Toolbar.Item.superclass.constructor.call(this,cfg);
30263 };
30264 Roo.extend(Roo.Toolbar.Item, Roo.util.Observable, {
30265 //Roo.Toolbar.Item.prototype = {
30266     
30267     /**
30268      * Get this item's HTML Element
30269      * @return {HTMLElement}
30270      */
30271     getEl : function(){
30272        return this.el;  
30273     },
30274
30275     // private
30276     render : function(td){
30277         
30278          this.td = td;
30279         td.appendChild(this.el);
30280         
30281         this.fireEvent('render', this);
30282     },
30283     
30284     /**
30285      * Removes and destroys this item.
30286      */
30287     destroy : function(){
30288         this.td.parentNode.removeChild(this.td);
30289     },
30290     
30291     /**
30292      * Shows this item.
30293      */
30294     show: function(){
30295         this.hidden = false;
30296         this.td.style.display = "";
30297     },
30298     
30299     /**
30300      * Hides this item.
30301      */
30302     hide: function(){
30303         this.hidden = true;
30304         this.td.style.display = "none";
30305     },
30306     
30307     /**
30308      * Convenience function for boolean show/hide.
30309      * @param {Boolean} visible true to show/false to hide
30310      */
30311     setVisible: function(visible){
30312         if(visible) {
30313             this.show();
30314         }else{
30315             this.hide();
30316         }
30317     },
30318     
30319     /**
30320      * Try to focus this item.
30321      */
30322     focus : function(){
30323         Roo.fly(this.el).focus();
30324     },
30325     
30326     /**
30327      * Disables this item.
30328      */
30329     disable : function(){
30330         Roo.fly(this.td).addClass("x-item-disabled");
30331         this.disabled = true;
30332         this.el.disabled = true;
30333     },
30334     
30335     /**
30336      * Enables this item.
30337      */
30338     enable : function(){
30339         Roo.fly(this.td).removeClass("x-item-disabled");
30340         this.disabled = false;
30341         this.el.disabled = false;
30342     }
30343 });
30344
30345
30346 /**
30347  * @class Roo.Toolbar.Separator
30348  * @extends Roo.Toolbar.Item
30349  * A simple toolbar separator class
30350  * @constructor
30351  * Creates a new Separator
30352  */
30353 Roo.Toolbar.Separator = function(cfg){
30354     
30355     var s = document.createElement("span");
30356     s.className = "ytb-sep";
30357     if (cfg) {
30358         cfg.el = s;
30359     }
30360     
30361     Roo.Toolbar.Separator.superclass.constructor.call(this, cfg || s);
30362 };
30363 Roo.extend(Roo.Toolbar.Separator, Roo.Toolbar.Item, {
30364     enable:Roo.emptyFn,
30365     disable:Roo.emptyFn,
30366     focus:Roo.emptyFn
30367 });
30368
30369 /**
30370  * @class Roo.Toolbar.Spacer
30371  * @extends Roo.Toolbar.Item
30372  * A simple element that adds extra horizontal space to a toolbar.
30373  * @constructor
30374  * Creates a new Spacer
30375  */
30376 Roo.Toolbar.Spacer = function(cfg){
30377     var s = document.createElement("div");
30378     s.className = "ytb-spacer";
30379     if (cfg) {
30380         cfg.el = s;
30381     }
30382     Roo.Toolbar.Spacer.superclass.constructor.call(this, cfg || s);
30383 };
30384 Roo.extend(Roo.Toolbar.Spacer, Roo.Toolbar.Item, {
30385     enable:Roo.emptyFn,
30386     disable:Roo.emptyFn,
30387     focus:Roo.emptyFn
30388 });
30389
30390 /**
30391  * @class Roo.Toolbar.Fill
30392  * @extends Roo.Toolbar.Spacer
30393  * A simple element that adds a greedy (100% width) horizontal space to a toolbar.
30394  * @constructor
30395  * Creates a new Spacer
30396  */
30397 Roo.Toolbar.Fill = Roo.extend(Roo.Toolbar.Spacer, {
30398     // private
30399     render : function(td){
30400         td.style.width = '100%';
30401         Roo.Toolbar.Fill.superclass.render.call(this, td);
30402     }
30403 });
30404
30405 /**
30406  * @class Roo.Toolbar.TextItem
30407  * @extends Roo.Toolbar.Item
30408  * A simple class that renders text directly into a toolbar.
30409  * @constructor
30410  * Creates a new TextItem
30411  * @param {String} text
30412  */
30413 Roo.Toolbar.TextItem = function(cfg){
30414     var  text = cfg || "";
30415     if (typeof(cfg) == 'object') {
30416         text = cfg.text || "";
30417     }  else {
30418         cfg = null;
30419     }
30420     var s = document.createElement("span");
30421     s.className = "ytb-text";
30422     s.innerHTML = text;
30423     if (cfg) {
30424         cfg.el  = s;
30425     }
30426     
30427     Roo.Toolbar.TextItem.superclass.constructor.call(this, cfg ||  s);
30428 };
30429 Roo.extend(Roo.Toolbar.TextItem, Roo.Toolbar.Item, {
30430     
30431      
30432     enable:Roo.emptyFn,
30433     disable:Roo.emptyFn,
30434     focus:Roo.emptyFn
30435 });
30436
30437 /**
30438  * @class Roo.Toolbar.Button
30439  * @extends Roo.Button
30440  * A button that renders into a toolbar.
30441  * @constructor
30442  * Creates a new Button
30443  * @param {Object} config A standard {@link Roo.Button} config object
30444  */
30445 Roo.Toolbar.Button = function(config){
30446     Roo.Toolbar.Button.superclass.constructor.call(this, null, config);
30447 };
30448 Roo.extend(Roo.Toolbar.Button, Roo.Button, {
30449     render : function(td){
30450         this.td = td;
30451         Roo.Toolbar.Button.superclass.render.call(this, td);
30452     },
30453     
30454     /**
30455      * Removes and destroys this button
30456      */
30457     destroy : function(){
30458         Roo.Toolbar.Button.superclass.destroy.call(this);
30459         this.td.parentNode.removeChild(this.td);
30460     },
30461     
30462     /**
30463      * Shows this button
30464      */
30465     show: function(){
30466         this.hidden = false;
30467         this.td.style.display = "";
30468     },
30469     
30470     /**
30471      * Hides this button
30472      */
30473     hide: function(){
30474         this.hidden = true;
30475         this.td.style.display = "none";
30476     },
30477
30478     /**
30479      * Disables this item
30480      */
30481     disable : function(){
30482         Roo.fly(this.td).addClass("x-item-disabled");
30483         this.disabled = true;
30484     },
30485
30486     /**
30487      * Enables this item
30488      */
30489     enable : function(){
30490         Roo.fly(this.td).removeClass("x-item-disabled");
30491         this.disabled = false;
30492     }
30493 });
30494 // backwards compat
30495 Roo.ToolbarButton = Roo.Toolbar.Button;
30496
30497 /**
30498  * @class Roo.Toolbar.SplitButton
30499  * @extends Roo.SplitButton
30500  * A menu button that renders into a toolbar.
30501  * @constructor
30502  * Creates a new SplitButton
30503  * @param {Object} config A standard {@link Roo.SplitButton} config object
30504  */
30505 Roo.Toolbar.SplitButton = function(config){
30506     Roo.Toolbar.SplitButton.superclass.constructor.call(this, null, config);
30507 };
30508 Roo.extend(Roo.Toolbar.SplitButton, Roo.SplitButton, {
30509     render : function(td){
30510         this.td = td;
30511         Roo.Toolbar.SplitButton.superclass.render.call(this, td);
30512     },
30513     
30514     /**
30515      * Removes and destroys this button
30516      */
30517     destroy : function(){
30518         Roo.Toolbar.SplitButton.superclass.destroy.call(this);
30519         this.td.parentNode.removeChild(this.td);
30520     },
30521     
30522     /**
30523      * Shows this button
30524      */
30525     show: function(){
30526         this.hidden = false;
30527         this.td.style.display = "";
30528     },
30529     
30530     /**
30531      * Hides this button
30532      */
30533     hide: function(){
30534         this.hidden = true;
30535         this.td.style.display = "none";
30536     }
30537 });
30538
30539 // backwards compat
30540 Roo.Toolbar.MenuButton = Roo.Toolbar.SplitButton;/*
30541  * Based on:
30542  * Ext JS Library 1.1.1
30543  * Copyright(c) 2006-2007, Ext JS, LLC.
30544  *
30545  * Originally Released Under LGPL - original licence link has changed is not relivant.
30546  *
30547  * Fork - LGPL
30548  * <script type="text/javascript">
30549  */
30550  
30551 /**
30552  * @class Roo.PagingToolbar
30553  * @extends Roo.Toolbar
30554  * A specialized toolbar that is bound to a {@link Roo.data.Store} and provides automatic paging controls.
30555  * @constructor
30556  * Create a new PagingToolbar
30557  * @param {Object} config The config object
30558  */
30559 Roo.PagingToolbar = function(el, ds, config)
30560 {
30561     // old args format still supported... - xtype is prefered..
30562     if (typeof(el) == 'object' && el.xtype) {
30563         // created from xtype...
30564         config = el;
30565         ds = el.dataSource;
30566         el = config.container;
30567     }
30568     var items = [];
30569     if (config.items) {
30570         items = config.items;
30571         config.items = [];
30572     }
30573     
30574     Roo.PagingToolbar.superclass.constructor.call(this, el, null, config);
30575     this.ds = ds;
30576     this.cursor = 0;
30577     this.renderButtons(this.el);
30578     this.bind(ds);
30579     
30580     // supprot items array.
30581    
30582     Roo.each(items, function(e) {
30583         this.add(Roo.factory(e));
30584     },this);
30585     
30586 };
30587
30588 Roo.extend(Roo.PagingToolbar, Roo.Toolbar, {
30589     /**
30590      * @cfg {Roo.data.Store} dataSource
30591      * The underlying data store providing the paged data
30592      */
30593     /**
30594      * @cfg {String/HTMLElement/Element} container
30595      * container The id or element that will contain the toolbar
30596      */
30597     /**
30598      * @cfg {Boolean} displayInfo
30599      * True to display the displayMsg (defaults to false)
30600      */
30601     /**
30602      * @cfg {Number} pageSize
30603      * The number of records to display per page (defaults to 20)
30604      */
30605     pageSize: 20,
30606     /**
30607      * @cfg {String} displayMsg
30608      * The paging status message to display (defaults to "Displaying {start} - {end} of {total}")
30609      */
30610     displayMsg : 'Displaying {0} - {1} of {2}',
30611     /**
30612      * @cfg {String} emptyMsg
30613      * The message to display when no records are found (defaults to "No data to display")
30614      */
30615     emptyMsg : 'No data to display',
30616     /**
30617      * Customizable piece of the default paging text (defaults to "Page")
30618      * @type String
30619      */
30620     beforePageText : "Page",
30621     /**
30622      * Customizable piece of the default paging text (defaults to "of %0")
30623      * @type String
30624      */
30625     afterPageText : "of {0}",
30626     /**
30627      * Customizable piece of the default paging text (defaults to "First Page")
30628      * @type String
30629      */
30630     firstText : "First Page",
30631     /**
30632      * Customizable piece of the default paging text (defaults to "Previous Page")
30633      * @type String
30634      */
30635     prevText : "Previous Page",
30636     /**
30637      * Customizable piece of the default paging text (defaults to "Next Page")
30638      * @type String
30639      */
30640     nextText : "Next Page",
30641     /**
30642      * Customizable piece of the default paging text (defaults to "Last Page")
30643      * @type String
30644      */
30645     lastText : "Last Page",
30646     /**
30647      * Customizable piece of the default paging text (defaults to "Refresh")
30648      * @type String
30649      */
30650     refreshText : "Refresh",
30651
30652     // private
30653     renderButtons : function(el){
30654         Roo.PagingToolbar.superclass.render.call(this, el);
30655         this.first = this.addButton({
30656             tooltip: this.firstText,
30657             cls: "x-btn-icon x-grid-page-first",
30658             disabled: true,
30659             handler: this.onClick.createDelegate(this, ["first"])
30660         });
30661         this.prev = this.addButton({
30662             tooltip: this.prevText,
30663             cls: "x-btn-icon x-grid-page-prev",
30664             disabled: true,
30665             handler: this.onClick.createDelegate(this, ["prev"])
30666         });
30667         //this.addSeparator();
30668         this.add(this.beforePageText);
30669         this.field = Roo.get(this.addDom({
30670            tag: "input",
30671            type: "text",
30672            size: "3",
30673            value: "1",
30674            cls: "x-grid-page-number"
30675         }).el);
30676         this.field.on("keydown", this.onPagingKeydown, this);
30677         this.field.on("focus", function(){this.dom.select();});
30678         this.afterTextEl = this.addText(String.format(this.afterPageText, 1));
30679         this.field.setHeight(18);
30680         //this.addSeparator();
30681         this.next = this.addButton({
30682             tooltip: this.nextText,
30683             cls: "x-btn-icon x-grid-page-next",
30684             disabled: true,
30685             handler: this.onClick.createDelegate(this, ["next"])
30686         });
30687         this.last = this.addButton({
30688             tooltip: this.lastText,
30689             cls: "x-btn-icon x-grid-page-last",
30690             disabled: true,
30691             handler: this.onClick.createDelegate(this, ["last"])
30692         });
30693         //this.addSeparator();
30694         this.loading = this.addButton({
30695             tooltip: this.refreshText,
30696             cls: "x-btn-icon x-grid-loading",
30697             handler: this.onClick.createDelegate(this, ["refresh"])
30698         });
30699
30700         if(this.displayInfo){
30701             this.displayEl = Roo.fly(this.el.dom.firstChild).createChild({cls:'x-paging-info'});
30702         }
30703     },
30704
30705     // private
30706     updateInfo : function(){
30707         if(this.displayEl){
30708             var count = this.ds.getCount();
30709             var msg = count == 0 ?
30710                 this.emptyMsg :
30711                 String.format(
30712                     this.displayMsg,
30713                     this.cursor+1, this.cursor+count, this.ds.getTotalCount()    
30714                 );
30715             this.displayEl.update(msg);
30716         }
30717     },
30718
30719     // private
30720     onLoad : function(ds, r, o){
30721        this.cursor = o.params ? o.params.start : 0;
30722        var d = this.getPageData(), ap = d.activePage, ps = d.pages;
30723
30724        this.afterTextEl.el.innerHTML = String.format(this.afterPageText, d.pages);
30725        this.field.dom.value = ap;
30726        this.first.setDisabled(ap == 1);
30727        this.prev.setDisabled(ap == 1);
30728        this.next.setDisabled(ap == ps);
30729        this.last.setDisabled(ap == ps);
30730        this.loading.enable();
30731        this.updateInfo();
30732     },
30733
30734     // private
30735     getPageData : function(){
30736         var total = this.ds.getTotalCount();
30737         return {
30738             total : total,
30739             activePage : Math.ceil((this.cursor+this.pageSize)/this.pageSize),
30740             pages :  total < this.pageSize ? 1 : Math.ceil(total/this.pageSize)
30741         };
30742     },
30743
30744     // private
30745     onLoadError : function(){
30746         this.loading.enable();
30747     },
30748
30749     // private
30750     onPagingKeydown : function(e){
30751         var k = e.getKey();
30752         var d = this.getPageData();
30753         if(k == e.RETURN){
30754             var v = this.field.dom.value, pageNum;
30755             if(!v || isNaN(pageNum = parseInt(v, 10))){
30756                 this.field.dom.value = d.activePage;
30757                 return;
30758             }
30759             pageNum = Math.min(Math.max(1, pageNum), d.pages) - 1;
30760             this.ds.load({params:{start: pageNum * this.pageSize, limit: this.pageSize}});
30761             e.stopEvent();
30762         }
30763         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))
30764         {
30765           var pageNum = (k == e.HOME || (k == e.DOWN && e.ctrlKey) || (k == e.LEFT && e.ctrlKey) || (k == e.PAGEDOWN && e.ctrlKey)) ? 1 : d.pages;
30766           this.field.dom.value = pageNum;
30767           this.ds.load({params:{start: (pageNum - 1) * this.pageSize, limit: this.pageSize}});
30768           e.stopEvent();
30769         }
30770         else if(k == e.UP || k == e.RIGHT || k == e.PAGEUP || k == e.DOWN || k == e.LEFT || k == e.PAGEDOWN)
30771         {
30772           var v = this.field.dom.value, pageNum; 
30773           var increment = (e.shiftKey) ? 10 : 1;
30774           if(k == e.DOWN || k == e.LEFT || k == e.PAGEDOWN) {
30775             increment *= -1;
30776           }
30777           if(!v || isNaN(pageNum = parseInt(v, 10))) {
30778             this.field.dom.value = d.activePage;
30779             return;
30780           }
30781           else if(parseInt(v, 10) + increment >= 1 & parseInt(v, 10) + increment <= d.pages)
30782           {
30783             this.field.dom.value = parseInt(v, 10) + increment;
30784             pageNum = Math.min(Math.max(1, pageNum + increment), d.pages) - 1;
30785             this.ds.load({params:{start: pageNum * this.pageSize, limit: this.pageSize}});
30786           }
30787           e.stopEvent();
30788         }
30789     },
30790
30791     // private
30792     beforeLoad : function(){
30793         if(this.loading){
30794             this.loading.disable();
30795         }
30796     },
30797
30798     // private
30799     onClick : function(which){
30800         var ds = this.ds;
30801         switch(which){
30802             case "first":
30803                 ds.load({params:{start: 0, limit: this.pageSize}});
30804             break;
30805             case "prev":
30806                 ds.load({params:{start: Math.max(0, this.cursor-this.pageSize), limit: this.pageSize}});
30807             break;
30808             case "next":
30809                 ds.load({params:{start: this.cursor+this.pageSize, limit: this.pageSize}});
30810             break;
30811             case "last":
30812                 var total = ds.getTotalCount();
30813                 var extra = total % this.pageSize;
30814                 var lastStart = extra ? (total - extra) : total-this.pageSize;
30815                 ds.load({params:{start: lastStart, limit: this.pageSize}});
30816             break;
30817             case "refresh":
30818                 ds.load({params:{start: this.cursor, limit: this.pageSize}});
30819             break;
30820         }
30821     },
30822
30823     /**
30824      * Unbinds the paging toolbar from the specified {@link Roo.data.Store}
30825      * @param {Roo.data.Store} store The data store to unbind
30826      */
30827     unbind : function(ds){
30828         ds.un("beforeload", this.beforeLoad, this);
30829         ds.un("load", this.onLoad, this);
30830         ds.un("loadexception", this.onLoadError, this);
30831         ds.un("remove", this.updateInfo, this);
30832         ds.un("add", this.updateInfo, this);
30833         this.ds = undefined;
30834     },
30835
30836     /**
30837      * Binds the paging toolbar to the specified {@link Roo.data.Store}
30838      * @param {Roo.data.Store} store The data store to bind
30839      */
30840     bind : function(ds){
30841         ds.on("beforeload", this.beforeLoad, this);
30842         ds.on("load", this.onLoad, this);
30843         ds.on("loadexception", this.onLoadError, this);
30844         ds.on("remove", this.updateInfo, this);
30845         ds.on("add", this.updateInfo, this);
30846         this.ds = ds;
30847     }
30848 });/*
30849  * Based on:
30850  * Ext JS Library 1.1.1
30851  * Copyright(c) 2006-2007, Ext JS, LLC.
30852  *
30853  * Originally Released Under LGPL - original licence link has changed is not relivant.
30854  *
30855  * Fork - LGPL
30856  * <script type="text/javascript">
30857  */
30858
30859 /**
30860  * @class Roo.Resizable
30861  * @extends Roo.util.Observable
30862  * <p>Applies drag handles to an element to make it resizable. The drag handles are inserted into the element
30863  * and positioned absolute. Some elements, such as a textarea or image, don't support this. To overcome that, you can wrap
30864  * 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
30865  * the element will be wrapped for you automatically.</p>
30866  * <p>Here is the list of valid resize handles:</p>
30867  * <pre>
30868 Value   Description
30869 ------  -------------------
30870  'n'     north
30871  's'     south
30872  'e'     east
30873  'w'     west
30874  'nw'    northwest
30875  'sw'    southwest
30876  'se'    southeast
30877  'ne'    northeast
30878  'hd'    horizontal drag
30879  'all'   all
30880 </pre>
30881  * <p>Here's an example showing the creation of a typical Resizable:</p>
30882  * <pre><code>
30883 var resizer = new Roo.Resizable("element-id", {
30884     handles: 'all',
30885     minWidth: 200,
30886     minHeight: 100,
30887     maxWidth: 500,
30888     maxHeight: 400,
30889     pinned: true
30890 });
30891 resizer.on("resize", myHandler);
30892 </code></pre>
30893  * <p>To hide a particular handle, set its display to none in CSS, or through script:<br>
30894  * resizer.east.setDisplayed(false);</p>
30895  * @cfg {Boolean/String/Element} resizeChild True to resize the first child, or id/element to resize (defaults to false)
30896  * @cfg {Array/String} adjustments String "auto" or an array [width, height] with values to be <b>added</b> to the
30897  * resize operation's new size (defaults to [0, 0])
30898  * @cfg {Number} minWidth The minimum width for the element (defaults to 5)
30899  * @cfg {Number} minHeight The minimum height for the element (defaults to 5)
30900  * @cfg {Number} maxWidth The maximum width for the element (defaults to 10000)
30901  * @cfg {Number} maxHeight The maximum height for the element (defaults to 10000)
30902  * @cfg {Boolean} enabled False to disable resizing (defaults to true)
30903  * @cfg {Boolean} wrap True to wrap an element with a div if needed (required for textareas and images, defaults to false)
30904  * @cfg {Number} width The width of the element in pixels (defaults to null)
30905  * @cfg {Number} height The height of the element in pixels (defaults to null)
30906  * @cfg {Boolean} animate True to animate the resize (not compatible with dynamic sizing, defaults to false)
30907  * @cfg {Number} duration Animation duration if animate = true (defaults to .35)
30908  * @cfg {Boolean} dynamic True to resize the element while dragging instead of using a proxy (defaults to false)
30909  * @cfg {String} handles String consisting of the resize handles to display (defaults to undefined)
30910  * @cfg {Boolean} multiDirectional <b>Deprecated</b>.  The old style of adding multi-direction resize handles, deprecated
30911  * in favor of the handles config option (defaults to false)
30912  * @cfg {Boolean} disableTrackOver True to disable mouse tracking. This is only applied at config time. (defaults to false)
30913  * @cfg {String} easing Animation easing if animate = true (defaults to 'easingOutStrong')
30914  * @cfg {Number} widthIncrement The increment to snap the width resize in pixels (dynamic must be true, defaults to 0)
30915  * @cfg {Number} heightIncrement The increment to snap the height resize in pixels (dynamic must be true, defaults to 0)
30916  * @cfg {Boolean} pinned True to ensure that the resize handles are always visible, false to display them only when the
30917  * user mouses over the resizable borders. This is only applied at config time. (defaults to false)
30918  * @cfg {Boolean} preserveRatio True to preserve the original ratio between height and width during resize (defaults to false)
30919  * @cfg {Boolean} transparent True for transparent handles. This is only applied at config time. (defaults to false)
30920  * @cfg {Number} minX The minimum allowed page X for the element (only used for west resizing, defaults to 0)
30921  * @cfg {Number} minY The minimum allowed page Y for the element (only used for north resizing, defaults to 0)
30922  * @cfg {Boolean} draggable Convenience to initialize drag drop (defaults to false)
30923  * @constructor
30924  * Create a new resizable component
30925  * @param {String/HTMLElement/Roo.Element} el The id or element to resize
30926  * @param {Object} config configuration options
30927   */
30928 Roo.Resizable = function(el, config)
30929 {
30930     this.el = Roo.get(el);
30931
30932     if(config && config.wrap){
30933         config.resizeChild = this.el;
30934         this.el = this.el.wrap(typeof config.wrap == "object" ? config.wrap : {cls:"xresizable-wrap"});
30935         this.el.id = this.el.dom.id = config.resizeChild.id + "-rzwrap";
30936         this.el.setStyle("overflow", "hidden");
30937         this.el.setPositioning(config.resizeChild.getPositioning());
30938         config.resizeChild.clearPositioning();
30939         if(!config.width || !config.height){
30940             var csize = config.resizeChild.getSize();
30941             this.el.setSize(csize.width, csize.height);
30942         }
30943         if(config.pinned && !config.adjustments){
30944             config.adjustments = "auto";
30945         }
30946     }
30947
30948     this.proxy = this.el.createProxy({tag: "div", cls: "x-resizable-proxy", id: this.el.id + "-rzproxy"});
30949     this.proxy.unselectable();
30950     this.proxy.enableDisplayMode('block');
30951
30952     Roo.apply(this, config);
30953
30954     if(this.pinned){
30955         this.disableTrackOver = true;
30956         this.el.addClass("x-resizable-pinned");
30957     }
30958     // if the element isn't positioned, make it relative
30959     var position = this.el.getStyle("position");
30960     if(position != "absolute" && position != "fixed"){
30961         this.el.setStyle("position", "relative");
30962     }
30963     if(!this.handles){ // no handles passed, must be legacy style
30964         this.handles = 's,e,se';
30965         if(this.multiDirectional){
30966             this.handles += ',n,w';
30967         }
30968     }
30969     if(this.handles == "all"){
30970         this.handles = "n s e w ne nw se sw";
30971     }
30972     var hs = this.handles.split(/\s*?[,;]\s*?| /);
30973     var ps = Roo.Resizable.positions;
30974     for(var i = 0, len = hs.length; i < len; i++){
30975         if(hs[i] && ps[hs[i]]){
30976             var pos = ps[hs[i]];
30977             this[pos] = new Roo.Resizable.Handle(this, pos, this.disableTrackOver, this.transparent);
30978         }
30979     }
30980     // legacy
30981     this.corner = this.southeast;
30982     
30983     // updateBox = the box can move..
30984     if(this.handles.indexOf("n") != -1 || this.handles.indexOf("w") != -1 || this.handles.indexOf("hd") != -1) {
30985         this.updateBox = true;
30986     }
30987
30988     this.activeHandle = null;
30989
30990     if(this.resizeChild){
30991         if(typeof this.resizeChild == "boolean"){
30992             this.resizeChild = Roo.get(this.el.dom.firstChild, true);
30993         }else{
30994             this.resizeChild = Roo.get(this.resizeChild, true);
30995         }
30996     }
30997     
30998     if(this.adjustments == "auto"){
30999         var rc = this.resizeChild;
31000         var hw = this.west, he = this.east, hn = this.north, hs = this.south;
31001         if(rc && (hw || hn)){
31002             rc.position("relative");
31003             rc.setLeft(hw ? hw.el.getWidth() : 0);
31004             rc.setTop(hn ? hn.el.getHeight() : 0);
31005         }
31006         this.adjustments = [
31007             (he ? -he.el.getWidth() : 0) + (hw ? -hw.el.getWidth() : 0),
31008             (hn ? -hn.el.getHeight() : 0) + (hs ? -hs.el.getHeight() : 0) -1
31009         ];
31010     }
31011
31012     if(this.draggable){
31013         this.dd = this.dynamic ?
31014             this.el.initDD(null) : this.el.initDDProxy(null, {dragElId: this.proxy.id});
31015         this.dd.setHandleElId(this.resizeChild ? this.resizeChild.id : this.el.id);
31016     }
31017
31018     // public events
31019     this.addEvents({
31020         /**
31021          * @event beforeresize
31022          * Fired before resize is allowed. Set enabled to false to cancel resize.
31023          * @param {Roo.Resizable} this
31024          * @param {Roo.EventObject} e The mousedown event
31025          */
31026         "beforeresize" : true,
31027         /**
31028          * @event resizing
31029          * Fired a resizing.
31030          * @param {Roo.Resizable} this
31031          * @param {Number} x The new x position
31032          * @param {Number} y The new y position
31033          * @param {Number} w The new w width
31034          * @param {Number} h The new h hight
31035          * @param {Roo.EventObject} e The mouseup event
31036          */
31037         "resizing" : true,
31038         /**
31039          * @event resize
31040          * Fired after a resize.
31041          * @param {Roo.Resizable} this
31042          * @param {Number} width The new width
31043          * @param {Number} height The new height
31044          * @param {Roo.EventObject} e The mouseup event
31045          */
31046         "resize" : true
31047     });
31048
31049     if(this.width !== null && this.height !== null){
31050         this.resizeTo(this.width, this.height);
31051     }else{
31052         this.updateChildSize();
31053     }
31054     if(Roo.isIE){
31055         this.el.dom.style.zoom = 1;
31056     }
31057     Roo.Resizable.superclass.constructor.call(this);
31058 };
31059
31060 Roo.extend(Roo.Resizable, Roo.util.Observable, {
31061         resizeChild : false,
31062         adjustments : [0, 0],
31063         minWidth : 5,
31064         minHeight : 5,
31065         maxWidth : 10000,
31066         maxHeight : 10000,
31067         enabled : true,
31068         animate : false,
31069         duration : .35,
31070         dynamic : false,
31071         handles : false,
31072         multiDirectional : false,
31073         disableTrackOver : false,
31074         easing : 'easeOutStrong',
31075         widthIncrement : 0,
31076         heightIncrement : 0,
31077         pinned : false,
31078         width : null,
31079         height : null,
31080         preserveRatio : false,
31081         transparent: false,
31082         minX: 0,
31083         minY: 0,
31084         draggable: false,
31085
31086         /**
31087          * @cfg {String/HTMLElement/Element} constrainTo Constrain the resize to a particular element
31088          */
31089         constrainTo: undefined,
31090         /**
31091          * @cfg {Roo.lib.Region} resizeRegion Constrain the resize to a particular region
31092          */
31093         resizeRegion: undefined,
31094
31095
31096     /**
31097      * Perform a manual resize
31098      * @param {Number} width
31099      * @param {Number} height
31100      */
31101     resizeTo : function(width, height){
31102         this.el.setSize(width, height);
31103         this.updateChildSize();
31104         this.fireEvent("resize", this, width, height, null);
31105     },
31106
31107     // private
31108     startSizing : function(e, handle){
31109         this.fireEvent("beforeresize", this, e);
31110         if(this.enabled){ // 2nd enabled check in case disabled before beforeresize handler
31111
31112             if(!this.overlay){
31113                 this.overlay = this.el.createProxy({tag: "div", cls: "x-resizable-overlay", html: "&#160;"});
31114                 this.overlay.unselectable();
31115                 this.overlay.enableDisplayMode("block");
31116                 this.overlay.on("mousemove", this.onMouseMove, this);
31117                 this.overlay.on("mouseup", this.onMouseUp, this);
31118             }
31119             this.overlay.setStyle("cursor", handle.el.getStyle("cursor"));
31120
31121             this.resizing = true;
31122             this.startBox = this.el.getBox();
31123             this.startPoint = e.getXY();
31124             this.offsets = [(this.startBox.x + this.startBox.width) - this.startPoint[0],
31125                             (this.startBox.y + this.startBox.height) - this.startPoint[1]];
31126
31127             this.overlay.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
31128             this.overlay.show();
31129
31130             if(this.constrainTo) {
31131                 var ct = Roo.get(this.constrainTo);
31132                 this.resizeRegion = ct.getRegion().adjust(
31133                     ct.getFrameWidth('t'),
31134                     ct.getFrameWidth('l'),
31135                     -ct.getFrameWidth('b'),
31136                     -ct.getFrameWidth('r')
31137                 );
31138             }
31139
31140             this.proxy.setStyle('visibility', 'hidden'); // workaround display none
31141             this.proxy.show();
31142             this.proxy.setBox(this.startBox);
31143             if(!this.dynamic){
31144                 this.proxy.setStyle('visibility', 'visible');
31145             }
31146         }
31147     },
31148
31149     // private
31150     onMouseDown : function(handle, e){
31151         if(this.enabled){
31152             e.stopEvent();
31153             this.activeHandle = handle;
31154             this.startSizing(e, handle);
31155         }
31156     },
31157
31158     // private
31159     onMouseUp : function(e){
31160         var size = this.resizeElement();
31161         this.resizing = false;
31162         this.handleOut();
31163         this.overlay.hide();
31164         this.proxy.hide();
31165         this.fireEvent("resize", this, size.width, size.height, e);
31166     },
31167
31168     // private
31169     updateChildSize : function(){
31170         
31171         if(this.resizeChild){
31172             var el = this.el;
31173             var child = this.resizeChild;
31174             var adj = this.adjustments;
31175             if(el.dom.offsetWidth){
31176                 var b = el.getSize(true);
31177                 child.setSize(b.width+adj[0], b.height+adj[1]);
31178             }
31179             // Second call here for IE
31180             // The first call enables instant resizing and
31181             // the second call corrects scroll bars if they
31182             // exist
31183             if(Roo.isIE){
31184                 setTimeout(function(){
31185                     if(el.dom.offsetWidth){
31186                         var b = el.getSize(true);
31187                         child.setSize(b.width+adj[0], b.height+adj[1]);
31188                     }
31189                 }, 10);
31190             }
31191         }
31192     },
31193
31194     // private
31195     snap : function(value, inc, min){
31196         if(!inc || !value) {
31197             return value;
31198         }
31199         var newValue = value;
31200         var m = value % inc;
31201         if(m > 0){
31202             if(m > (inc/2)){
31203                 newValue = value + (inc-m);
31204             }else{
31205                 newValue = value - m;
31206             }
31207         }
31208         return Math.max(min, newValue);
31209     },
31210
31211     // private
31212     resizeElement : function(){
31213         var box = this.proxy.getBox();
31214         if(this.updateBox){
31215             this.el.setBox(box, false, this.animate, this.duration, null, this.easing);
31216         }else{
31217             this.el.setSize(box.width, box.height, this.animate, this.duration, null, this.easing);
31218         }
31219         this.updateChildSize();
31220         if(!this.dynamic){
31221             this.proxy.hide();
31222         }
31223         return box;
31224     },
31225
31226     // private
31227     constrain : function(v, diff, m, mx){
31228         if(v - diff < m){
31229             diff = v - m;
31230         }else if(v - diff > mx){
31231             diff = mx - v;
31232         }
31233         return diff;
31234     },
31235
31236     // private
31237     onMouseMove : function(e){
31238         
31239         if(this.enabled){
31240             try{// try catch so if something goes wrong the user doesn't get hung
31241
31242             if(this.resizeRegion && !this.resizeRegion.contains(e.getPoint())) {
31243                 return;
31244             }
31245
31246             //var curXY = this.startPoint;
31247             var curSize = this.curSize || this.startBox;
31248             var x = this.startBox.x, y = this.startBox.y;
31249             var ox = x, oy = y;
31250             var w = curSize.width, h = curSize.height;
31251             var ow = w, oh = h;
31252             var mw = this.minWidth, mh = this.minHeight;
31253             var mxw = this.maxWidth, mxh = this.maxHeight;
31254             var wi = this.widthIncrement;
31255             var hi = this.heightIncrement;
31256
31257             var eventXY = e.getXY();
31258             var diffX = -(this.startPoint[0] - Math.max(this.minX, eventXY[0]));
31259             var diffY = -(this.startPoint[1] - Math.max(this.minY, eventXY[1]));
31260
31261             var pos = this.activeHandle.position;
31262
31263             switch(pos){
31264                 case "east":
31265                     w += diffX;
31266                     w = Math.min(Math.max(mw, w), mxw);
31267                     break;
31268              
31269                 case "south":
31270                     h += diffY;
31271                     h = Math.min(Math.max(mh, h), mxh);
31272                     break;
31273                 case "southeast":
31274                     w += diffX;
31275                     h += diffY;
31276                     w = Math.min(Math.max(mw, w), mxw);
31277                     h = Math.min(Math.max(mh, h), mxh);
31278                     break;
31279                 case "north":
31280                     diffY = this.constrain(h, diffY, mh, mxh);
31281                     y += diffY;
31282                     h -= diffY;
31283                     break;
31284                 case "hdrag":
31285                     
31286                     if (wi) {
31287                         var adiffX = Math.abs(diffX);
31288                         var sub = (adiffX % wi); // how much 
31289                         if (sub > (wi/2)) { // far enough to snap
31290                             diffX = (diffX > 0) ? diffX-sub + wi : diffX+sub - wi;
31291                         } else {
31292                             // remove difference.. 
31293                             diffX = (diffX > 0) ? diffX-sub : diffX+sub;
31294                         }
31295                     }
31296                     x += diffX;
31297                     x = Math.max(this.minX, x);
31298                     break;
31299                 case "west":
31300                     diffX = this.constrain(w, diffX, mw, mxw);
31301                     x += diffX;
31302                     w -= diffX;
31303                     break;
31304                 case "northeast":
31305                     w += diffX;
31306                     w = Math.min(Math.max(mw, w), mxw);
31307                     diffY = this.constrain(h, diffY, mh, mxh);
31308                     y += diffY;
31309                     h -= diffY;
31310                     break;
31311                 case "northwest":
31312                     diffX = this.constrain(w, diffX, mw, mxw);
31313                     diffY = this.constrain(h, diffY, mh, mxh);
31314                     y += diffY;
31315                     h -= diffY;
31316                     x += diffX;
31317                     w -= diffX;
31318                     break;
31319                case "southwest":
31320                     diffX = this.constrain(w, diffX, mw, mxw);
31321                     h += diffY;
31322                     h = Math.min(Math.max(mh, h), mxh);
31323                     x += diffX;
31324                     w -= diffX;
31325                     break;
31326             }
31327
31328             var sw = this.snap(w, wi, mw);
31329             var sh = this.snap(h, hi, mh);
31330             if(sw != w || sh != h){
31331                 switch(pos){
31332                     case "northeast":
31333                         y -= sh - h;
31334                     break;
31335                     case "north":
31336                         y -= sh - h;
31337                         break;
31338                     case "southwest":
31339                         x -= sw - w;
31340                     break;
31341                     case "west":
31342                         x -= sw - w;
31343                         break;
31344                     case "northwest":
31345                         x -= sw - w;
31346                         y -= sh - h;
31347                     break;
31348                 }
31349                 w = sw;
31350                 h = sh;
31351             }
31352
31353             if(this.preserveRatio){
31354                 switch(pos){
31355                     case "southeast":
31356                     case "east":
31357                         h = oh * (w/ow);
31358                         h = Math.min(Math.max(mh, h), mxh);
31359                         w = ow * (h/oh);
31360                        break;
31361                     case "south":
31362                         w = ow * (h/oh);
31363                         w = Math.min(Math.max(mw, w), mxw);
31364                         h = oh * (w/ow);
31365                         break;
31366                     case "northeast":
31367                         w = ow * (h/oh);
31368                         w = Math.min(Math.max(mw, w), mxw);
31369                         h = oh * (w/ow);
31370                     break;
31371                     case "north":
31372                         var tw = w;
31373                         w = ow * (h/oh);
31374                         w = Math.min(Math.max(mw, w), mxw);
31375                         h = oh * (w/ow);
31376                         x += (tw - w) / 2;
31377                         break;
31378                     case "southwest":
31379                         h = oh * (w/ow);
31380                         h = Math.min(Math.max(mh, h), mxh);
31381                         var tw = w;
31382                         w = ow * (h/oh);
31383                         x += tw - w;
31384                         break;
31385                     case "west":
31386                         var th = h;
31387                         h = oh * (w/ow);
31388                         h = Math.min(Math.max(mh, h), mxh);
31389                         y += (th - h) / 2;
31390                         var tw = w;
31391                         w = ow * (h/oh);
31392                         x += tw - w;
31393                        break;
31394                     case "northwest":
31395                         var tw = w;
31396                         var th = h;
31397                         h = oh * (w/ow);
31398                         h = Math.min(Math.max(mh, h), mxh);
31399                         w = ow * (h/oh);
31400                         y += th - h;
31401                         x += tw - w;
31402                        break;
31403
31404                 }
31405             }
31406             if (pos == 'hdrag') {
31407                 w = ow;
31408             }
31409             this.proxy.setBounds(x, y, w, h);
31410             if(this.dynamic){
31411                 this.resizeElement();
31412             }
31413             }catch(e){}
31414         }
31415         this.fireEvent("resizing", this, x, y, w, h, e);
31416     },
31417
31418     // private
31419     handleOver : function(){
31420         if(this.enabled){
31421             this.el.addClass("x-resizable-over");
31422         }
31423     },
31424
31425     // private
31426     handleOut : function(){
31427         if(!this.resizing){
31428             this.el.removeClass("x-resizable-over");
31429         }
31430     },
31431
31432     /**
31433      * Returns the element this component is bound to.
31434      * @return {Roo.Element}
31435      */
31436     getEl : function(){
31437         return this.el;
31438     },
31439
31440     /**
31441      * Returns the resizeChild element (or null).
31442      * @return {Roo.Element}
31443      */
31444     getResizeChild : function(){
31445         return this.resizeChild;
31446     },
31447     groupHandler : function()
31448     {
31449         
31450     },
31451     /**
31452      * Destroys this resizable. If the element was wrapped and
31453      * removeEl is not true then the element remains.
31454      * @param {Boolean} removeEl (optional) true to remove the element from the DOM
31455      */
31456     destroy : function(removeEl){
31457         this.proxy.remove();
31458         if(this.overlay){
31459             this.overlay.removeAllListeners();
31460             this.overlay.remove();
31461         }
31462         var ps = Roo.Resizable.positions;
31463         for(var k in ps){
31464             if(typeof ps[k] != "function" && this[ps[k]]){
31465                 var h = this[ps[k]];
31466                 h.el.removeAllListeners();
31467                 h.el.remove();
31468             }
31469         }
31470         if(removeEl){
31471             this.el.update("");
31472             this.el.remove();
31473         }
31474     }
31475 });
31476
31477 // private
31478 // hash to map config positions to true positions
31479 Roo.Resizable.positions = {
31480     n: "north", s: "south", e: "east", w: "west", se: "southeast", sw: "southwest", nw: "northwest", ne: "northeast", 
31481     hd: "hdrag"
31482 };
31483
31484 // private
31485 Roo.Resizable.Handle = function(rz, pos, disableTrackOver, transparent){
31486     if(!this.tpl){
31487         // only initialize the template if resizable is used
31488         var tpl = Roo.DomHelper.createTemplate(
31489             {tag: "div", cls: "x-resizable-handle x-resizable-handle-{0}"}
31490         );
31491         tpl.compile();
31492         Roo.Resizable.Handle.prototype.tpl = tpl;
31493     }
31494     this.position = pos;
31495     this.rz = rz;
31496     // show north drag fro topdra
31497     var handlepos = pos == 'hdrag' ? 'north' : pos;
31498     
31499     this.el = this.tpl.append(rz.el.dom, [handlepos], true);
31500     if (pos == 'hdrag') {
31501         this.el.setStyle('cursor', 'pointer');
31502     }
31503     this.el.unselectable();
31504     if(transparent){
31505         this.el.setOpacity(0);
31506     }
31507     this.el.on("mousedown", this.onMouseDown, this);
31508     if(!disableTrackOver){
31509         this.el.on("mouseover", this.onMouseOver, this);
31510         this.el.on("mouseout", this.onMouseOut, this);
31511     }
31512 };
31513
31514 // private
31515 Roo.Resizable.Handle.prototype = {
31516     afterResize : function(rz){
31517         Roo.log('after?');
31518         // do nothing
31519     },
31520     // private
31521     onMouseDown : function(e){
31522         this.rz.onMouseDown(this, e);
31523     },
31524     // private
31525     onMouseOver : function(e){
31526         this.rz.handleOver(this, e);
31527     },
31528     // private
31529     onMouseOut : function(e){
31530         this.rz.handleOut(this, e);
31531     }
31532 };/*
31533  * Based on:
31534  * Ext JS Library 1.1.1
31535  * Copyright(c) 2006-2007, Ext JS, LLC.
31536  *
31537  * Originally Released Under LGPL - original licence link has changed is not relivant.
31538  *
31539  * Fork - LGPL
31540  * <script type="text/javascript">
31541  */
31542
31543 /**
31544  * @class Roo.Editor
31545  * @extends Roo.Component
31546  * A base editor field that handles displaying/hiding on demand and has some built-in sizing and event handling logic.
31547  * @constructor
31548  * Create a new Editor
31549  * @param {Roo.form.Field} field The Field object (or descendant)
31550  * @param {Object} config The config object
31551  */
31552 Roo.Editor = function(field, config){
31553     Roo.Editor.superclass.constructor.call(this, config);
31554     this.field = field;
31555     this.addEvents({
31556         /**
31557              * @event beforestartedit
31558              * Fires when editing is initiated, but before the value changes.  Editing can be canceled by returning
31559              * false from the handler of this event.
31560              * @param {Editor} this
31561              * @param {Roo.Element} boundEl The underlying element bound to this editor
31562              * @param {Mixed} value The field value being set
31563              */
31564         "beforestartedit" : true,
31565         /**
31566              * @event startedit
31567              * Fires when this editor is displayed
31568              * @param {Roo.Element} boundEl The underlying element bound to this editor
31569              * @param {Mixed} value The starting field value
31570              */
31571         "startedit" : true,
31572         /**
31573              * @event beforecomplete
31574              * Fires after a change has been made to the field, but before the change is reflected in the underlying
31575              * field.  Saving the change to the field can be canceled by returning false from the handler of this event.
31576              * Note that if the value has not changed and ignoreNoChange = true, the editing will still end but this
31577              * event will not fire since no edit actually occurred.
31578              * @param {Editor} this
31579              * @param {Mixed} value The current field value
31580              * @param {Mixed} startValue The original field value
31581              */
31582         "beforecomplete" : true,
31583         /**
31584              * @event complete
31585              * Fires after editing is complete and any changed value has been written to the underlying field.
31586              * @param {Editor} this
31587              * @param {Mixed} value The current field value
31588              * @param {Mixed} startValue The original field value
31589              */
31590         "complete" : true,
31591         /**
31592          * @event specialkey
31593          * Fires when any key related to navigation (arrows, tab, enter, esc, etc.) is pressed.  You can check
31594          * {@link Roo.EventObject#getKey} to determine which key was pressed.
31595          * @param {Roo.form.Field} this
31596          * @param {Roo.EventObject} e The event object
31597          */
31598         "specialkey" : true
31599     });
31600 };
31601
31602 Roo.extend(Roo.Editor, Roo.Component, {
31603     /**
31604      * @cfg {Boolean/String} autosize
31605      * True for the editor to automatically adopt the size of the underlying field, "width" to adopt the width only,
31606      * or "height" to adopt the height only (defaults to false)
31607      */
31608     /**
31609      * @cfg {Boolean} revertInvalid
31610      * True to automatically revert the field value and cancel the edit when the user completes an edit and the field
31611      * validation fails (defaults to true)
31612      */
31613     /**
31614      * @cfg {Boolean} ignoreNoChange
31615      * True to skip the the edit completion process (no save, no events fired) if the user completes an edit and
31616      * the value has not changed (defaults to false).  Applies only to string values - edits for other data types
31617      * will never be ignored.
31618      */
31619     /**
31620      * @cfg {Boolean} hideEl
31621      * False to keep the bound element visible while the editor is displayed (defaults to true)
31622      */
31623     /**
31624      * @cfg {Mixed} value
31625      * The data value of the underlying field (defaults to "")
31626      */
31627     value : "",
31628     /**
31629      * @cfg {String} alignment
31630      * The position to align to (see {@link Roo.Element#alignTo} for more details, defaults to "c-c?").
31631      */
31632     alignment: "c-c?",
31633     /**
31634      * @cfg {Boolean/String} shadow "sides" for sides/bottom only, "frame" for 4-way shadow, and "drop"
31635      * for bottom-right shadow (defaults to "frame")
31636      */
31637     shadow : "frame",
31638     /**
31639      * @cfg {Boolean} constrain True to constrain the editor to the viewport
31640      */
31641     constrain : false,
31642     /**
31643      * @cfg {Boolean} completeOnEnter True to complete the edit when the enter key is pressed (defaults to false)
31644      */
31645     completeOnEnter : false,
31646     /**
31647      * @cfg {Boolean} cancelOnEsc True to cancel the edit when the escape key is pressed (defaults to false)
31648      */
31649     cancelOnEsc : false,
31650     /**
31651      * @cfg {Boolean} updateEl True to update the innerHTML of the bound element when the update completes (defaults to false)
31652      */
31653     updateEl : false,
31654
31655     // private
31656     onRender : function(ct, position){
31657         this.el = new Roo.Layer({
31658             shadow: this.shadow,
31659             cls: "x-editor",
31660             parentEl : ct,
31661             shim : this.shim,
31662             shadowOffset:4,
31663             id: this.id,
31664             constrain: this.constrain
31665         });
31666         this.el.setStyle("overflow", Roo.isGecko ? "auto" : "hidden");
31667         if(this.field.msgTarget != 'title'){
31668             this.field.msgTarget = 'qtip';
31669         }
31670         this.field.render(this.el);
31671         if(Roo.isGecko){
31672             this.field.el.dom.setAttribute('autocomplete', 'off');
31673         }
31674         this.field.on("specialkey", this.onSpecialKey, this);
31675         if(this.swallowKeys){
31676             this.field.el.swallowEvent(['keydown','keypress']);
31677         }
31678         this.field.show();
31679         this.field.on("blur", this.onBlur, this);
31680         if(this.field.grow){
31681             this.field.on("autosize", this.el.sync,  this.el, {delay:1});
31682         }
31683     },
31684
31685     onSpecialKey : function(field, e)
31686     {
31687         //Roo.log('editor onSpecialKey');
31688         if(this.completeOnEnter && e.getKey() == e.ENTER){
31689             e.stopEvent();
31690             this.completeEdit();
31691             return;
31692         }
31693         // do not fire special key otherwise it might hide close the editor...
31694         if(e.getKey() == e.ENTER){    
31695             return;
31696         }
31697         if(this.cancelOnEsc && e.getKey() == e.ESC){
31698             this.cancelEdit();
31699             return;
31700         } 
31701         this.fireEvent('specialkey', field, e);
31702     
31703     },
31704
31705     /**
31706      * Starts the editing process and shows the editor.
31707      * @param {String/HTMLElement/Element} el The element to edit
31708      * @param {String} value (optional) A value to initialize the editor with. If a value is not provided, it defaults
31709       * to the innerHTML of el.
31710      */
31711     startEdit : function(el, value){
31712         if(this.editing){
31713             this.completeEdit();
31714         }
31715         this.boundEl = Roo.get(el);
31716         var v = value !== undefined ? value : this.boundEl.dom.innerHTML;
31717         if(!this.rendered){
31718             this.render(this.parentEl || document.body);
31719         }
31720         if(this.fireEvent("beforestartedit", this, this.boundEl, v) === false){
31721             return;
31722         }
31723         this.startValue = v;
31724         this.field.setValue(v);
31725         if(this.autoSize){
31726             var sz = this.boundEl.getSize();
31727             switch(this.autoSize){
31728                 case "width":
31729                 this.setSize(sz.width,  "");
31730                 break;
31731                 case "height":
31732                 this.setSize("",  sz.height);
31733                 break;
31734                 default:
31735                 this.setSize(sz.width,  sz.height);
31736             }
31737         }
31738         this.el.alignTo(this.boundEl, this.alignment);
31739         this.editing = true;
31740         if(Roo.QuickTips){
31741             Roo.QuickTips.disable();
31742         }
31743         this.show();
31744     },
31745
31746     /**
31747      * Sets the height and width of this editor.
31748      * @param {Number} width The new width
31749      * @param {Number} height The new height
31750      */
31751     setSize : function(w, h){
31752         this.field.setSize(w, h);
31753         if(this.el){
31754             this.el.sync();
31755         }
31756     },
31757
31758     /**
31759      * Realigns the editor to the bound field based on the current alignment config value.
31760      */
31761     realign : function(){
31762         this.el.alignTo(this.boundEl, this.alignment);
31763     },
31764
31765     /**
31766      * Ends the editing process, persists the changed value to the underlying field, and hides the editor.
31767      * @param {Boolean} remainVisible Override the default behavior and keep the editor visible after edit (defaults to false)
31768      */
31769     completeEdit : function(remainVisible){
31770         if(!this.editing){
31771             return;
31772         }
31773         var v = this.getValue();
31774         if(this.revertInvalid !== false && !this.field.isValid()){
31775             v = this.startValue;
31776             this.cancelEdit(true);
31777         }
31778         if(String(v) === String(this.startValue) && this.ignoreNoChange){
31779             this.editing = false;
31780             this.hide();
31781             return;
31782         }
31783         if(this.fireEvent("beforecomplete", this, v, this.startValue) !== false){
31784             this.editing = false;
31785             if(this.updateEl && this.boundEl){
31786                 this.boundEl.update(v);
31787             }
31788             if(remainVisible !== true){
31789                 this.hide();
31790             }
31791             this.fireEvent("complete", this, v, this.startValue);
31792         }
31793     },
31794
31795     // private
31796     onShow : function(){
31797         this.el.show();
31798         if(this.hideEl !== false){
31799             this.boundEl.hide();
31800         }
31801         this.field.show();
31802         if(Roo.isIE && !this.fixIEFocus){ // IE has problems with focusing the first time
31803             this.fixIEFocus = true;
31804             this.deferredFocus.defer(50, this);
31805         }else{
31806             this.field.focus();
31807         }
31808         this.fireEvent("startedit", this.boundEl, this.startValue);
31809     },
31810
31811     deferredFocus : function(){
31812         if(this.editing){
31813             this.field.focus();
31814         }
31815     },
31816
31817     /**
31818      * Cancels the editing process and hides the editor without persisting any changes.  The field value will be
31819      * reverted to the original starting value.
31820      * @param {Boolean} remainVisible Override the default behavior and keep the editor visible after
31821      * cancel (defaults to false)
31822      */
31823     cancelEdit : function(remainVisible){
31824         if(this.editing){
31825             this.setValue(this.startValue);
31826             if(remainVisible !== true){
31827                 this.hide();
31828             }
31829         }
31830     },
31831
31832     // private
31833     onBlur : function(){
31834         if(this.allowBlur !== true && this.editing){
31835             this.completeEdit();
31836         }
31837     },
31838
31839     // private
31840     onHide : function(){
31841         if(this.editing){
31842             this.completeEdit();
31843             return;
31844         }
31845         this.field.blur();
31846         if(this.field.collapse){
31847             this.field.collapse();
31848         }
31849         this.el.hide();
31850         if(this.hideEl !== false){
31851             this.boundEl.show();
31852         }
31853         if(Roo.QuickTips){
31854             Roo.QuickTips.enable();
31855         }
31856     },
31857
31858     /**
31859      * Sets the data value of the editor
31860      * @param {Mixed} value Any valid value supported by the underlying field
31861      */
31862     setValue : function(v){
31863         this.field.setValue(v);
31864     },
31865
31866     /**
31867      * Gets the data value of the editor
31868      * @return {Mixed} The data value
31869      */
31870     getValue : function(){
31871         return this.field.getValue();
31872     }
31873 });/*
31874  * Based on:
31875  * Ext JS Library 1.1.1
31876  * Copyright(c) 2006-2007, Ext JS, LLC.
31877  *
31878  * Originally Released Under LGPL - original licence link has changed is not relivant.
31879  *
31880  * Fork - LGPL
31881  * <script type="text/javascript">
31882  */
31883  
31884 /**
31885  * @class Roo.BasicDialog
31886  * @extends Roo.util.Observable
31887  * Lightweight Dialog Class.  The code below shows the creation of a typical dialog using existing HTML markup:
31888  * <pre><code>
31889 var dlg = new Roo.BasicDialog("my-dlg", {
31890     height: 200,
31891     width: 300,
31892     minHeight: 100,
31893     minWidth: 150,
31894     modal: true,
31895     proxyDrag: true,
31896     shadow: true
31897 });
31898 dlg.addKeyListener(27, dlg.hide, dlg); // ESC can also close the dialog
31899 dlg.addButton('OK', dlg.hide, dlg);    // Could call a save function instead of hiding
31900 dlg.addButton('Cancel', dlg.hide, dlg);
31901 dlg.show();
31902 </code></pre>
31903   <b>A Dialog should always be a direct child of the body element.</b>
31904  * @cfg {Boolean/DomHelper} autoCreate True to auto create from scratch, or using a DomHelper Object (defaults to false)
31905  * @cfg {String} title Default text to display in the title bar (defaults to null)
31906  * @cfg {Number} width Width of the dialog in pixels (can also be set via CSS).  Determined by browser if unspecified.
31907  * @cfg {Number} height Height of the dialog in pixels (can also be set via CSS).  Determined by browser if unspecified.
31908  * @cfg {Number} x The default left page coordinate of the dialog (defaults to center screen)
31909  * @cfg {Number} y The default top page coordinate of the dialog (defaults to center screen)
31910  * @cfg {String/Element} animateTarget Id or element from which the dialog should animate while opening
31911  * (defaults to null with no animation)
31912  * @cfg {Boolean} resizable False to disable manual dialog resizing (defaults to true)
31913  * @cfg {String} resizeHandles Which resize handles to display - see the {@link Roo.Resizable} handles config
31914  * property for valid values (defaults to 'all')
31915  * @cfg {Number} minHeight The minimum allowable height for a resizable dialog (defaults to 80)
31916  * @cfg {Number} minWidth The minimum allowable width for a resizable dialog (defaults to 200)
31917  * @cfg {Boolean} modal True to show the dialog modally, preventing user interaction with the rest of the page (defaults to false)
31918  * @cfg {Boolean} autoScroll True to allow the dialog body contents to overflow and display scrollbars (defaults to false)
31919  * @cfg {Boolean} closable False to remove the built-in top-right corner close button (defaults to true)
31920  * @cfg {Boolean} collapsible False to remove the built-in top-right corner collapse button (defaults to true)
31921  * @cfg {Boolean} constraintoviewport True to keep the dialog constrained within the visible viewport boundaries (defaults to true)
31922  * @cfg {Boolean} syncHeightBeforeShow True to cause the dimensions to be recalculated before the dialog is shown (defaults to false)
31923  * @cfg {Boolean} draggable False to disable dragging of the dialog within the viewport (defaults to true)
31924  * @cfg {Boolean} autoTabs If true, all elements with class 'x-dlg-tab' will get automatically converted to tabs (defaults to false)
31925  * @cfg {String} tabTag The tag name of tab elements, used when autoTabs = true (defaults to 'div')
31926  * @cfg {Boolean} proxyDrag True to drag a lightweight proxy element rather than the dialog itself, used when
31927  * draggable = true (defaults to false)
31928  * @cfg {Boolean} fixedcenter True to ensure that anytime the dialog is shown or resized it gets centered (defaults to false)
31929  * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
31930  * shadow (defaults to false)
31931  * @cfg {Number} shadowOffset The number of pixels to offset the shadow if displayed (defaults to 5)
31932  * @cfg {String} buttonAlign Valid values are "left," "center" and "right" (defaults to "right")
31933  * @cfg {Number} minButtonWidth Minimum width of all dialog buttons (defaults to 75)
31934  * @cfg {Array} buttons Array of buttons
31935  * @cfg {Boolean} shim True to create an iframe shim that prevents selects from showing through (defaults to false)
31936  * @constructor
31937  * Create a new BasicDialog.
31938  * @param {String/HTMLElement/Roo.Element} el The container element or DOM node, or its id
31939  * @param {Object} config Configuration options
31940  */
31941 Roo.BasicDialog = function(el, config){
31942     this.el = Roo.get(el);
31943     var dh = Roo.DomHelper;
31944     if(!this.el && config && config.autoCreate){
31945         if(typeof config.autoCreate == "object"){
31946             if(!config.autoCreate.id){
31947                 config.autoCreate.id = el;
31948             }
31949             this.el = dh.append(document.body,
31950                         config.autoCreate, true);
31951         }else{
31952             this.el = dh.append(document.body,
31953                         {tag: "div", id: el, style:'visibility:hidden;'}, true);
31954         }
31955     }
31956     el = this.el;
31957     el.setDisplayed(true);
31958     el.hide = this.hideAction;
31959     this.id = el.id;
31960     el.addClass("x-dlg");
31961
31962     Roo.apply(this, config);
31963
31964     this.proxy = el.createProxy("x-dlg-proxy");
31965     this.proxy.hide = this.hideAction;
31966     this.proxy.setOpacity(.5);
31967     this.proxy.hide();
31968
31969     if(config.width){
31970         el.setWidth(config.width);
31971     }
31972     if(config.height){
31973         el.setHeight(config.height);
31974     }
31975     this.size = el.getSize();
31976     if(typeof config.x != "undefined" && typeof config.y != "undefined"){
31977         this.xy = [config.x,config.y];
31978     }else{
31979         this.xy = el.getCenterXY(true);
31980     }
31981     /** The header element @type Roo.Element */
31982     this.header = el.child("> .x-dlg-hd");
31983     /** The body element @type Roo.Element */
31984     this.body = el.child("> .x-dlg-bd");
31985     /** The footer element @type Roo.Element */
31986     this.footer = el.child("> .x-dlg-ft");
31987
31988     if(!this.header){
31989         this.header = el.createChild({tag: "div", cls:"x-dlg-hd", html: "&#160;"}, this.body ? this.body.dom : null);
31990     }
31991     if(!this.body){
31992         this.body = el.createChild({tag: "div", cls:"x-dlg-bd"});
31993     }
31994
31995     this.header.unselectable();
31996     if(this.title){
31997         this.header.update(this.title);
31998     }
31999     // this element allows the dialog to be focused for keyboard event
32000     this.focusEl = el.createChild({tag: "a", href:"#", cls:"x-dlg-focus", tabIndex:"-1"});
32001     this.focusEl.swallowEvent("click", true);
32002
32003     this.header.wrap({cls:"x-dlg-hd-right"}).wrap({cls:"x-dlg-hd-left"}, true);
32004
32005     // wrap the body and footer for special rendering
32006     this.bwrap = this.body.wrap({tag: "div", cls:"x-dlg-dlg-body"});
32007     if(this.footer){
32008         this.bwrap.dom.appendChild(this.footer.dom);
32009     }
32010
32011     this.bg = this.el.createChild({
32012         tag: "div", cls:"x-dlg-bg",
32013         html: '<div class="x-dlg-bg-left"><div class="x-dlg-bg-right"><div class="x-dlg-bg-center">&#160;</div></div></div>'
32014     });
32015     this.centerBg = this.bg.child("div.x-dlg-bg-center");
32016
32017
32018     if(this.autoScroll !== false && !this.autoTabs){
32019         this.body.setStyle("overflow", "auto");
32020     }
32021
32022     this.toolbox = this.el.createChild({cls: "x-dlg-toolbox"});
32023
32024     if(this.closable !== false){
32025         this.el.addClass("x-dlg-closable");
32026         this.close = this.toolbox.createChild({cls:"x-dlg-close"});
32027         this.close.on("click", this.closeClick, this);
32028         this.close.addClassOnOver("x-dlg-close-over");
32029     }
32030     if(this.collapsible !== false){
32031         this.collapseBtn = this.toolbox.createChild({cls:"x-dlg-collapse"});
32032         this.collapseBtn.on("click", this.collapseClick, this);
32033         this.collapseBtn.addClassOnOver("x-dlg-collapse-over");
32034         this.header.on("dblclick", this.collapseClick, this);
32035     }
32036     if(this.resizable !== false){
32037         this.el.addClass("x-dlg-resizable");
32038         this.resizer = new Roo.Resizable(el, {
32039             minWidth: this.minWidth || 80,
32040             minHeight:this.minHeight || 80,
32041             handles: this.resizeHandles || "all",
32042             pinned: true
32043         });
32044         this.resizer.on("beforeresize", this.beforeResize, this);
32045         this.resizer.on("resize", this.onResize, this);
32046     }
32047     if(this.draggable !== false){
32048         el.addClass("x-dlg-draggable");
32049         if (!this.proxyDrag) {
32050             var dd = new Roo.dd.DD(el.dom.id, "WindowDrag");
32051         }
32052         else {
32053             var dd = new Roo.dd.DDProxy(el.dom.id, "WindowDrag", {dragElId: this.proxy.id});
32054         }
32055         dd.setHandleElId(this.header.id);
32056         dd.endDrag = this.endMove.createDelegate(this);
32057         dd.startDrag = this.startMove.createDelegate(this);
32058         dd.onDrag = this.onDrag.createDelegate(this);
32059         dd.scroll = false;
32060         this.dd = dd;
32061     }
32062     if(this.modal){
32063         this.mask = dh.append(document.body, {tag: "div", cls:"x-dlg-mask"}, true);
32064         this.mask.enableDisplayMode("block");
32065         this.mask.hide();
32066         this.el.addClass("x-dlg-modal");
32067     }
32068     if(this.shadow){
32069         this.shadow = new Roo.Shadow({
32070             mode : typeof this.shadow == "string" ? this.shadow : "sides",
32071             offset : this.shadowOffset
32072         });
32073     }else{
32074         this.shadowOffset = 0;
32075     }
32076     if(Roo.useShims && this.shim !== false){
32077         this.shim = this.el.createShim();
32078         this.shim.hide = this.hideAction;
32079         this.shim.hide();
32080     }else{
32081         this.shim = false;
32082     }
32083     if(this.autoTabs){
32084         this.initTabs();
32085     }
32086     if (this.buttons) { 
32087         var bts= this.buttons;
32088         this.buttons = [];
32089         Roo.each(bts, function(b) {
32090             this.addButton(b);
32091         }, this);
32092     }
32093     
32094     
32095     this.addEvents({
32096         /**
32097          * @event keydown
32098          * Fires when a key is pressed
32099          * @param {Roo.BasicDialog} this
32100          * @param {Roo.EventObject} e
32101          */
32102         "keydown" : true,
32103         /**
32104          * @event move
32105          * Fires when this dialog is moved by the user.
32106          * @param {Roo.BasicDialog} this
32107          * @param {Number} x The new page X
32108          * @param {Number} y The new page Y
32109          */
32110         "move" : true,
32111         /**
32112          * @event resize
32113          * Fires when this dialog is resized by the user.
32114          * @param {Roo.BasicDialog} this
32115          * @param {Number} width The new width
32116          * @param {Number} height The new height
32117          */
32118         "resize" : true,
32119         /**
32120          * @event beforehide
32121          * Fires before this dialog is hidden.
32122          * @param {Roo.BasicDialog} this
32123          */
32124         "beforehide" : true,
32125         /**
32126          * @event hide
32127          * Fires when this dialog is hidden.
32128          * @param {Roo.BasicDialog} this
32129          */
32130         "hide" : true,
32131         /**
32132          * @event beforeshow
32133          * Fires before this dialog is shown.
32134          * @param {Roo.BasicDialog} this
32135          */
32136         "beforeshow" : true,
32137         /**
32138          * @event show
32139          * Fires when this dialog is shown.
32140          * @param {Roo.BasicDialog} this
32141          */
32142         "show" : true
32143     });
32144     el.on("keydown", this.onKeyDown, this);
32145     el.on("mousedown", this.toFront, this);
32146     Roo.EventManager.onWindowResize(this.adjustViewport, this, true);
32147     this.el.hide();
32148     Roo.DialogManager.register(this);
32149     Roo.BasicDialog.superclass.constructor.call(this);
32150 };
32151
32152 Roo.extend(Roo.BasicDialog, Roo.util.Observable, {
32153     shadowOffset: Roo.isIE ? 6 : 5,
32154     minHeight: 80,
32155     minWidth: 200,
32156     minButtonWidth: 75,
32157     defaultButton: null,
32158     buttonAlign: "right",
32159     tabTag: 'div',
32160     firstShow: true,
32161
32162     /**
32163      * Sets the dialog title text
32164      * @param {String} text The title text to display
32165      * @return {Roo.BasicDialog} this
32166      */
32167     setTitle : function(text){
32168         this.header.update(text);
32169         return this;
32170     },
32171
32172     // private
32173     closeClick : function(){
32174         this.hide();
32175     },
32176
32177     // private
32178     collapseClick : function(){
32179         this[this.collapsed ? "expand" : "collapse"]();
32180     },
32181
32182     /**
32183      * Collapses the dialog to its minimized state (only the title bar is visible).
32184      * Equivalent to the user clicking the collapse dialog button.
32185      */
32186     collapse : function(){
32187         if(!this.collapsed){
32188             this.collapsed = true;
32189             this.el.addClass("x-dlg-collapsed");
32190             this.restoreHeight = this.el.getHeight();
32191             this.resizeTo(this.el.getWidth(), this.header.getHeight());
32192         }
32193     },
32194
32195     /**
32196      * Expands a collapsed dialog back to its normal state.  Equivalent to the user
32197      * clicking the expand dialog button.
32198      */
32199     expand : function(){
32200         if(this.collapsed){
32201             this.collapsed = false;
32202             this.el.removeClass("x-dlg-collapsed");
32203             this.resizeTo(this.el.getWidth(), this.restoreHeight);
32204         }
32205     },
32206
32207     /**
32208      * Reinitializes the tabs component, clearing out old tabs and finding new ones.
32209      * @return {Roo.TabPanel} The tabs component
32210      */
32211     initTabs : function(){
32212         var tabs = this.getTabs();
32213         while(tabs.getTab(0)){
32214             tabs.removeTab(0);
32215         }
32216         this.el.select(this.tabTag+'.x-dlg-tab').each(function(el){
32217             var dom = el.dom;
32218             tabs.addTab(Roo.id(dom), dom.title);
32219             dom.title = "";
32220         });
32221         tabs.activate(0);
32222         return tabs;
32223     },
32224
32225     // private
32226     beforeResize : function(){
32227         this.resizer.minHeight = Math.max(this.minHeight, this.getHeaderFooterHeight(true)+40);
32228     },
32229
32230     // private
32231     onResize : function(){
32232         this.refreshSize();
32233         this.syncBodyHeight();
32234         this.adjustAssets();
32235         this.focus();
32236         this.fireEvent("resize", this, this.size.width, this.size.height);
32237     },
32238
32239     // private
32240     onKeyDown : function(e){
32241         if(this.isVisible()){
32242             this.fireEvent("keydown", this, e);
32243         }
32244     },
32245
32246     /**
32247      * Resizes the dialog.
32248      * @param {Number} width
32249      * @param {Number} height
32250      * @return {Roo.BasicDialog} this
32251      */
32252     resizeTo : function(width, height){
32253         this.el.setSize(width, height);
32254         this.size = {width: width, height: height};
32255         this.syncBodyHeight();
32256         if(this.fixedcenter){
32257             this.center();
32258         }
32259         if(this.isVisible()){
32260             this.constrainXY();
32261             this.adjustAssets();
32262         }
32263         this.fireEvent("resize", this, width, height);
32264         return this;
32265     },
32266
32267
32268     /**
32269      * Resizes the dialog to fit the specified content size.
32270      * @param {Number} width
32271      * @param {Number} height
32272      * @return {Roo.BasicDialog} this
32273      */
32274     setContentSize : function(w, h){
32275         h += this.getHeaderFooterHeight() + this.body.getMargins("tb");
32276         w += this.body.getMargins("lr") + this.bwrap.getMargins("lr") + this.centerBg.getPadding("lr");
32277         //if(!this.el.isBorderBox()){
32278             h +=  this.body.getPadding("tb") + this.bwrap.getBorderWidth("tb") + this.body.getBorderWidth("tb") + this.el.getBorderWidth("tb");
32279             w += this.body.getPadding("lr") + this.bwrap.getBorderWidth("lr") + this.body.getBorderWidth("lr") + this.bwrap.getPadding("lr") + this.el.getBorderWidth("lr");
32280         //}
32281         if(this.tabs){
32282             h += this.tabs.stripWrap.getHeight() + this.tabs.bodyEl.getMargins("tb") + this.tabs.bodyEl.getPadding("tb");
32283             w += this.tabs.bodyEl.getMargins("lr") + this.tabs.bodyEl.getPadding("lr");
32284         }
32285         this.resizeTo(w, h);
32286         return this;
32287     },
32288
32289     /**
32290      * Adds a key listener for when this dialog is displayed.  This allows you to hook in a function that will be
32291      * executed in response to a particular key being pressed while the dialog is active.
32292      * @param {Number/Array/Object} key Either the numeric key code, array of key codes or an object with the following options:
32293      *                                  {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
32294      * @param {Function} fn The function to call
32295      * @param {Object} scope (optional) The scope of the function
32296      * @return {Roo.BasicDialog} this
32297      */
32298     addKeyListener : function(key, fn, scope){
32299         var keyCode, shift, ctrl, alt;
32300         if(typeof key == "object" && !(key instanceof Array)){
32301             keyCode = key["key"];
32302             shift = key["shift"];
32303             ctrl = key["ctrl"];
32304             alt = key["alt"];
32305         }else{
32306             keyCode = key;
32307         }
32308         var handler = function(dlg, e){
32309             if((!shift || e.shiftKey) && (!ctrl || e.ctrlKey) &&  (!alt || e.altKey)){
32310                 var k = e.getKey();
32311                 if(keyCode instanceof Array){
32312                     for(var i = 0, len = keyCode.length; i < len; i++){
32313                         if(keyCode[i] == k){
32314                           fn.call(scope || window, dlg, k, e);
32315                           return;
32316                         }
32317                     }
32318                 }else{
32319                     if(k == keyCode){
32320                         fn.call(scope || window, dlg, k, e);
32321                     }
32322                 }
32323             }
32324         };
32325         this.on("keydown", handler);
32326         return this;
32327     },
32328
32329     /**
32330      * Returns the TabPanel component (creates it if it doesn't exist).
32331      * Note: If you wish to simply check for the existence of tabs without creating them,
32332      * check for a null 'tabs' property.
32333      * @return {Roo.TabPanel} The tabs component
32334      */
32335     getTabs : function(){
32336         if(!this.tabs){
32337             this.el.addClass("x-dlg-auto-tabs");
32338             this.body.addClass(this.tabPosition == "bottom" ? "x-tabs-bottom" : "x-tabs-top");
32339             this.tabs = new Roo.TabPanel(this.body.dom, this.tabPosition == "bottom");
32340         }
32341         return this.tabs;
32342     },
32343
32344     /**
32345      * Adds a button to the footer section of the dialog.
32346      * @param {String/Object} config A string becomes the button text, an object can either be a Button config
32347      * object or a valid Roo.DomHelper element config
32348      * @param {Function} handler The function called when the button is clicked
32349      * @param {Object} scope (optional) The scope of the handler function (accepts position as a property)
32350      * @return {Roo.Button} The new button
32351      */
32352     addButton : function(config, handler, scope){
32353         var dh = Roo.DomHelper;
32354         if(!this.footer){
32355             this.footer = dh.append(this.bwrap, {tag: "div", cls:"x-dlg-ft"}, true);
32356         }
32357         if(!this.btnContainer){
32358             var tb = this.footer.createChild({
32359
32360                 cls:"x-dlg-btns x-dlg-btns-"+this.buttonAlign,
32361                 html:'<table cellspacing="0"><tbody><tr></tr></tbody></table><div class="x-clear"></div>'
32362             }, null, true);
32363             this.btnContainer = tb.firstChild.firstChild.firstChild;
32364         }
32365         var bconfig = {
32366             handler: handler,
32367             scope: scope,
32368             minWidth: this.minButtonWidth,
32369             hideParent:true
32370         };
32371         if(typeof config == "string"){
32372             bconfig.text = config;
32373         }else{
32374             if(config.tag){
32375                 bconfig.dhconfig = config;
32376             }else{
32377                 Roo.apply(bconfig, config);
32378             }
32379         }
32380         var fc = false;
32381         if ((typeof(bconfig.position) != 'undefined') && bconfig.position < this.btnContainer.childNodes.length-1) {
32382             bconfig.position = Math.max(0, bconfig.position);
32383             fc = this.btnContainer.childNodes[bconfig.position];
32384         }
32385          
32386         var btn = new Roo.Button(
32387             fc ? 
32388                 this.btnContainer.insertBefore(document.createElement("td"),fc)
32389                 : this.btnContainer.appendChild(document.createElement("td")),
32390             //Roo.get(this.btnContainer).createChild( { tag: 'td'},  fc ),
32391             bconfig
32392         );
32393         this.syncBodyHeight();
32394         if(!this.buttons){
32395             /**
32396              * Array of all the buttons that have been added to this dialog via addButton
32397              * @type Array
32398              */
32399             this.buttons = [];
32400         }
32401         this.buttons.push(btn);
32402         return btn;
32403     },
32404
32405     /**
32406      * Sets the default button to be focused when the dialog is displayed.
32407      * @param {Roo.BasicDialog.Button} btn The button object returned by {@link #addButton}
32408      * @return {Roo.BasicDialog} this
32409      */
32410     setDefaultButton : function(btn){
32411         this.defaultButton = btn;
32412         return this;
32413     },
32414
32415     // private
32416     getHeaderFooterHeight : function(safe){
32417         var height = 0;
32418         if(this.header){
32419            height += this.header.getHeight();
32420         }
32421         if(this.footer){
32422            var fm = this.footer.getMargins();
32423             height += (this.footer.getHeight()+fm.top+fm.bottom);
32424         }
32425         height += this.bwrap.getPadding("tb")+this.bwrap.getBorderWidth("tb");
32426         height += this.centerBg.getPadding("tb");
32427         return height;
32428     },
32429
32430     // private
32431     syncBodyHeight : function()
32432     {
32433         var bd = this.body, // the text
32434             cb = this.centerBg, // wrapper around bottom.. but does not seem to be used..
32435             bw = this.bwrap;
32436         var height = this.size.height - this.getHeaderFooterHeight(false);
32437         bd.setHeight(height-bd.getMargins("tb"));
32438         var hh = this.header.getHeight();
32439         var h = this.size.height-hh;
32440         cb.setHeight(h);
32441         
32442         bw.setLeftTop(cb.getPadding("l"), hh+cb.getPadding("t"));
32443         bw.setHeight(h-cb.getPadding("tb"));
32444         
32445         bw.setWidth(this.el.getWidth(true)-cb.getPadding("lr"));
32446         bd.setWidth(bw.getWidth(true));
32447         if(this.tabs){
32448             this.tabs.syncHeight();
32449             if(Roo.isIE){
32450                 this.tabs.el.repaint();
32451             }
32452         }
32453     },
32454
32455     /**
32456      * Restores the previous state of the dialog if Roo.state is configured.
32457      * @return {Roo.BasicDialog} this
32458      */
32459     restoreState : function(){
32460         var box = Roo.state.Manager.get(this.stateId || (this.el.id + "-state"));
32461         if(box && box.width){
32462             this.xy = [box.x, box.y];
32463             this.resizeTo(box.width, box.height);
32464         }
32465         return this;
32466     },
32467
32468     // private
32469     beforeShow : function(){
32470         this.expand();
32471         if(this.fixedcenter){
32472             this.xy = this.el.getCenterXY(true);
32473         }
32474         if(this.modal){
32475             Roo.get(document.body).addClass("x-body-masked");
32476             this.mask.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
32477             this.mask.show();
32478         }
32479         this.constrainXY();
32480     },
32481
32482     // private
32483     animShow : function(){
32484         var b = Roo.get(this.animateTarget).getBox();
32485         this.proxy.setSize(b.width, b.height);
32486         this.proxy.setLocation(b.x, b.y);
32487         this.proxy.show();
32488         this.proxy.setBounds(this.xy[0], this.xy[1], this.size.width, this.size.height,
32489                     true, .35, this.showEl.createDelegate(this));
32490     },
32491
32492     /**
32493      * Shows the dialog.
32494      * @param {String/HTMLElement/Roo.Element} animateTarget (optional) Reset the animation target
32495      * @return {Roo.BasicDialog} this
32496      */
32497     show : function(animateTarget){
32498         if (this.fireEvent("beforeshow", this) === false){
32499             return;
32500         }
32501         if(this.syncHeightBeforeShow){
32502             this.syncBodyHeight();
32503         }else if(this.firstShow){
32504             this.firstShow = false;
32505             this.syncBodyHeight(); // sync the height on the first show instead of in the constructor
32506         }
32507         this.animateTarget = animateTarget || this.animateTarget;
32508         if(!this.el.isVisible()){
32509             this.beforeShow();
32510             if(this.animateTarget && Roo.get(this.animateTarget)){
32511                 this.animShow();
32512             }else{
32513                 this.showEl();
32514             }
32515         }
32516         return this;
32517     },
32518
32519     // private
32520     showEl : function(){
32521         this.proxy.hide();
32522         this.el.setXY(this.xy);
32523         this.el.show();
32524         this.adjustAssets(true);
32525         this.toFront();
32526         this.focus();
32527         // IE peekaboo bug - fix found by Dave Fenwick
32528         if(Roo.isIE){
32529             this.el.repaint();
32530         }
32531         this.fireEvent("show", this);
32532     },
32533
32534     /**
32535      * Focuses the dialog.  If a defaultButton is set, it will receive focus, otherwise the
32536      * dialog itself will receive focus.
32537      */
32538     focus : function(){
32539         if(this.defaultButton){
32540             this.defaultButton.focus();
32541         }else{
32542             this.focusEl.focus();
32543         }
32544     },
32545
32546     // private
32547     constrainXY : function(){
32548         if(this.constraintoviewport !== false){
32549             if(!this.viewSize){
32550                 if(this.container){
32551                     var s = this.container.getSize();
32552                     this.viewSize = [s.width, s.height];
32553                 }else{
32554                     this.viewSize = [Roo.lib.Dom.getViewWidth(),Roo.lib.Dom.getViewHeight()];
32555                 }
32556             }
32557             var s = Roo.get(this.container||document).getScroll();
32558
32559             var x = this.xy[0], y = this.xy[1];
32560             var w = this.size.width, h = this.size.height;
32561             var vw = this.viewSize[0], vh = this.viewSize[1];
32562             // only move it if it needs it
32563             var moved = false;
32564             // first validate right/bottom
32565             if(x + w > vw+s.left){
32566                 x = vw - w;
32567                 moved = true;
32568             }
32569             if(y + h > vh+s.top){
32570                 y = vh - h;
32571                 moved = true;
32572             }
32573             // then make sure top/left isn't negative
32574             if(x < s.left){
32575                 x = s.left;
32576                 moved = true;
32577             }
32578             if(y < s.top){
32579                 y = s.top;
32580                 moved = true;
32581             }
32582             if(moved){
32583                 // cache xy
32584                 this.xy = [x, y];
32585                 if(this.isVisible()){
32586                     this.el.setLocation(x, y);
32587                     this.adjustAssets();
32588                 }
32589             }
32590         }
32591     },
32592
32593     // private
32594     onDrag : function(){
32595         if(!this.proxyDrag){
32596             this.xy = this.el.getXY();
32597             this.adjustAssets();
32598         }
32599     },
32600
32601     // private
32602     adjustAssets : function(doShow){
32603         var x = this.xy[0], y = this.xy[1];
32604         var w = this.size.width, h = this.size.height;
32605         if(doShow === true){
32606             if(this.shadow){
32607                 this.shadow.show(this.el);
32608             }
32609             if(this.shim){
32610                 this.shim.show();
32611             }
32612         }
32613         if(this.shadow && this.shadow.isVisible()){
32614             this.shadow.show(this.el);
32615         }
32616         if(this.shim && this.shim.isVisible()){
32617             this.shim.setBounds(x, y, w, h);
32618         }
32619     },
32620
32621     // private
32622     adjustViewport : function(w, h){
32623         if(!w || !h){
32624             w = Roo.lib.Dom.getViewWidth();
32625             h = Roo.lib.Dom.getViewHeight();
32626         }
32627         // cache the size
32628         this.viewSize = [w, h];
32629         if(this.modal && this.mask.isVisible()){
32630             this.mask.setSize(w, h); // first make sure the mask isn't causing overflow
32631             this.mask.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
32632         }
32633         if(this.isVisible()){
32634             this.constrainXY();
32635         }
32636     },
32637
32638     /**
32639      * Destroys this dialog and all its supporting elements (including any tabs, shim,
32640      * shadow, proxy, mask, etc.)  Also removes all event listeners.
32641      * @param {Boolean} removeEl (optional) true to remove the element from the DOM
32642      */
32643     destroy : function(removeEl){
32644         if(this.isVisible()){
32645             this.animateTarget = null;
32646             this.hide();
32647         }
32648         Roo.EventManager.removeResizeListener(this.adjustViewport, this);
32649         if(this.tabs){
32650             this.tabs.destroy(removeEl);
32651         }
32652         Roo.destroy(
32653              this.shim,
32654              this.proxy,
32655              this.resizer,
32656              this.close,
32657              this.mask
32658         );
32659         if(this.dd){
32660             this.dd.unreg();
32661         }
32662         if(this.buttons){
32663            for(var i = 0, len = this.buttons.length; i < len; i++){
32664                this.buttons[i].destroy();
32665            }
32666         }
32667         this.el.removeAllListeners();
32668         if(removeEl === true){
32669             this.el.update("");
32670             this.el.remove();
32671         }
32672         Roo.DialogManager.unregister(this);
32673     },
32674
32675     // private
32676     startMove : function(){
32677         if(this.proxyDrag){
32678             this.proxy.show();
32679         }
32680         if(this.constraintoviewport !== false){
32681             this.dd.constrainTo(document.body, {right: this.shadowOffset, bottom: this.shadowOffset});
32682         }
32683     },
32684
32685     // private
32686     endMove : function(){
32687         if(!this.proxyDrag){
32688             Roo.dd.DD.prototype.endDrag.apply(this.dd, arguments);
32689         }else{
32690             Roo.dd.DDProxy.prototype.endDrag.apply(this.dd, arguments);
32691             this.proxy.hide();
32692         }
32693         this.refreshSize();
32694         this.adjustAssets();
32695         this.focus();
32696         this.fireEvent("move", this, this.xy[0], this.xy[1]);
32697     },
32698
32699     /**
32700      * Brings this dialog to the front of any other visible dialogs
32701      * @return {Roo.BasicDialog} this
32702      */
32703     toFront : function(){
32704         Roo.DialogManager.bringToFront(this);
32705         return this;
32706     },
32707
32708     /**
32709      * Sends this dialog to the back (under) of any other visible dialogs
32710      * @return {Roo.BasicDialog} this
32711      */
32712     toBack : function(){
32713         Roo.DialogManager.sendToBack(this);
32714         return this;
32715     },
32716
32717     /**
32718      * Centers this dialog in the viewport
32719      * @return {Roo.BasicDialog} this
32720      */
32721     center : function(){
32722         var xy = this.el.getCenterXY(true);
32723         this.moveTo(xy[0], xy[1]);
32724         return this;
32725     },
32726
32727     /**
32728      * Moves the dialog's top-left corner to the specified point
32729      * @param {Number} x
32730      * @param {Number} y
32731      * @return {Roo.BasicDialog} this
32732      */
32733     moveTo : function(x, y){
32734         this.xy = [x,y];
32735         if(this.isVisible()){
32736             this.el.setXY(this.xy);
32737             this.adjustAssets();
32738         }
32739         return this;
32740     },
32741
32742     /**
32743      * Aligns the dialog to the specified element
32744      * @param {String/HTMLElement/Roo.Element} element The element to align to.
32745      * @param {String} position The position to align to (see {@link Roo.Element#alignTo} for more details).
32746      * @param {Array} offsets (optional) Offset the positioning by [x, y]
32747      * @return {Roo.BasicDialog} this
32748      */
32749     alignTo : function(element, position, offsets){
32750         this.xy = this.el.getAlignToXY(element, position, offsets);
32751         if(this.isVisible()){
32752             this.el.setXY(this.xy);
32753             this.adjustAssets();
32754         }
32755         return this;
32756     },
32757
32758     /**
32759      * Anchors an element to another element and realigns it when the window is resized.
32760      * @param {String/HTMLElement/Roo.Element} element The element to align to.
32761      * @param {String} position The position to align to (see {@link Roo.Element#alignTo} for more details)
32762      * @param {Array} offsets (optional) Offset the positioning by [x, y]
32763      * @param {Boolean/Number} monitorScroll (optional) true to monitor body scroll and reposition. If this parameter
32764      * is a number, it is used as the buffer delay (defaults to 50ms).
32765      * @return {Roo.BasicDialog} this
32766      */
32767     anchorTo : function(el, alignment, offsets, monitorScroll){
32768         var action = function(){
32769             this.alignTo(el, alignment, offsets);
32770         };
32771         Roo.EventManager.onWindowResize(action, this);
32772         var tm = typeof monitorScroll;
32773         if(tm != 'undefined'){
32774             Roo.EventManager.on(window, 'scroll', action, this,
32775                 {buffer: tm == 'number' ? monitorScroll : 50});
32776         }
32777         action.call(this);
32778         return this;
32779     },
32780
32781     /**
32782      * Returns true if the dialog is visible
32783      * @return {Boolean}
32784      */
32785     isVisible : function(){
32786         return this.el.isVisible();
32787     },
32788
32789     // private
32790     animHide : function(callback){
32791         var b = Roo.get(this.animateTarget).getBox();
32792         this.proxy.show();
32793         this.proxy.setBounds(this.xy[0], this.xy[1], this.size.width, this.size.height);
32794         this.el.hide();
32795         this.proxy.setBounds(b.x, b.y, b.width, b.height, true, .35,
32796                     this.hideEl.createDelegate(this, [callback]));
32797     },
32798
32799     /**
32800      * Hides the dialog.
32801      * @param {Function} callback (optional) Function to call when the dialog is hidden
32802      * @return {Roo.BasicDialog} this
32803      */
32804     hide : function(callback){
32805         if (this.fireEvent("beforehide", this) === false){
32806             return;
32807         }
32808         if(this.shadow){
32809             this.shadow.hide();
32810         }
32811         if(this.shim) {
32812           this.shim.hide();
32813         }
32814         // sometimes animateTarget seems to get set.. causing problems...
32815         // this just double checks..
32816         if(this.animateTarget && Roo.get(this.animateTarget)) {
32817            this.animHide(callback);
32818         }else{
32819             this.el.hide();
32820             this.hideEl(callback);
32821         }
32822         return this;
32823     },
32824
32825     // private
32826     hideEl : function(callback){
32827         this.proxy.hide();
32828         if(this.modal){
32829             this.mask.hide();
32830             Roo.get(document.body).removeClass("x-body-masked");
32831         }
32832         this.fireEvent("hide", this);
32833         if(typeof callback == "function"){
32834             callback();
32835         }
32836     },
32837
32838     // private
32839     hideAction : function(){
32840         this.setLeft("-10000px");
32841         this.setTop("-10000px");
32842         this.setStyle("visibility", "hidden");
32843     },
32844
32845     // private
32846     refreshSize : function(){
32847         this.size = this.el.getSize();
32848         this.xy = this.el.getXY();
32849         Roo.state.Manager.set(this.stateId || this.el.id + "-state", this.el.getBox());
32850     },
32851
32852     // private
32853     // z-index is managed by the DialogManager and may be overwritten at any time
32854     setZIndex : function(index){
32855         if(this.modal){
32856             this.mask.setStyle("z-index", index);
32857         }
32858         if(this.shim){
32859             this.shim.setStyle("z-index", ++index);
32860         }
32861         if(this.shadow){
32862             this.shadow.setZIndex(++index);
32863         }
32864         this.el.setStyle("z-index", ++index);
32865         if(this.proxy){
32866             this.proxy.setStyle("z-index", ++index);
32867         }
32868         if(this.resizer){
32869             this.resizer.proxy.setStyle("z-index", ++index);
32870         }
32871
32872         this.lastZIndex = index;
32873     },
32874
32875     /**
32876      * Returns the element for this dialog
32877      * @return {Roo.Element} The underlying dialog Element
32878      */
32879     getEl : function(){
32880         return this.el;
32881     }
32882 });
32883
32884 /**
32885  * @class Roo.DialogManager
32886  * Provides global access to BasicDialogs that have been created and
32887  * support for z-indexing (layering) multiple open dialogs.
32888  */
32889 Roo.DialogManager = function(){
32890     var list = {};
32891     var accessList = [];
32892     var front = null;
32893
32894     // private
32895     var sortDialogs = function(d1, d2){
32896         return (!d1._lastAccess || d1._lastAccess < d2._lastAccess) ? -1 : 1;
32897     };
32898
32899     // private
32900     var orderDialogs = function(){
32901         accessList.sort(sortDialogs);
32902         var seed = Roo.DialogManager.zseed;
32903         for(var i = 0, len = accessList.length; i < len; i++){
32904             var dlg = accessList[i];
32905             if(dlg){
32906                 dlg.setZIndex(seed + (i*10));
32907             }
32908         }
32909     };
32910
32911     return {
32912         /**
32913          * The starting z-index for BasicDialogs (defaults to 9000)
32914          * @type Number The z-index value
32915          */
32916         zseed : 9000,
32917
32918         // private
32919         register : function(dlg){
32920             list[dlg.id] = dlg;
32921             accessList.push(dlg);
32922         },
32923
32924         // private
32925         unregister : function(dlg){
32926             delete list[dlg.id];
32927             var i=0;
32928             var len=0;
32929             if(!accessList.indexOf){
32930                 for(  i = 0, len = accessList.length; i < len; i++){
32931                     if(accessList[i] == dlg){
32932                         accessList.splice(i, 1);
32933                         return;
32934                     }
32935                 }
32936             }else{
32937                  i = accessList.indexOf(dlg);
32938                 if(i != -1){
32939                     accessList.splice(i, 1);
32940                 }
32941             }
32942         },
32943
32944         /**
32945          * Gets a registered dialog by id
32946          * @param {String/Object} id The id of the dialog or a dialog
32947          * @return {Roo.BasicDialog} this
32948          */
32949         get : function(id){
32950             return typeof id == "object" ? id : list[id];
32951         },
32952
32953         /**
32954          * Brings the specified dialog to the front
32955          * @param {String/Object} dlg The id of the dialog or a dialog
32956          * @return {Roo.BasicDialog} this
32957          */
32958         bringToFront : function(dlg){
32959             dlg = this.get(dlg);
32960             if(dlg != front){
32961                 front = dlg;
32962                 dlg._lastAccess = new Date().getTime();
32963                 orderDialogs();
32964             }
32965             return dlg;
32966         },
32967
32968         /**
32969          * Sends the specified dialog to the back
32970          * @param {String/Object} dlg The id of the dialog or a dialog
32971          * @return {Roo.BasicDialog} this
32972          */
32973         sendToBack : function(dlg){
32974             dlg = this.get(dlg);
32975             dlg._lastAccess = -(new Date().getTime());
32976             orderDialogs();
32977             return dlg;
32978         },
32979
32980         /**
32981          * Hides all dialogs
32982          */
32983         hideAll : function(){
32984             for(var id in list){
32985                 if(list[id] && typeof list[id] != "function" && list[id].isVisible()){
32986                     list[id].hide();
32987                 }
32988             }
32989         }
32990     };
32991 }();
32992
32993 /**
32994  * @class Roo.LayoutDialog
32995  * @extends Roo.BasicDialog
32996  * Dialog which provides adjustments for working with a layout in a Dialog.
32997  * Add your necessary layout config options to the dialog's config.<br>
32998  * Example usage (including a nested layout):
32999  * <pre><code>
33000 if(!dialog){
33001     dialog = new Roo.LayoutDialog("download-dlg", {
33002         modal: true,
33003         width:600,
33004         height:450,
33005         shadow:true,
33006         minWidth:500,
33007         minHeight:350,
33008         autoTabs:true,
33009         proxyDrag:true,
33010         // layout config merges with the dialog config
33011         center:{
33012             tabPosition: "top",
33013             alwaysShowTabs: true
33014         }
33015     });
33016     dialog.addKeyListener(27, dialog.hide, dialog);
33017     dialog.setDefaultButton(dialog.addButton("Close", dialog.hide, dialog));
33018     dialog.addButton("Build It!", this.getDownload, this);
33019
33020     // we can even add nested layouts
33021     var innerLayout = new Roo.BorderLayout("dl-inner", {
33022         east: {
33023             initialSize: 200,
33024             autoScroll:true,
33025             split:true
33026         },
33027         center: {
33028             autoScroll:true
33029         }
33030     });
33031     innerLayout.beginUpdate();
33032     innerLayout.add("east", new Roo.ContentPanel("dl-details"));
33033     innerLayout.add("center", new Roo.ContentPanel("selection-panel"));
33034     innerLayout.endUpdate(true);
33035
33036     var layout = dialog.getLayout();
33037     layout.beginUpdate();
33038     layout.add("center", new Roo.ContentPanel("standard-panel",
33039                         {title: "Download the Source", fitToFrame:true}));
33040     layout.add("center", new Roo.NestedLayoutPanel(innerLayout,
33041                {title: "Build your own roo.js"}));
33042     layout.getRegion("center").showPanel(sp);
33043     layout.endUpdate();
33044 }
33045 </code></pre>
33046     * @constructor
33047     * @param {String/HTMLElement/Roo.Element} el The id of or container element, or config
33048     * @param {Object} config configuration options
33049   */
33050 Roo.LayoutDialog = function(el, cfg){
33051     
33052     var config=  cfg;
33053     if (typeof(cfg) == 'undefined') {
33054         config = Roo.apply({}, el);
33055         // not sure why we use documentElement here.. - it should always be body.
33056         // IE7 borks horribly if we use documentElement.
33057         // webkit also does not like documentElement - it creates a body element...
33058         el = Roo.get( document.body || document.documentElement ).createChild();
33059         //config.autoCreate = true;
33060     }
33061     
33062     
33063     config.autoTabs = false;
33064     Roo.LayoutDialog.superclass.constructor.call(this, el, config);
33065     this.body.setStyle({overflow:"hidden", position:"relative"});
33066     this.layout = new Roo.BorderLayout(this.body.dom, config);
33067     this.layout.monitorWindowResize = false;
33068     this.el.addClass("x-dlg-auto-layout");
33069     // fix case when center region overwrites center function
33070     this.center = Roo.BasicDialog.prototype.center;
33071     this.on("show", this.layout.layout, this.layout, true);
33072     if (config.items) {
33073         var xitems = config.items;
33074         delete config.items;
33075         Roo.each(xitems, this.addxtype, this);
33076     }
33077     
33078     
33079 };
33080 Roo.extend(Roo.LayoutDialog, Roo.BasicDialog, {
33081     /**
33082      * Ends update of the layout <strike>and resets display to none</strike>. Use standard beginUpdate/endUpdate on the layout.
33083      * @deprecated
33084      */
33085     endUpdate : function(){
33086         this.layout.endUpdate();
33087     },
33088
33089     /**
33090      * Begins an update of the layout <strike>and sets display to block and visibility to hidden</strike>. Use standard beginUpdate/endUpdate on the layout.
33091      *  @deprecated
33092      */
33093     beginUpdate : function(){
33094         this.layout.beginUpdate();
33095     },
33096
33097     /**
33098      * Get the BorderLayout for this dialog
33099      * @return {Roo.BorderLayout}
33100      */
33101     getLayout : function(){
33102         return this.layout;
33103     },
33104
33105     showEl : function(){
33106         Roo.LayoutDialog.superclass.showEl.apply(this, arguments);
33107         if(Roo.isIE7){
33108             this.layout.layout();
33109         }
33110     },
33111
33112     // private
33113     // Use the syncHeightBeforeShow config option to control this automatically
33114     syncBodyHeight : function(){
33115         Roo.LayoutDialog.superclass.syncBodyHeight.call(this);
33116         if(this.layout){this.layout.layout();}
33117     },
33118     
33119       /**
33120      * Add an xtype element (actually adds to the layout.)
33121      * @return {Object} xdata xtype object data.
33122      */
33123     
33124     addxtype : function(c) {
33125         return this.layout.addxtype(c);
33126     }
33127 });/*
33128  * Based on:
33129  * Ext JS Library 1.1.1
33130  * Copyright(c) 2006-2007, Ext JS, LLC.
33131  *
33132  * Originally Released Under LGPL - original licence link has changed is not relivant.
33133  *
33134  * Fork - LGPL
33135  * <script type="text/javascript">
33136  */
33137  
33138 /**
33139  * @class Roo.MessageBox
33140  * Utility class for generating different styles of message boxes.  The alias Roo.Msg can also be used.
33141  * Example usage:
33142  *<pre><code>
33143 // Basic alert:
33144 Roo.Msg.alert('Status', 'Changes saved successfully.');
33145
33146 // Prompt for user data:
33147 Roo.Msg.prompt('Name', 'Please enter your name:', function(btn, text){
33148     if (btn == 'ok'){
33149         // process text value...
33150     }
33151 });
33152
33153 // Show a dialog using config options:
33154 Roo.Msg.show({
33155    title:'Save Changes?',
33156    msg: 'Your are closing a tab that has unsaved changes. Would you like to save your changes?',
33157    buttons: Roo.Msg.YESNOCANCEL,
33158    fn: processResult,
33159    animEl: 'elId'
33160 });
33161 </code></pre>
33162  * @singleton
33163  */
33164 Roo.MessageBox = function(){
33165     var dlg, opt, mask, waitTimer;
33166     var bodyEl, msgEl, textboxEl, textareaEl, progressEl, pp;
33167     var buttons, activeTextEl, bwidth;
33168
33169     // private
33170     var handleButton = function(button){
33171         dlg.hide();
33172         Roo.callback(opt.fn, opt.scope||window, [button, activeTextEl.dom.value], 1);
33173     };
33174
33175     // private
33176     var handleHide = function(){
33177         if(opt && opt.cls){
33178             dlg.el.removeClass(opt.cls);
33179         }
33180         if(waitTimer){
33181             Roo.TaskMgr.stop(waitTimer);
33182             waitTimer = null;
33183         }
33184     };
33185
33186     // private
33187     var updateButtons = function(b){
33188         var width = 0;
33189         if(!b){
33190             buttons["ok"].hide();
33191             buttons["cancel"].hide();
33192             buttons["yes"].hide();
33193             buttons["no"].hide();
33194             dlg.footer.dom.style.display = 'none';
33195             return width;
33196         }
33197         dlg.footer.dom.style.display = '';
33198         for(var k in buttons){
33199             if(typeof buttons[k] != "function"){
33200                 if(b[k]){
33201                     buttons[k].show();
33202                     buttons[k].setText(typeof b[k] == "string" ? b[k] : Roo.MessageBox.buttonText[k]);
33203                     width += buttons[k].el.getWidth()+15;
33204                 }else{
33205                     buttons[k].hide();
33206                 }
33207             }
33208         }
33209         return width;
33210     };
33211
33212     // private
33213     var handleEsc = function(d, k, e){
33214         if(opt && opt.closable !== false){
33215             dlg.hide();
33216         }
33217         if(e){
33218             e.stopEvent();
33219         }
33220     };
33221
33222     return {
33223         /**
33224          * Returns a reference to the underlying {@link Roo.BasicDialog} element
33225          * @return {Roo.BasicDialog} The BasicDialog element
33226          */
33227         getDialog : function(){
33228            if(!dlg){
33229                 dlg = new Roo.BasicDialog("x-msg-box", {
33230                     autoCreate : true,
33231                     shadow: true,
33232                     draggable: true,
33233                     resizable:false,
33234                     constraintoviewport:false,
33235                     fixedcenter:true,
33236                     collapsible : false,
33237                     shim:true,
33238                     modal: true,
33239                     width:400, height:100,
33240                     buttonAlign:"center",
33241                     closeClick : function(){
33242                         if(opt && opt.buttons && opt.buttons.no && !opt.buttons.cancel){
33243                             handleButton("no");
33244                         }else{
33245                             handleButton("cancel");
33246                         }
33247                     }
33248                 });
33249                 dlg.on("hide", handleHide);
33250                 mask = dlg.mask;
33251                 dlg.addKeyListener(27, handleEsc);
33252                 buttons = {};
33253                 var bt = this.buttonText;
33254                 buttons["ok"] = dlg.addButton(bt["ok"], handleButton.createCallback("ok"));
33255                 buttons["yes"] = dlg.addButton(bt["yes"], handleButton.createCallback("yes"));
33256                 buttons["no"] = dlg.addButton(bt["no"], handleButton.createCallback("no"));
33257                 buttons["cancel"] = dlg.addButton(bt["cancel"], handleButton.createCallback("cancel"));
33258                 bodyEl = dlg.body.createChild({
33259
33260                     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>'
33261                 });
33262                 msgEl = bodyEl.dom.firstChild;
33263                 textboxEl = Roo.get(bodyEl.dom.childNodes[2]);
33264                 textboxEl.enableDisplayMode();
33265                 textboxEl.addKeyListener([10,13], function(){
33266                     if(dlg.isVisible() && opt && opt.buttons){
33267                         if(opt.buttons.ok){
33268                             handleButton("ok");
33269                         }else if(opt.buttons.yes){
33270                             handleButton("yes");
33271                         }
33272                     }
33273                 });
33274                 textareaEl = Roo.get(bodyEl.dom.childNodes[3]);
33275                 textareaEl.enableDisplayMode();
33276                 progressEl = Roo.get(bodyEl.dom.childNodes[4]);
33277                 progressEl.enableDisplayMode();
33278                 var pf = progressEl.dom.firstChild;
33279                 if (pf) {
33280                     pp = Roo.get(pf.firstChild);
33281                     pp.setHeight(pf.offsetHeight);
33282                 }
33283                 
33284             }
33285             return dlg;
33286         },
33287
33288         /**
33289          * Updates the message box body text
33290          * @param {String} text (optional) Replaces the message box element's innerHTML with the specified string (defaults to
33291          * the XHTML-compliant non-breaking space character '&amp;#160;')
33292          * @return {Roo.MessageBox} This message box
33293          */
33294         updateText : function(text){
33295             if(!dlg.isVisible() && !opt.width){
33296                 dlg.resizeTo(this.maxWidth, 100); // resize first so content is never clipped from previous shows
33297             }
33298             msgEl.innerHTML = text || '&#160;';
33299       
33300             var cw =  Math.max(msgEl.offsetWidth, msgEl.parentNode.scrollWidth);
33301             //Roo.log("guesed size: " + JSON.stringify([cw,msgEl.offsetWidth, msgEl.parentNode.scrollWidth]));
33302             var w = Math.max(
33303                     Math.min(opt.width || cw , this.maxWidth), 
33304                     Math.max(opt.minWidth || this.minWidth, bwidth)
33305             );
33306             if(opt.prompt){
33307                 activeTextEl.setWidth(w);
33308             }
33309             if(dlg.isVisible()){
33310                 dlg.fixedcenter = false;
33311             }
33312             // to big, make it scroll. = But as usual stupid IE does not support
33313             // !important..
33314             
33315             if ( bodyEl.getHeight() > (Roo.lib.Dom.getViewHeight() - 100)) {
33316                 bodyEl.setHeight ( Roo.lib.Dom.getViewHeight() - 100 );
33317                 bodyEl.dom.style.overflowY = 'auto' + ( Roo.isIE ? '' : ' !important');
33318             } else {
33319                 bodyEl.dom.style.height = '';
33320                 bodyEl.dom.style.overflowY = '';
33321             }
33322             if (cw > w) {
33323                 bodyEl.dom.style.get = 'auto' + ( Roo.isIE ? '' : ' !important');
33324             } else {
33325                 bodyEl.dom.style.overflowX = '';
33326             }
33327             
33328             dlg.setContentSize(w, bodyEl.getHeight());
33329             if(dlg.isVisible()){
33330                 dlg.fixedcenter = true;
33331             }
33332             return this;
33333         },
33334
33335         /**
33336          * Updates a progress-style message box's text and progress bar.  Only relevant on message boxes
33337          * initiated via {@link Roo.MessageBox#progress} or by calling {@link Roo.MessageBox#show} with progress: true.
33338          * @param {Number} value Any number between 0 and 1 (e.g., .5)
33339          * @param {String} text (optional) If defined, the message box's body text is replaced with the specified string (defaults to undefined)
33340          * @return {Roo.MessageBox} This message box
33341          */
33342         updateProgress : function(value, text){
33343             if(text){
33344                 this.updateText(text);
33345             }
33346             if (pp) { // weird bug on my firefox - for some reason this is not defined
33347                 pp.setWidth(Math.floor(value*progressEl.dom.firstChild.offsetWidth));
33348             }
33349             return this;
33350         },        
33351
33352         /**
33353          * Returns true if the message box is currently displayed
33354          * @return {Boolean} True if the message box is visible, else false
33355          */
33356         isVisible : function(){
33357             return dlg && dlg.isVisible();  
33358         },
33359
33360         /**
33361          * Hides the message box if it is displayed
33362          */
33363         hide : function(){
33364             if(this.isVisible()){
33365                 dlg.hide();
33366             }  
33367         },
33368
33369         /**
33370          * Displays a new message box, or reinitializes an existing message box, based on the config options
33371          * passed in. All functions (e.g. prompt, alert, etc) on MessageBox call this function internally.
33372          * The following config object properties are supported:
33373          * <pre>
33374 Property    Type             Description
33375 ----------  ---------------  ------------------------------------------------------------------------------------
33376 animEl            String/Element   An id or Element from which the message box should animate as it opens and
33377                                    closes (defaults to undefined)
33378 buttons           Object/Boolean   A button config object (e.g., Roo.MessageBox.OKCANCEL or {ok:'Foo',
33379                                    cancel:'Bar'}), or false to not show any buttons (defaults to false)
33380 closable          Boolean          False to hide the top-right close button (defaults to true).  Note that
33381                                    progress and wait dialogs will ignore this property and always hide the
33382                                    close button as they can only be closed programmatically.
33383 cls               String           A custom CSS class to apply to the message box element
33384 defaultTextHeight Number           The default height in pixels of the message box's multiline textarea if
33385                                    displayed (defaults to 75)
33386 fn                Function         A callback function to execute after closing the dialog.  The arguments to the
33387                                    function will be btn (the name of the button that was clicked, if applicable,
33388                                    e.g. "ok"), and text (the value of the active text field, if applicable).
33389                                    Progress and wait dialogs will ignore this option since they do not respond to
33390                                    user actions and can only be closed programmatically, so any required function
33391                                    should be called by the same code after it closes the dialog.
33392 icon              String           A CSS class that provides a background image to be used as an icon for
33393                                    the dialog (e.g., Roo.MessageBox.WARNING or 'custom-class', defaults to '')
33394 maxWidth          Number           The maximum width in pixels of the message box (defaults to 600)
33395 minWidth          Number           The minimum width in pixels of the message box (defaults to 100)
33396 modal             Boolean          False to allow user interaction with the page while the message box is
33397                                    displayed (defaults to true)
33398 msg               String           A string that will replace the existing message box body text (defaults
33399                                    to the XHTML-compliant non-breaking space character '&#160;')
33400 multiline         Boolean          True to prompt the user to enter multi-line text (defaults to false)
33401 progress          Boolean          True to display a progress bar (defaults to false)
33402 progressText      String           The text to display inside the progress bar if progress = true (defaults to '')
33403 prompt            Boolean          True to prompt the user to enter single-line text (defaults to false)
33404 proxyDrag         Boolean          True to display a lightweight proxy while dragging (defaults to false)
33405 title             String           The title text
33406 value             String           The string value to set into the active textbox element if displayed
33407 wait              Boolean          True to display a progress bar (defaults to false)
33408 width             Number           The width of the dialog in pixels
33409 </pre>
33410          *
33411          * Example usage:
33412          * <pre><code>
33413 Roo.Msg.show({
33414    title: 'Address',
33415    msg: 'Please enter your address:',
33416    width: 300,
33417    buttons: Roo.MessageBox.OKCANCEL,
33418    multiline: true,
33419    fn: saveAddress,
33420    animEl: 'addAddressBtn'
33421 });
33422 </code></pre>
33423          * @param {Object} config Configuration options
33424          * @return {Roo.MessageBox} This message box
33425          */
33426         show : function(options)
33427         {
33428             
33429             // this causes nightmares if you show one dialog after another
33430             // especially on callbacks..
33431              
33432             if(this.isVisible()){
33433                 
33434                 this.hide();
33435                 Roo.log("[Roo.Messagebox] Show called while message displayed:" );
33436                 Roo.log("Old Dialog Message:" +  msgEl.innerHTML );
33437                 Roo.log("New Dialog Message:" +  options.msg )
33438                 //this.alert("ERROR", "Multiple dialogs where displayed at the same time");
33439                 //throw "Roo.MessageBox ERROR : Multiple dialogs where displayed at the same time";
33440                 
33441             }
33442             var d = this.getDialog();
33443             opt = options;
33444             d.setTitle(opt.title || "&#160;");
33445             d.close.setDisplayed(opt.closable !== false);
33446             activeTextEl = textboxEl;
33447             opt.prompt = opt.prompt || (opt.multiline ? true : false);
33448             if(opt.prompt){
33449                 if(opt.multiline){
33450                     textboxEl.hide();
33451                     textareaEl.show();
33452                     textareaEl.setHeight(typeof opt.multiline == "number" ?
33453                         opt.multiline : this.defaultTextHeight);
33454                     activeTextEl = textareaEl;
33455                 }else{
33456                     textboxEl.show();
33457                     textareaEl.hide();
33458                 }
33459             }else{
33460                 textboxEl.hide();
33461                 textareaEl.hide();
33462             }
33463             progressEl.setDisplayed(opt.progress === true);
33464             this.updateProgress(0);
33465             activeTextEl.dom.value = opt.value || "";
33466             if(opt.prompt){
33467                 dlg.setDefaultButton(activeTextEl);
33468             }else{
33469                 var bs = opt.buttons;
33470                 var db = null;
33471                 if(bs && bs.ok){
33472                     db = buttons["ok"];
33473                 }else if(bs && bs.yes){
33474                     db = buttons["yes"];
33475                 }
33476                 dlg.setDefaultButton(db);
33477             }
33478             bwidth = updateButtons(opt.buttons);
33479             this.updateText(opt.msg);
33480             if(opt.cls){
33481                 d.el.addClass(opt.cls);
33482             }
33483             d.proxyDrag = opt.proxyDrag === true;
33484             d.modal = opt.modal !== false;
33485             d.mask = opt.modal !== false ? mask : false;
33486             if(!d.isVisible()){
33487                 // force it to the end of the z-index stack so it gets a cursor in FF
33488                 document.body.appendChild(dlg.el.dom);
33489                 d.animateTarget = null;
33490                 d.show(options.animEl);
33491             }
33492             return this;
33493         },
33494
33495         /**
33496          * Displays a message box with a progress bar.  This message box has no buttons and is not closeable by
33497          * the user.  You are responsible for updating the progress bar as needed via {@link Roo.MessageBox#updateProgress}
33498          * and closing the message box when the process is complete.
33499          * @param {String} title The title bar text
33500          * @param {String} msg The message box body text
33501          * @return {Roo.MessageBox} This message box
33502          */
33503         progress : function(title, msg){
33504             this.show({
33505                 title : title,
33506                 msg : msg,
33507                 buttons: false,
33508                 progress:true,
33509                 closable:false,
33510                 minWidth: this.minProgressWidth,
33511                 modal : true
33512             });
33513             return this;
33514         },
33515
33516         /**
33517          * Displays a standard read-only message box with an OK button (comparable to the basic JavaScript Window.alert).
33518          * If a callback function is passed it will be called after the user clicks the button, and the
33519          * id of the button that was clicked will be passed as the only parameter to the callback
33520          * (could also be the top-right close button).
33521          * @param {String} title The title bar text
33522          * @param {String} msg The message box body text
33523          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33524          * @param {Object} scope (optional) The scope of the callback function
33525          * @return {Roo.MessageBox} This message box
33526          */
33527         alert : function(title, msg, fn, scope){
33528             this.show({
33529                 title : title,
33530                 msg : msg,
33531                 buttons: this.OK,
33532                 fn: fn,
33533                 scope : scope,
33534                 modal : true
33535             });
33536             return this;
33537         },
33538
33539         /**
33540          * Displays a message box with an infinitely auto-updating progress bar.  This can be used to block user
33541          * interaction while waiting for a long-running process to complete that does not have defined intervals.
33542          * You are responsible for closing the message box when the process is complete.
33543          * @param {String} msg The message box body text
33544          * @param {String} title (optional) The title bar text
33545          * @return {Roo.MessageBox} This message box
33546          */
33547         wait : function(msg, title){
33548             this.show({
33549                 title : title,
33550                 msg : msg,
33551                 buttons: false,
33552                 closable:false,
33553                 progress:true,
33554                 modal:true,
33555                 width:300,
33556                 wait:true
33557             });
33558             waitTimer = Roo.TaskMgr.start({
33559                 run: function(i){
33560                     Roo.MessageBox.updateProgress(((((i+20)%20)+1)*5)*.01);
33561                 },
33562                 interval: 1000
33563             });
33564             return this;
33565         },
33566
33567         /**
33568          * Displays a confirmation message box with Yes and No buttons (comparable to JavaScript's Window.confirm).
33569          * If a callback function is passed it will be called after the user clicks either button, and the id of the
33570          * button that was clicked will be passed as the only parameter to the callback (could also be the top-right close button).
33571          * @param {String} title The title bar text
33572          * @param {String} msg The message box body text
33573          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33574          * @param {Object} scope (optional) The scope of the callback function
33575          * @return {Roo.MessageBox} This message box
33576          */
33577         confirm : function(title, msg, fn, scope){
33578             this.show({
33579                 title : title,
33580                 msg : msg,
33581                 buttons: this.YESNO,
33582                 fn: fn,
33583                 scope : scope,
33584                 modal : true
33585             });
33586             return this;
33587         },
33588
33589         /**
33590          * Displays a message box with OK and Cancel buttons prompting the user to enter some text (comparable to
33591          * JavaScript's Window.prompt).  The prompt can be a single-line or multi-line textbox.  If a callback function
33592          * is passed it will be called after the user clicks either button, and the id of the button that was clicked
33593          * (could also be the top-right close button) and the text that was entered will be passed as the two
33594          * parameters to the callback.
33595          * @param {String} title The title bar text
33596          * @param {String} msg The message box body text
33597          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33598          * @param {Object} scope (optional) The scope of the callback function
33599          * @param {Boolean/Number} multiline (optional) True to create a multiline textbox using the defaultTextHeight
33600          * property, or the height in pixels to create the textbox (defaults to false / single-line)
33601          * @return {Roo.MessageBox} This message box
33602          */
33603         prompt : function(title, msg, fn, scope, multiline){
33604             this.show({
33605                 title : title,
33606                 msg : msg,
33607                 buttons: this.OKCANCEL,
33608                 fn: fn,
33609                 minWidth:250,
33610                 scope : scope,
33611                 prompt:true,
33612                 multiline: multiline,
33613                 modal : true
33614             });
33615             return this;
33616         },
33617
33618         /**
33619          * Button config that displays a single OK button
33620          * @type Object
33621          */
33622         OK : {ok:true},
33623         /**
33624          * Button config that displays Yes and No buttons
33625          * @type Object
33626          */
33627         YESNO : {yes:true, no:true},
33628         /**
33629          * Button config that displays OK and Cancel buttons
33630          * @type Object
33631          */
33632         OKCANCEL : {ok:true, cancel:true},
33633         /**
33634          * Button config that displays Yes, No and Cancel buttons
33635          * @type Object
33636          */
33637         YESNOCANCEL : {yes:true, no:true, cancel:true},
33638
33639         /**
33640          * The default height in pixels of the message box's multiline textarea if displayed (defaults to 75)
33641          * @type Number
33642          */
33643         defaultTextHeight : 75,
33644         /**
33645          * The maximum width in pixels of the message box (defaults to 600)
33646          * @type Number
33647          */
33648         maxWidth : 600,
33649         /**
33650          * The minimum width in pixels of the message box (defaults to 100)
33651          * @type Number
33652          */
33653         minWidth : 100,
33654         /**
33655          * The minimum width in pixels of the message box if it is a progress-style dialog.  This is useful
33656          * for setting a different minimum width than text-only dialogs may need (defaults to 250)
33657          * @type Number
33658          */
33659         minProgressWidth : 250,
33660         /**
33661          * An object containing the default button text strings that can be overriden for localized language support.
33662          * Supported properties are: ok, cancel, yes and no.
33663          * Customize the default text like so: Roo.MessageBox.buttonText.yes = "S?";
33664          * @type Object
33665          */
33666         buttonText : {
33667             ok : "OK",
33668             cancel : "Cancel",
33669             yes : "Yes",
33670             no : "No"
33671         }
33672     };
33673 }();
33674
33675 /**
33676  * Shorthand for {@link Roo.MessageBox}
33677  */
33678 Roo.Msg = Roo.MessageBox;/*
33679  * Based on:
33680  * Ext JS Library 1.1.1
33681  * Copyright(c) 2006-2007, Ext JS, LLC.
33682  *
33683  * Originally Released Under LGPL - original licence link has changed is not relivant.
33684  *
33685  * Fork - LGPL
33686  * <script type="text/javascript">
33687  */
33688 /**
33689  * @class Roo.QuickTips
33690  * Provides attractive and customizable tooltips for any element.
33691  * @singleton
33692  */
33693 Roo.QuickTips = function(){
33694     var el, tipBody, tipBodyText, tipTitle, tm, cfg, close, tagEls = {}, esc, removeCls = null, bdLeft, bdRight;
33695     var ce, bd, xy, dd;
33696     var visible = false, disabled = true, inited = false;
33697     var showProc = 1, hideProc = 1, dismissProc = 1, locks = [];
33698     
33699     var onOver = function(e){
33700         if(disabled){
33701             return;
33702         }
33703         var t = e.getTarget();
33704         if(!t || t.nodeType !== 1 || t == document || t == document.body){
33705             return;
33706         }
33707         if(ce && t == ce.el){
33708             clearTimeout(hideProc);
33709             return;
33710         }
33711         if(t && tagEls[t.id]){
33712             tagEls[t.id].el = t;
33713             showProc = show.defer(tm.showDelay, tm, [tagEls[t.id]]);
33714             return;
33715         }
33716         var ttp, et = Roo.fly(t);
33717         var ns = cfg.namespace;
33718         if(tm.interceptTitles && t.title){
33719             ttp = t.title;
33720             t.qtip = ttp;
33721             t.removeAttribute("title");
33722             e.preventDefault();
33723         }else{
33724             ttp = t.qtip || et.getAttributeNS(ns, cfg.attribute) || et.getAttributeNS(cfg.alt_namespace, cfg.attribute) ;
33725         }
33726         if(ttp){
33727             showProc = show.defer(tm.showDelay, tm, [{
33728                 el: t, 
33729                 text: ttp.replace(/\\n/g,'<br/>'),
33730                 width: et.getAttributeNS(ns, cfg.width),
33731                 autoHide: et.getAttributeNS(ns, cfg.hide) != "user",
33732                 title: et.getAttributeNS(ns, cfg.title),
33733                     cls: et.getAttributeNS(ns, cfg.cls)
33734             }]);
33735         }
33736     };
33737     
33738     var onOut = function(e){
33739         clearTimeout(showProc);
33740         var t = e.getTarget();
33741         if(t && ce && ce.el == t && (tm.autoHide && ce.autoHide !== false)){
33742             hideProc = setTimeout(hide, tm.hideDelay);
33743         }
33744     };
33745     
33746     var onMove = function(e){
33747         if(disabled){
33748             return;
33749         }
33750         xy = e.getXY();
33751         xy[1] += 18;
33752         if(tm.trackMouse && ce){
33753             el.setXY(xy);
33754         }
33755     };
33756     
33757     var onDown = function(e){
33758         clearTimeout(showProc);
33759         clearTimeout(hideProc);
33760         if(!e.within(el)){
33761             if(tm.hideOnClick){
33762                 hide();
33763                 tm.disable();
33764                 tm.enable.defer(100, tm);
33765             }
33766         }
33767     };
33768     
33769     var getPad = function(){
33770         return 2;//bdLeft.getPadding('l')+bdRight.getPadding('r');
33771     };
33772
33773     var show = function(o){
33774         if(disabled){
33775             return;
33776         }
33777         clearTimeout(dismissProc);
33778         ce = o;
33779         if(removeCls){ // in case manually hidden
33780             el.removeClass(removeCls);
33781             removeCls = null;
33782         }
33783         if(ce.cls){
33784             el.addClass(ce.cls);
33785             removeCls = ce.cls;
33786         }
33787         if(ce.title){
33788             tipTitle.update(ce.title);
33789             tipTitle.show();
33790         }else{
33791             tipTitle.update('');
33792             tipTitle.hide();
33793         }
33794         el.dom.style.width  = tm.maxWidth+'px';
33795         //tipBody.dom.style.width = '';
33796         tipBodyText.update(o.text);
33797         var p = getPad(), w = ce.width;
33798         if(!w){
33799             var td = tipBodyText.dom;
33800             var aw = Math.max(td.offsetWidth, td.clientWidth, td.scrollWidth);
33801             if(aw > tm.maxWidth){
33802                 w = tm.maxWidth;
33803             }else if(aw < tm.minWidth){
33804                 w = tm.minWidth;
33805             }else{
33806                 w = aw;
33807             }
33808         }
33809         //tipBody.setWidth(w);
33810         el.setWidth(parseInt(w, 10) + p);
33811         if(ce.autoHide === false){
33812             close.setDisplayed(true);
33813             if(dd){
33814                 dd.unlock();
33815             }
33816         }else{
33817             close.setDisplayed(false);
33818             if(dd){
33819                 dd.lock();
33820             }
33821         }
33822         if(xy){
33823             el.avoidY = xy[1]-18;
33824             el.setXY(xy);
33825         }
33826         if(tm.animate){
33827             el.setOpacity(.1);
33828             el.setStyle("visibility", "visible");
33829             el.fadeIn({callback: afterShow});
33830         }else{
33831             afterShow();
33832         }
33833     };
33834     
33835     var afterShow = function(){
33836         if(ce){
33837             el.show();
33838             esc.enable();
33839             if(tm.autoDismiss && ce.autoHide !== false){
33840                 dismissProc = setTimeout(hide, tm.autoDismissDelay);
33841             }
33842         }
33843     };
33844     
33845     var hide = function(noanim){
33846         clearTimeout(dismissProc);
33847         clearTimeout(hideProc);
33848         ce = null;
33849         if(el.isVisible()){
33850             esc.disable();
33851             if(noanim !== true && tm.animate){
33852                 el.fadeOut({callback: afterHide});
33853             }else{
33854                 afterHide();
33855             } 
33856         }
33857     };
33858     
33859     var afterHide = function(){
33860         el.hide();
33861         if(removeCls){
33862             el.removeClass(removeCls);
33863             removeCls = null;
33864         }
33865     };
33866     
33867     return {
33868         /**
33869         * @cfg {Number} minWidth
33870         * The minimum width of the quick tip (defaults to 40)
33871         */
33872        minWidth : 40,
33873         /**
33874         * @cfg {Number} maxWidth
33875         * The maximum width of the quick tip (defaults to 300)
33876         */
33877        maxWidth : 300,
33878         /**
33879         * @cfg {Boolean} interceptTitles
33880         * True to automatically use the element's DOM title value if available (defaults to false)
33881         */
33882        interceptTitles : false,
33883         /**
33884         * @cfg {Boolean} trackMouse
33885         * True to have the quick tip follow the mouse as it moves over the target element (defaults to false)
33886         */
33887        trackMouse : false,
33888         /**
33889         * @cfg {Boolean} hideOnClick
33890         * True to hide the quick tip if the user clicks anywhere in the document (defaults to true)
33891         */
33892        hideOnClick : true,
33893         /**
33894         * @cfg {Number} showDelay
33895         * Delay in milliseconds before the quick tip displays after the mouse enters the target element (defaults to 500)
33896         */
33897        showDelay : 500,
33898         /**
33899         * @cfg {Number} hideDelay
33900         * Delay in milliseconds before the quick tip hides when autoHide = true (defaults to 200)
33901         */
33902        hideDelay : 200,
33903         /**
33904         * @cfg {Boolean} autoHide
33905         * True to automatically hide the quick tip after the mouse exits the target element (defaults to true).
33906         * Used in conjunction with hideDelay.
33907         */
33908        autoHide : true,
33909         /**
33910         * @cfg {Boolean}
33911         * True to automatically hide the quick tip after a set period of time, regardless of the user's actions
33912         * (defaults to true).  Used in conjunction with autoDismissDelay.
33913         */
33914        autoDismiss : true,
33915         /**
33916         * @cfg {Number}
33917         * Delay in milliseconds before the quick tip hides when autoDismiss = true (defaults to 5000)
33918         */
33919        autoDismissDelay : 5000,
33920        /**
33921         * @cfg {Boolean} animate
33922         * True to turn on fade animation. Defaults to false (ClearType/scrollbar flicker issues in IE7).
33923         */
33924        animate : false,
33925
33926        /**
33927         * @cfg {String} title
33928         * Title text to display (defaults to '').  This can be any valid HTML markup.
33929         */
33930         title: '',
33931        /**
33932         * @cfg {String} text
33933         * Body text to display (defaults to '').  This can be any valid HTML markup.
33934         */
33935         text : '',
33936        /**
33937         * @cfg {String} cls
33938         * A CSS class to apply to the base quick tip element (defaults to '').
33939         */
33940         cls : '',
33941        /**
33942         * @cfg {Number} width
33943         * Width in pixels of the quick tip (defaults to auto).  Width will be ignored if it exceeds the bounds of
33944         * minWidth or maxWidth.
33945         */
33946         width : null,
33947
33948     /**
33949      * Initialize and enable QuickTips for first use.  This should be called once before the first attempt to access
33950      * or display QuickTips in a page.
33951      */
33952        init : function(){
33953           tm = Roo.QuickTips;
33954           cfg = tm.tagConfig;
33955           if(!inited){
33956               if(!Roo.isReady){ // allow calling of init() before onReady
33957                   Roo.onReady(Roo.QuickTips.init, Roo.QuickTips);
33958                   return;
33959               }
33960               el = new Roo.Layer({cls:"x-tip", shadow:"drop", shim: true, constrain:true, shadowOffset:4});
33961               el.fxDefaults = {stopFx: true};
33962               // maximum custom styling
33963               //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>');
33964               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>');              
33965               tipTitle = el.child('h3');
33966               tipTitle.enableDisplayMode("block");
33967               tipBody = el.child('div.x-tip-bd');
33968               tipBodyText = el.child('div.x-tip-bd-inner');
33969               //bdLeft = el.child('div.x-tip-bd-left');
33970               //bdRight = el.child('div.x-tip-bd-right');
33971               close = el.child('div.x-tip-close');
33972               close.enableDisplayMode("block");
33973               close.on("click", hide);
33974               var d = Roo.get(document);
33975               d.on("mousedown", onDown);
33976               d.on("mouseover", onOver);
33977               d.on("mouseout", onOut);
33978               d.on("mousemove", onMove);
33979               esc = d.addKeyListener(27, hide);
33980               esc.disable();
33981               if(Roo.dd.DD){
33982                   dd = el.initDD("default", null, {
33983                       onDrag : function(){
33984                           el.sync();  
33985                       }
33986                   });
33987                   dd.setHandleElId(tipTitle.id);
33988                   dd.lock();
33989               }
33990               inited = true;
33991           }
33992           this.enable(); 
33993        },
33994
33995     /**
33996      * Configures a new quick tip instance and assigns it to a target element.  The following config options
33997      * are supported:
33998      * <pre>
33999 Property    Type                   Description
34000 ----------  ---------------------  ------------------------------------------------------------------------
34001 target      Element/String/Array   An Element, id or array of ids that this quick tip should be tied to
34002      * </ul>
34003      * @param {Object} config The config object
34004      */
34005        register : function(config){
34006            var cs = config instanceof Array ? config : arguments;
34007            for(var i = 0, len = cs.length; i < len; i++) {
34008                var c = cs[i];
34009                var target = c.target;
34010                if(target){
34011                    if(target instanceof Array){
34012                        for(var j = 0, jlen = target.length; j < jlen; j++){
34013                            tagEls[target[j]] = c;
34014                        }
34015                    }else{
34016                        tagEls[typeof target == 'string' ? target : Roo.id(target)] = c;
34017                    }
34018                }
34019            }
34020        },
34021
34022     /**
34023      * Removes this quick tip from its element and destroys it.
34024      * @param {String/HTMLElement/Element} el The element from which the quick tip is to be removed.
34025      */
34026        unregister : function(el){
34027            delete tagEls[Roo.id(el)];
34028        },
34029
34030     /**
34031      * Enable this quick tip.
34032      */
34033        enable : function(){
34034            if(inited && disabled){
34035                locks.pop();
34036                if(locks.length < 1){
34037                    disabled = false;
34038                }
34039            }
34040        },
34041
34042     /**
34043      * Disable this quick tip.
34044      */
34045        disable : function(){
34046           disabled = true;
34047           clearTimeout(showProc);
34048           clearTimeout(hideProc);
34049           clearTimeout(dismissProc);
34050           if(ce){
34051               hide(true);
34052           }
34053           locks.push(1);
34054        },
34055
34056     /**
34057      * Returns true if the quick tip is enabled, else false.
34058      */
34059        isEnabled : function(){
34060             return !disabled;
34061        },
34062
34063         // private
34064        tagConfig : {
34065            namespace : "roo", // was ext?? this may break..
34066            alt_namespace : "ext",
34067            attribute : "qtip",
34068            width : "width",
34069            target : "target",
34070            title : "qtitle",
34071            hide : "hide",
34072            cls : "qclass"
34073        }
34074    };
34075 }();
34076
34077 // backwards compat
34078 Roo.QuickTips.tips = Roo.QuickTips.register;/*
34079  * Based on:
34080  * Ext JS Library 1.1.1
34081  * Copyright(c) 2006-2007, Ext JS, LLC.
34082  *
34083  * Originally Released Under LGPL - original licence link has changed is not relivant.
34084  *
34085  * Fork - LGPL
34086  * <script type="text/javascript">
34087  */
34088  
34089
34090 /**
34091  * @class Roo.tree.TreePanel
34092  * @extends Roo.data.Tree
34093
34094  * @cfg {Boolean} rootVisible false to hide the root node (defaults to true)
34095  * @cfg {Boolean} lines false to disable tree lines (defaults to true)
34096  * @cfg {Boolean} enableDD true to enable drag and drop
34097  * @cfg {Boolean} enableDrag true to enable just drag
34098  * @cfg {Boolean} enableDrop true to enable just drop
34099  * @cfg {Object} dragConfig Custom config to pass to the {@link Roo.tree.TreeDragZone} instance
34100  * @cfg {Object} dropConfig Custom config to pass to the {@link Roo.tree.TreeDropZone} instance
34101  * @cfg {String} ddGroup The DD group this TreePanel belongs to
34102  * @cfg {String} ddAppendOnly True if the tree should only allow append drops (use for trees which are sorted)
34103  * @cfg {Boolean} ddScroll true to enable YUI body scrolling
34104  * @cfg {Boolean} containerScroll true to register this container with ScrollManager
34105  * @cfg {Boolean} hlDrop false to disable node highlight on drop (defaults to the value of Roo.enableFx)
34106  * @cfg {String} hlColor The color of the node highlight (defaults to C3DAF9)
34107  * @cfg {Boolean} animate true to enable animated expand/collapse (defaults to the value of Roo.enableFx)
34108  * @cfg {Boolean} singleExpand true if only 1 node per branch may be expanded
34109  * @cfg {Boolean} selModel A tree selection model to use with this TreePanel (defaults to a {@link Roo.tree.DefaultSelectionModel})
34110  * @cfg {Boolean} loader A TreeLoader for use with this TreePanel
34111  * @cfg {Object|Roo.tree.TreeEditor} editor The TreeEditor or xtype data to display when clicked.
34112  * @cfg {String} pathSeparator The token used to separate sub-paths in path strings (defaults to '/')
34113  * @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>
34114  * @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>
34115  * 
34116  * @constructor
34117  * @param {String/HTMLElement/Element} el The container element
34118  * @param {Object} config
34119  */
34120 Roo.tree.TreePanel = function(el, config){
34121     var root = false;
34122     var loader = false;
34123     if (config.root) {
34124         root = config.root;
34125         delete config.root;
34126     }
34127     if (config.loader) {
34128         loader = config.loader;
34129         delete config.loader;
34130     }
34131     
34132     Roo.apply(this, config);
34133     Roo.tree.TreePanel.superclass.constructor.call(this);
34134     this.el = Roo.get(el);
34135     this.el.addClass('x-tree');
34136     //console.log(root);
34137     if (root) {
34138         this.setRootNode( Roo.factory(root, Roo.tree));
34139     }
34140     if (loader) {
34141         this.loader = Roo.factory(loader, Roo.tree);
34142     }
34143    /**
34144     * Read-only. The id of the container element becomes this TreePanel's id.
34145     */
34146     this.id = this.el.id;
34147     this.addEvents({
34148         /**
34149         * @event beforeload
34150         * Fires before a node is loaded, return false to cancel
34151         * @param {Node} node The node being loaded
34152         */
34153         "beforeload" : true,
34154         /**
34155         * @event load
34156         * Fires when a node is loaded
34157         * @param {Node} node The node that was loaded
34158         */
34159         "load" : true,
34160         /**
34161         * @event textchange
34162         * Fires when the text for a node is changed
34163         * @param {Node} node The node
34164         * @param {String} text The new text
34165         * @param {String} oldText The old text
34166         */
34167         "textchange" : true,
34168         /**
34169         * @event beforeexpand
34170         * Fires before a node is expanded, return false to cancel.
34171         * @param {Node} node The node
34172         * @param {Boolean} deep
34173         * @param {Boolean} anim
34174         */
34175         "beforeexpand" : true,
34176         /**
34177         * @event beforecollapse
34178         * Fires before a node is collapsed, return false to cancel.
34179         * @param {Node} node The node
34180         * @param {Boolean} deep
34181         * @param {Boolean} anim
34182         */
34183         "beforecollapse" : true,
34184         /**
34185         * @event expand
34186         * Fires when a node is expanded
34187         * @param {Node} node The node
34188         */
34189         "expand" : true,
34190         /**
34191         * @event disabledchange
34192         * Fires when the disabled status of a node changes
34193         * @param {Node} node The node
34194         * @param {Boolean} disabled
34195         */
34196         "disabledchange" : true,
34197         /**
34198         * @event collapse
34199         * Fires when a node is collapsed
34200         * @param {Node} node The node
34201         */
34202         "collapse" : true,
34203         /**
34204         * @event beforeclick
34205         * Fires before click processing on a node. Return false to cancel the default action.
34206         * @param {Node} node The node
34207         * @param {Roo.EventObject} e The event object
34208         */
34209         "beforeclick":true,
34210         /**
34211         * @event checkchange
34212         * Fires when a node with a checkbox's checked property changes
34213         * @param {Node} this This node
34214         * @param {Boolean} checked
34215         */
34216         "checkchange":true,
34217         /**
34218         * @event click
34219         * Fires when a node is clicked
34220         * @param {Node} node The node
34221         * @param {Roo.EventObject} e The event object
34222         */
34223         "click":true,
34224         /**
34225         * @event dblclick
34226         * Fires when a node is double clicked
34227         * @param {Node} node The node
34228         * @param {Roo.EventObject} e The event object
34229         */
34230         "dblclick":true,
34231         /**
34232         * @event contextmenu
34233         * Fires when a node is right clicked
34234         * @param {Node} node The node
34235         * @param {Roo.EventObject} e The event object
34236         */
34237         "contextmenu":true,
34238         /**
34239         * @event beforechildrenrendered
34240         * Fires right before the child nodes for a node are rendered
34241         * @param {Node} node The node
34242         */
34243         "beforechildrenrendered":true,
34244         /**
34245         * @event startdrag
34246         * Fires when a node starts being dragged
34247         * @param {Roo.tree.TreePanel} this
34248         * @param {Roo.tree.TreeNode} node
34249         * @param {event} e The raw browser event
34250         */ 
34251        "startdrag" : true,
34252        /**
34253         * @event enddrag
34254         * Fires when a drag operation is complete
34255         * @param {Roo.tree.TreePanel} this
34256         * @param {Roo.tree.TreeNode} node
34257         * @param {event} e The raw browser event
34258         */
34259        "enddrag" : true,
34260        /**
34261         * @event dragdrop
34262         * Fires when a dragged node is dropped on a valid DD target
34263         * @param {Roo.tree.TreePanel} this
34264         * @param {Roo.tree.TreeNode} node
34265         * @param {DD} dd The dd it was dropped on
34266         * @param {event} e The raw browser event
34267         */
34268        "dragdrop" : true,
34269        /**
34270         * @event beforenodedrop
34271         * Fires when a DD object is dropped on a node in this tree for preprocessing. Return false to cancel the drop. The dropEvent
34272         * passed to handlers has the following properties:<br />
34273         * <ul style="padding:5px;padding-left:16px;">
34274         * <li>tree - The TreePanel</li>
34275         * <li>target - The node being targeted for the drop</li>
34276         * <li>data - The drag data from the drag source</li>
34277         * <li>point - The point of the drop - append, above or below</li>
34278         * <li>source - The drag source</li>
34279         * <li>rawEvent - Raw mouse event</li>
34280         * <li>dropNode - Drop node(s) provided by the source <b>OR</b> you can supply node(s)
34281         * to be inserted by setting them on this object.</li>
34282         * <li>cancel - Set this to true to cancel the drop.</li>
34283         * </ul>
34284         * @param {Object} dropEvent
34285         */
34286        "beforenodedrop" : true,
34287        /**
34288         * @event nodedrop
34289         * Fires after a DD object is dropped on a node in this tree. The dropEvent
34290         * passed to handlers has the following properties:<br />
34291         * <ul style="padding:5px;padding-left:16px;">
34292         * <li>tree - The TreePanel</li>
34293         * <li>target - The node being targeted for the drop</li>
34294         * <li>data - The drag data from the drag source</li>
34295         * <li>point - The point of the drop - append, above or below</li>
34296         * <li>source - The drag source</li>
34297         * <li>rawEvent - Raw mouse event</li>
34298         * <li>dropNode - Dropped node(s).</li>
34299         * </ul>
34300         * @param {Object} dropEvent
34301         */
34302        "nodedrop" : true,
34303         /**
34304         * @event nodedragover
34305         * Fires when a tree node is being targeted for a drag drop, return false to signal drop not allowed. The dragOverEvent
34306         * passed to handlers has the following properties:<br />
34307         * <ul style="padding:5px;padding-left:16px;">
34308         * <li>tree - The TreePanel</li>
34309         * <li>target - The node being targeted for the drop</li>
34310         * <li>data - The drag data from the drag source</li>
34311         * <li>point - The point of the drop - append, above or below</li>
34312         * <li>source - The drag source</li>
34313         * <li>rawEvent - Raw mouse event</li>
34314         * <li>dropNode - Drop node(s) provided by the source.</li>
34315         * <li>cancel - Set this to true to signal drop not allowed.</li>
34316         * </ul>
34317         * @param {Object} dragOverEvent
34318         */
34319        "nodedragover" : true,
34320        /**
34321         * @event appendnode
34322         * Fires when append node to the tree
34323         * @param {Roo.tree.TreePanel} this
34324         * @param {Roo.tree.TreeNode} node
34325         * @param {Number} index The index of the newly appended node
34326         */
34327        "appendnode" : true
34328         
34329     });
34330     if(this.singleExpand){
34331        this.on("beforeexpand", this.restrictExpand, this);
34332     }
34333     if (this.editor) {
34334         this.editor.tree = this;
34335         this.editor = Roo.factory(this.editor, Roo.tree);
34336     }
34337     
34338     if (this.selModel) {
34339         this.selModel = Roo.factory(this.selModel, Roo.tree);
34340     }
34341    
34342 };
34343 Roo.extend(Roo.tree.TreePanel, Roo.data.Tree, {
34344     rootVisible : true,
34345     animate: Roo.enableFx,
34346     lines : true,
34347     enableDD : false,
34348     hlDrop : Roo.enableFx,
34349   
34350     renderer: false,
34351     
34352     rendererTip: false,
34353     // private
34354     restrictExpand : function(node){
34355         var p = node.parentNode;
34356         if(p){
34357             if(p.expandedChild && p.expandedChild.parentNode == p){
34358                 p.expandedChild.collapse();
34359             }
34360             p.expandedChild = node;
34361         }
34362     },
34363
34364     // private override
34365     setRootNode : function(node){
34366         Roo.tree.TreePanel.superclass.setRootNode.call(this, node);
34367         if(!this.rootVisible){
34368             node.ui = new Roo.tree.RootTreeNodeUI(node);
34369         }
34370         return node;
34371     },
34372
34373     /**
34374      * Returns the container element for this TreePanel
34375      */
34376     getEl : function(){
34377         return this.el;
34378     },
34379
34380     /**
34381      * Returns the default TreeLoader for this TreePanel
34382      */
34383     getLoader : function(){
34384         return this.loader;
34385     },
34386
34387     /**
34388      * Expand all nodes
34389      */
34390     expandAll : function(){
34391         this.root.expand(true);
34392     },
34393
34394     /**
34395      * Collapse all nodes
34396      */
34397     collapseAll : function(){
34398         this.root.collapse(true);
34399     },
34400
34401     /**
34402      * Returns the selection model used by this TreePanel
34403      */
34404     getSelectionModel : function(){
34405         if(!this.selModel){
34406             this.selModel = new Roo.tree.DefaultSelectionModel();
34407         }
34408         return this.selModel;
34409     },
34410
34411     /**
34412      * Retrieve an array of checked nodes, or an array of a specific attribute of checked nodes (e.g. "id")
34413      * @param {String} attribute (optional) Defaults to null (return the actual nodes)
34414      * @param {TreeNode} startNode (optional) The node to start from, defaults to the root
34415      * @return {Array}
34416      */
34417     getChecked : function(a, startNode){
34418         startNode = startNode || this.root;
34419         var r = [];
34420         var f = function(){
34421             if(this.attributes.checked){
34422                 r.push(!a ? this : (a == 'id' ? this.id : this.attributes[a]));
34423             }
34424         }
34425         startNode.cascade(f);
34426         return r;
34427     },
34428
34429     /**
34430      * Expands a specified path in this TreePanel. A path can be retrieved from a node with {@link Roo.data.Node#getPath}
34431      * @param {String} path
34432      * @param {String} attr (optional) The attribute used in the path (see {@link Roo.data.Node#getPath} for more info)
34433      * @param {Function} callback (optional) The callback to call when the expand is complete. The callback will be called with
34434      * (bSuccess, oLastNode) where bSuccess is if the expand was successful and oLastNode is the last node that was expanded.
34435      */
34436     expandPath : function(path, attr, callback){
34437         attr = attr || "id";
34438         var keys = path.split(this.pathSeparator);
34439         var curNode = this.root;
34440         if(curNode.attributes[attr] != keys[1]){ // invalid root
34441             if(callback){
34442                 callback(false, null);
34443             }
34444             return;
34445         }
34446         var index = 1;
34447         var f = function(){
34448             if(++index == keys.length){
34449                 if(callback){
34450                     callback(true, curNode);
34451                 }
34452                 return;
34453             }
34454             var c = curNode.findChild(attr, keys[index]);
34455             if(!c){
34456                 if(callback){
34457                     callback(false, curNode);
34458                 }
34459                 return;
34460             }
34461             curNode = c;
34462             c.expand(false, false, f);
34463         };
34464         curNode.expand(false, false, f);
34465     },
34466
34467     /**
34468      * Selects the node in this tree at the specified path. A path can be retrieved from a node with {@link Roo.data.Node#getPath}
34469      * @param {String} path
34470      * @param {String} attr (optional) The attribute used in the path (see {@link Roo.data.Node#getPath} for more info)
34471      * @param {Function} callback (optional) The callback to call when the selection is complete. The callback will be called with
34472      * (bSuccess, oSelNode) where bSuccess is if the selection was successful and oSelNode is the selected node.
34473      */
34474     selectPath : function(path, attr, callback){
34475         attr = attr || "id";
34476         var keys = path.split(this.pathSeparator);
34477         var v = keys.pop();
34478         if(keys.length > 0){
34479             var f = function(success, node){
34480                 if(success && node){
34481                     var n = node.findChild(attr, v);
34482                     if(n){
34483                         n.select();
34484                         if(callback){
34485                             callback(true, n);
34486                         }
34487                     }else if(callback){
34488                         callback(false, n);
34489                     }
34490                 }else{
34491                     if(callback){
34492                         callback(false, n);
34493                     }
34494                 }
34495             };
34496             this.expandPath(keys.join(this.pathSeparator), attr, f);
34497         }else{
34498             this.root.select();
34499             if(callback){
34500                 callback(true, this.root);
34501             }
34502         }
34503     },
34504
34505     getTreeEl : function(){
34506         return this.el;
34507     },
34508
34509     /**
34510      * Trigger rendering of this TreePanel
34511      */
34512     render : function(){
34513         if (this.innerCt) {
34514             return this; // stop it rendering more than once!!
34515         }
34516         
34517         this.innerCt = this.el.createChild({tag:"ul",
34518                cls:"x-tree-root-ct " +
34519                (this.lines ? "x-tree-lines" : "x-tree-no-lines")});
34520
34521         if(this.containerScroll){
34522             Roo.dd.ScrollManager.register(this.el);
34523         }
34524         if((this.enableDD || this.enableDrop) && !this.dropZone){
34525            /**
34526             * The dropZone used by this tree if drop is enabled
34527             * @type Roo.tree.TreeDropZone
34528             */
34529              this.dropZone = new Roo.tree.TreeDropZone(this, this.dropConfig || {
34530                ddGroup: this.ddGroup || "TreeDD", appendOnly: this.ddAppendOnly === true
34531            });
34532         }
34533         if((this.enableDD || this.enableDrag) && !this.dragZone){
34534            /**
34535             * The dragZone used by this tree if drag is enabled
34536             * @type Roo.tree.TreeDragZone
34537             */
34538             this.dragZone = new Roo.tree.TreeDragZone(this, this.dragConfig || {
34539                ddGroup: this.ddGroup || "TreeDD",
34540                scroll: this.ddScroll
34541            });
34542         }
34543         this.getSelectionModel().init(this);
34544         if (!this.root) {
34545             Roo.log("ROOT not set in tree");
34546             return this;
34547         }
34548         this.root.render();
34549         if(!this.rootVisible){
34550             this.root.renderChildren();
34551         }
34552         return this;
34553     }
34554 });/*
34555  * Based on:
34556  * Ext JS Library 1.1.1
34557  * Copyright(c) 2006-2007, Ext JS, LLC.
34558  *
34559  * Originally Released Under LGPL - original licence link has changed is not relivant.
34560  *
34561  * Fork - LGPL
34562  * <script type="text/javascript">
34563  */
34564  
34565
34566 /**
34567  * @class Roo.tree.DefaultSelectionModel
34568  * @extends Roo.util.Observable
34569  * The default single selection for a TreePanel.
34570  * @param {Object} cfg Configuration
34571  */
34572 Roo.tree.DefaultSelectionModel = function(cfg){
34573    this.selNode = null;
34574    
34575    
34576    
34577    this.addEvents({
34578        /**
34579         * @event selectionchange
34580         * Fires when the selected node changes
34581         * @param {DefaultSelectionModel} this
34582         * @param {TreeNode} node the new selection
34583         */
34584        "selectionchange" : true,
34585
34586        /**
34587         * @event beforeselect
34588         * Fires before the selected node changes, return false to cancel the change
34589         * @param {DefaultSelectionModel} this
34590         * @param {TreeNode} node the new selection
34591         * @param {TreeNode} node the old selection
34592         */
34593        "beforeselect" : true
34594    });
34595    
34596     Roo.tree.DefaultSelectionModel.superclass.constructor.call(this,cfg);
34597 };
34598
34599 Roo.extend(Roo.tree.DefaultSelectionModel, Roo.util.Observable, {
34600     init : function(tree){
34601         this.tree = tree;
34602         tree.getTreeEl().on("keydown", this.onKeyDown, this);
34603         tree.on("click", this.onNodeClick, this);
34604     },
34605     
34606     onNodeClick : function(node, e){
34607         if (e.ctrlKey && this.selNode == node)  {
34608             this.unselect(node);
34609             return;
34610         }
34611         this.select(node);
34612     },
34613     
34614     /**
34615      * Select a node.
34616      * @param {TreeNode} node The node to select
34617      * @return {TreeNode} The selected node
34618      */
34619     select : function(node){
34620         var last = this.selNode;
34621         if(last != node && this.fireEvent('beforeselect', this, node, last) !== false){
34622             if(last){
34623                 last.ui.onSelectedChange(false);
34624             }
34625             this.selNode = node;
34626             node.ui.onSelectedChange(true);
34627             this.fireEvent("selectionchange", this, node, last);
34628         }
34629         return node;
34630     },
34631     
34632     /**
34633      * Deselect a node.
34634      * @param {TreeNode} node The node to unselect
34635      */
34636     unselect : function(node){
34637         if(this.selNode == node){
34638             this.clearSelections();
34639         }    
34640     },
34641     
34642     /**
34643      * Clear all selections
34644      */
34645     clearSelections : function(){
34646         var n = this.selNode;
34647         if(n){
34648             n.ui.onSelectedChange(false);
34649             this.selNode = null;
34650             this.fireEvent("selectionchange", this, null);
34651         }
34652         return n;
34653     },
34654     
34655     /**
34656      * Get the selected node
34657      * @return {TreeNode} The selected node
34658      */
34659     getSelectedNode : function(){
34660         return this.selNode;    
34661     },
34662     
34663     /**
34664      * Returns true if the node is selected
34665      * @param {TreeNode} node The node to check
34666      * @return {Boolean}
34667      */
34668     isSelected : function(node){
34669         return this.selNode == node;  
34670     },
34671
34672     /**
34673      * Selects the node above the selected node in the tree, intelligently walking the nodes
34674      * @return TreeNode The new selection
34675      */
34676     selectPrevious : function(){
34677         var s = this.selNode || this.lastSelNode;
34678         if(!s){
34679             return null;
34680         }
34681         var ps = s.previousSibling;
34682         if(ps){
34683             if(!ps.isExpanded() || ps.childNodes.length < 1){
34684                 return this.select(ps);
34685             } else{
34686                 var lc = ps.lastChild;
34687                 while(lc && lc.isExpanded() && lc.childNodes.length > 0){
34688                     lc = lc.lastChild;
34689                 }
34690                 return this.select(lc);
34691             }
34692         } else if(s.parentNode && (this.tree.rootVisible || !s.parentNode.isRoot)){
34693             return this.select(s.parentNode);
34694         }
34695         return null;
34696     },
34697
34698     /**
34699      * Selects the node above the selected node in the tree, intelligently walking the nodes
34700      * @return TreeNode The new selection
34701      */
34702     selectNext : function(){
34703         var s = this.selNode || this.lastSelNode;
34704         if(!s){
34705             return null;
34706         }
34707         if(s.firstChild && s.isExpanded()){
34708              return this.select(s.firstChild);
34709          }else if(s.nextSibling){
34710              return this.select(s.nextSibling);
34711          }else if(s.parentNode){
34712             var newS = null;
34713             s.parentNode.bubble(function(){
34714                 if(this.nextSibling){
34715                     newS = this.getOwnerTree().selModel.select(this.nextSibling);
34716                     return false;
34717                 }
34718             });
34719             return newS;
34720          }
34721         return null;
34722     },
34723
34724     onKeyDown : function(e){
34725         var s = this.selNode || this.lastSelNode;
34726         // undesirable, but required
34727         var sm = this;
34728         if(!s){
34729             return;
34730         }
34731         var k = e.getKey();
34732         switch(k){
34733              case e.DOWN:
34734                  e.stopEvent();
34735                  this.selectNext();
34736              break;
34737              case e.UP:
34738                  e.stopEvent();
34739                  this.selectPrevious();
34740              break;
34741              case e.RIGHT:
34742                  e.preventDefault();
34743                  if(s.hasChildNodes()){
34744                      if(!s.isExpanded()){
34745                          s.expand();
34746                      }else if(s.firstChild){
34747                          this.select(s.firstChild, e);
34748                      }
34749                  }
34750              break;
34751              case e.LEFT:
34752                  e.preventDefault();
34753                  if(s.hasChildNodes() && s.isExpanded()){
34754                      s.collapse();
34755                  }else if(s.parentNode && (this.tree.rootVisible || s.parentNode != this.tree.getRootNode())){
34756                      this.select(s.parentNode, e);
34757                  }
34758              break;
34759         };
34760     }
34761 });
34762
34763 /**
34764  * @class Roo.tree.MultiSelectionModel
34765  * @extends Roo.util.Observable
34766  * Multi selection for a TreePanel.
34767  * @param {Object} cfg Configuration
34768  */
34769 Roo.tree.MultiSelectionModel = function(){
34770    this.selNodes = [];
34771    this.selMap = {};
34772    this.addEvents({
34773        /**
34774         * @event selectionchange
34775         * Fires when the selected nodes change
34776         * @param {MultiSelectionModel} this
34777         * @param {Array} nodes Array of the selected nodes
34778         */
34779        "selectionchange" : true
34780    });
34781    Roo.tree.MultiSelectionModel.superclass.constructor.call(this,cfg);
34782    
34783 };
34784
34785 Roo.extend(Roo.tree.MultiSelectionModel, Roo.util.Observable, {
34786     init : function(tree){
34787         this.tree = tree;
34788         tree.getTreeEl().on("keydown", this.onKeyDown, this);
34789         tree.on("click", this.onNodeClick, this);
34790     },
34791     
34792     onNodeClick : function(node, e){
34793         this.select(node, e, e.ctrlKey);
34794     },
34795     
34796     /**
34797      * Select a node.
34798      * @param {TreeNode} node The node to select
34799      * @param {EventObject} e (optional) An event associated with the selection
34800      * @param {Boolean} keepExisting True to retain existing selections
34801      * @return {TreeNode} The selected node
34802      */
34803     select : function(node, e, keepExisting){
34804         if(keepExisting !== true){
34805             this.clearSelections(true);
34806         }
34807         if(this.isSelected(node)){
34808             this.lastSelNode = node;
34809             return node;
34810         }
34811         this.selNodes.push(node);
34812         this.selMap[node.id] = node;
34813         this.lastSelNode = node;
34814         node.ui.onSelectedChange(true);
34815         this.fireEvent("selectionchange", this, this.selNodes);
34816         return node;
34817     },
34818     
34819     /**
34820      * Deselect a node.
34821      * @param {TreeNode} node The node to unselect
34822      */
34823     unselect : function(node){
34824         if(this.selMap[node.id]){
34825             node.ui.onSelectedChange(false);
34826             var sn = this.selNodes;
34827             var index = -1;
34828             if(sn.indexOf){
34829                 index = sn.indexOf(node);
34830             }else{
34831                 for(var i = 0, len = sn.length; i < len; i++){
34832                     if(sn[i] == node){
34833                         index = i;
34834                         break;
34835                     }
34836                 }
34837             }
34838             if(index != -1){
34839                 this.selNodes.splice(index, 1);
34840             }
34841             delete this.selMap[node.id];
34842             this.fireEvent("selectionchange", this, this.selNodes);
34843         }
34844     },
34845     
34846     /**
34847      * Clear all selections
34848      */
34849     clearSelections : function(suppressEvent){
34850         var sn = this.selNodes;
34851         if(sn.length > 0){
34852             for(var i = 0, len = sn.length; i < len; i++){
34853                 sn[i].ui.onSelectedChange(false);
34854             }
34855             this.selNodes = [];
34856             this.selMap = {};
34857             if(suppressEvent !== true){
34858                 this.fireEvent("selectionchange", this, this.selNodes);
34859             }
34860         }
34861     },
34862     
34863     /**
34864      * Returns true if the node is selected
34865      * @param {TreeNode} node The node to check
34866      * @return {Boolean}
34867      */
34868     isSelected : function(node){
34869         return this.selMap[node.id] ? true : false;  
34870     },
34871     
34872     /**
34873      * Returns an array of the selected nodes
34874      * @return {Array}
34875      */
34876     getSelectedNodes : function(){
34877         return this.selNodes;    
34878     },
34879
34880     onKeyDown : Roo.tree.DefaultSelectionModel.prototype.onKeyDown,
34881
34882     selectNext : Roo.tree.DefaultSelectionModel.prototype.selectNext,
34883
34884     selectPrevious : Roo.tree.DefaultSelectionModel.prototype.selectPrevious
34885 });/*
34886  * Based on:
34887  * Ext JS Library 1.1.1
34888  * Copyright(c) 2006-2007, Ext JS, LLC.
34889  *
34890  * Originally Released Under LGPL - original licence link has changed is not relivant.
34891  *
34892  * Fork - LGPL
34893  * <script type="text/javascript">
34894  */
34895  
34896 /**
34897  * @class Roo.tree.TreeNode
34898  * @extends Roo.data.Node
34899  * @cfg {String} text The text for this node
34900  * @cfg {Boolean} expanded true to start the node expanded
34901  * @cfg {Boolean} allowDrag false to make this node undraggable if DD is on (defaults to true)
34902  * @cfg {Boolean} allowDrop false if this node cannot be drop on
34903  * @cfg {Boolean} disabled true to start the node disabled
34904  * @cfg {String} icon The path to an icon for the node. The preferred way to do this
34905  *    is to use the cls or iconCls attributes and add the icon via a CSS background image.
34906  * @cfg {String} cls A css class to be added to the node
34907  * @cfg {String} iconCls A css class to be added to the nodes icon element for applying css background images
34908  * @cfg {String} href URL of the link used for the node (defaults to #)
34909  * @cfg {String} hrefTarget target frame for the link
34910  * @cfg {String} qtip An Ext QuickTip for the node
34911  * @cfg {String} qtipCfg An Ext QuickTip config for the node (used instead of qtip)
34912  * @cfg {Boolean} singleClickExpand True for single click expand on this node
34913  * @cfg {Function} uiProvider A UI <b>class</b> to use for this node (defaults to Roo.tree.TreeNodeUI)
34914  * @cfg {Boolean} checked True to render a checked checkbox for this node, false to render an unchecked checkbox
34915  * (defaults to undefined with no checkbox rendered)
34916  * @constructor
34917  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node
34918  */
34919 Roo.tree.TreeNode = function(attributes){
34920     attributes = attributes || {};
34921     if(typeof attributes == "string"){
34922         attributes = {text: attributes};
34923     }
34924     this.childrenRendered = false;
34925     this.rendered = false;
34926     Roo.tree.TreeNode.superclass.constructor.call(this, attributes);
34927     this.expanded = attributes.expanded === true;
34928     this.isTarget = attributes.isTarget !== false;
34929     this.draggable = attributes.draggable !== false && attributes.allowDrag !== false;
34930     this.allowChildren = attributes.allowChildren !== false && attributes.allowDrop !== false;
34931
34932     /**
34933      * Read-only. The text for this node. To change it use setText().
34934      * @type String
34935      */
34936     this.text = attributes.text;
34937     /**
34938      * True if this node is disabled.
34939      * @type Boolean
34940      */
34941     this.disabled = attributes.disabled === true;
34942
34943     this.addEvents({
34944         /**
34945         * @event textchange
34946         * Fires when the text for this node is changed
34947         * @param {Node} this This node
34948         * @param {String} text The new text
34949         * @param {String} oldText The old text
34950         */
34951         "textchange" : true,
34952         /**
34953         * @event beforeexpand
34954         * Fires before this node is expanded, return false to cancel.
34955         * @param {Node} this This node
34956         * @param {Boolean} deep
34957         * @param {Boolean} anim
34958         */
34959         "beforeexpand" : true,
34960         /**
34961         * @event beforecollapse
34962         * Fires before this node is collapsed, return false to cancel.
34963         * @param {Node} this This node
34964         * @param {Boolean} deep
34965         * @param {Boolean} anim
34966         */
34967         "beforecollapse" : true,
34968         /**
34969         * @event expand
34970         * Fires when this node is expanded
34971         * @param {Node} this This node
34972         */
34973         "expand" : true,
34974         /**
34975         * @event disabledchange
34976         * Fires when the disabled status of this node changes
34977         * @param {Node} this This node
34978         * @param {Boolean} disabled
34979         */
34980         "disabledchange" : true,
34981         /**
34982         * @event collapse
34983         * Fires when this node is collapsed
34984         * @param {Node} this This node
34985         */
34986         "collapse" : true,
34987         /**
34988         * @event beforeclick
34989         * Fires before click processing. Return false to cancel the default action.
34990         * @param {Node} this This node
34991         * @param {Roo.EventObject} e The event object
34992         */
34993         "beforeclick":true,
34994         /**
34995         * @event checkchange
34996         * Fires when a node with a checkbox's checked property changes
34997         * @param {Node} this This node
34998         * @param {Boolean} checked
34999         */
35000         "checkchange":true,
35001         /**
35002         * @event click
35003         * Fires when this node is clicked
35004         * @param {Node} this This node
35005         * @param {Roo.EventObject} e The event object
35006         */
35007         "click":true,
35008         /**
35009         * @event dblclick
35010         * Fires when this node is double clicked
35011         * @param {Node} this This node
35012         * @param {Roo.EventObject} e The event object
35013         */
35014         "dblclick":true,
35015         /**
35016         * @event contextmenu
35017         * Fires when this node is right clicked
35018         * @param {Node} this This node
35019         * @param {Roo.EventObject} e The event object
35020         */
35021         "contextmenu":true,
35022         /**
35023         * @event beforechildrenrendered
35024         * Fires right before the child nodes for this node are rendered
35025         * @param {Node} this This node
35026         */
35027         "beforechildrenrendered":true
35028     });
35029
35030     var uiClass = this.attributes.uiProvider || Roo.tree.TreeNodeUI;
35031
35032     /**
35033      * Read-only. The UI for this node
35034      * @type TreeNodeUI
35035      */
35036     this.ui = new uiClass(this);
35037     
35038     // finally support items[]
35039     if (typeof(this.attributes.items) == 'undefined' || !this.attributes.items) {
35040         return;
35041     }
35042     
35043     
35044     Roo.each(this.attributes.items, function(c) {
35045         this.appendChild(Roo.factory(c,Roo.Tree));
35046     }, this);
35047     delete this.attributes.items;
35048     
35049     
35050     
35051 };
35052 Roo.extend(Roo.tree.TreeNode, Roo.data.Node, {
35053     preventHScroll: true,
35054     /**
35055      * Returns true if this node is expanded
35056      * @return {Boolean}
35057      */
35058     isExpanded : function(){
35059         return this.expanded;
35060     },
35061
35062     /**
35063      * Returns the UI object for this node
35064      * @return {TreeNodeUI}
35065      */
35066     getUI : function(){
35067         return this.ui;
35068     },
35069
35070     // private override
35071     setFirstChild : function(node){
35072         var of = this.firstChild;
35073         Roo.tree.TreeNode.superclass.setFirstChild.call(this, node);
35074         if(this.childrenRendered && of && node != of){
35075             of.renderIndent(true, true);
35076         }
35077         if(this.rendered){
35078             this.renderIndent(true, true);
35079         }
35080     },
35081
35082     // private override
35083     setLastChild : function(node){
35084         var ol = this.lastChild;
35085         Roo.tree.TreeNode.superclass.setLastChild.call(this, node);
35086         if(this.childrenRendered && ol && node != ol){
35087             ol.renderIndent(true, true);
35088         }
35089         if(this.rendered){
35090             this.renderIndent(true, true);
35091         }
35092     },
35093
35094     // these methods are overridden to provide lazy rendering support
35095     // private override
35096     appendChild : function()
35097     {
35098         var node = Roo.tree.TreeNode.superclass.appendChild.apply(this, arguments);
35099         if(node && this.childrenRendered){
35100             node.render();
35101         }
35102         this.ui.updateExpandIcon();
35103         return node;
35104     },
35105
35106     // private override
35107     removeChild : function(node){
35108         this.ownerTree.getSelectionModel().unselect(node);
35109         Roo.tree.TreeNode.superclass.removeChild.apply(this, arguments);
35110         // if it's been rendered remove dom node
35111         if(this.childrenRendered){
35112             node.ui.remove();
35113         }
35114         if(this.childNodes.length < 1){
35115             this.collapse(false, false);
35116         }else{
35117             this.ui.updateExpandIcon();
35118         }
35119         if(!this.firstChild) {
35120             this.childrenRendered = false;
35121         }
35122         return node;
35123     },
35124
35125     // private override
35126     insertBefore : function(node, refNode){
35127         var newNode = Roo.tree.TreeNode.superclass.insertBefore.apply(this, arguments);
35128         if(newNode && refNode && this.childrenRendered){
35129             node.render();
35130         }
35131         this.ui.updateExpandIcon();
35132         return newNode;
35133     },
35134
35135     /**
35136      * Sets the text for this node
35137      * @param {String} text
35138      */
35139     setText : function(text){
35140         var oldText = this.text;
35141         this.text = text;
35142         this.attributes.text = text;
35143         if(this.rendered){ // event without subscribing
35144             this.ui.onTextChange(this, text, oldText);
35145         }
35146         this.fireEvent("textchange", this, text, oldText);
35147     },
35148
35149     /**
35150      * Triggers selection of this node
35151      */
35152     select : function(){
35153         this.getOwnerTree().getSelectionModel().select(this);
35154     },
35155
35156     /**
35157      * Triggers deselection of this node
35158      */
35159     unselect : function(){
35160         this.getOwnerTree().getSelectionModel().unselect(this);
35161     },
35162
35163     /**
35164      * Returns true if this node is selected
35165      * @return {Boolean}
35166      */
35167     isSelected : function(){
35168         return this.getOwnerTree().getSelectionModel().isSelected(this);
35169     },
35170
35171     /**
35172      * Expand this node.
35173      * @param {Boolean} deep (optional) True to expand all children as well
35174      * @param {Boolean} anim (optional) false to cancel the default animation
35175      * @param {Function} callback (optional) A callback to be called when
35176      * expanding this node completes (does not wait for deep expand to complete).
35177      * Called with 1 parameter, this node.
35178      */
35179     expand : function(deep, anim, callback){
35180         if(!this.expanded){
35181             if(this.fireEvent("beforeexpand", this, deep, anim) === false){
35182                 return;
35183             }
35184             if(!this.childrenRendered){
35185                 this.renderChildren();
35186             }
35187             this.expanded = true;
35188             
35189             if(!this.isHiddenRoot() && (this.getOwnerTree() && this.getOwnerTree().animate && anim !== false) || anim){
35190                 this.ui.animExpand(function(){
35191                     this.fireEvent("expand", this);
35192                     if(typeof callback == "function"){
35193                         callback(this);
35194                     }
35195                     if(deep === true){
35196                         this.expandChildNodes(true);
35197                     }
35198                 }.createDelegate(this));
35199                 return;
35200             }else{
35201                 this.ui.expand();
35202                 this.fireEvent("expand", this);
35203                 if(typeof callback == "function"){
35204                     callback(this);
35205                 }
35206             }
35207         }else{
35208            if(typeof callback == "function"){
35209                callback(this);
35210            }
35211         }
35212         if(deep === true){
35213             this.expandChildNodes(true);
35214         }
35215     },
35216
35217     isHiddenRoot : function(){
35218         return this.isRoot && !this.getOwnerTree().rootVisible;
35219     },
35220
35221     /**
35222      * Collapse this node.
35223      * @param {Boolean} deep (optional) True to collapse all children as well
35224      * @param {Boolean} anim (optional) false to cancel the default animation
35225      */
35226     collapse : function(deep, anim){
35227         if(this.expanded && !this.isHiddenRoot()){
35228             if(this.fireEvent("beforecollapse", this, deep, anim) === false){
35229                 return;
35230             }
35231             this.expanded = false;
35232             if((this.getOwnerTree().animate && anim !== false) || anim){
35233                 this.ui.animCollapse(function(){
35234                     this.fireEvent("collapse", this);
35235                     if(deep === true){
35236                         this.collapseChildNodes(true);
35237                     }
35238                 }.createDelegate(this));
35239                 return;
35240             }else{
35241                 this.ui.collapse();
35242                 this.fireEvent("collapse", this);
35243             }
35244         }
35245         if(deep === true){
35246             var cs = this.childNodes;
35247             for(var i = 0, len = cs.length; i < len; i++) {
35248                 cs[i].collapse(true, false);
35249             }
35250         }
35251     },
35252
35253     // private
35254     delayedExpand : function(delay){
35255         if(!this.expandProcId){
35256             this.expandProcId = this.expand.defer(delay, this);
35257         }
35258     },
35259
35260     // private
35261     cancelExpand : function(){
35262         if(this.expandProcId){
35263             clearTimeout(this.expandProcId);
35264         }
35265         this.expandProcId = false;
35266     },
35267
35268     /**
35269      * Toggles expanded/collapsed state of the node
35270      */
35271     toggle : function(){
35272         if(this.expanded){
35273             this.collapse();
35274         }else{
35275             this.expand();
35276         }
35277     },
35278
35279     /**
35280      * Ensures all parent nodes are expanded
35281      */
35282     ensureVisible : function(callback){
35283         var tree = this.getOwnerTree();
35284         tree.expandPath(this.parentNode.getPath(), false, function(){
35285             tree.getTreeEl().scrollChildIntoView(this.ui.anchor);
35286             Roo.callback(callback);
35287         }.createDelegate(this));
35288     },
35289
35290     /**
35291      * Expand all child nodes
35292      * @param {Boolean} deep (optional) true if the child nodes should also expand their child nodes
35293      */
35294     expandChildNodes : function(deep){
35295         var cs = this.childNodes;
35296         for(var i = 0, len = cs.length; i < len; i++) {
35297                 cs[i].expand(deep);
35298         }
35299     },
35300
35301     /**
35302      * Collapse all child nodes
35303      * @param {Boolean} deep (optional) true if the child nodes should also collapse their child nodes
35304      */
35305     collapseChildNodes : function(deep){
35306         var cs = this.childNodes;
35307         for(var i = 0, len = cs.length; i < len; i++) {
35308                 cs[i].collapse(deep);
35309         }
35310     },
35311
35312     /**
35313      * Disables this node
35314      */
35315     disable : function(){
35316         this.disabled = true;
35317         this.unselect();
35318         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
35319             this.ui.onDisableChange(this, true);
35320         }
35321         this.fireEvent("disabledchange", this, true);
35322     },
35323
35324     /**
35325      * Enables this node
35326      */
35327     enable : function(){
35328         this.disabled = false;
35329         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
35330             this.ui.onDisableChange(this, false);
35331         }
35332         this.fireEvent("disabledchange", this, false);
35333     },
35334
35335     // private
35336     renderChildren : function(suppressEvent){
35337         if(suppressEvent !== false){
35338             this.fireEvent("beforechildrenrendered", this);
35339         }
35340         var cs = this.childNodes;
35341         for(var i = 0, len = cs.length; i < len; i++){
35342             cs[i].render(true);
35343         }
35344         this.childrenRendered = true;
35345     },
35346
35347     // private
35348     sort : function(fn, scope){
35349         Roo.tree.TreeNode.superclass.sort.apply(this, arguments);
35350         if(this.childrenRendered){
35351             var cs = this.childNodes;
35352             for(var i = 0, len = cs.length; i < len; i++){
35353                 cs[i].render(true);
35354             }
35355         }
35356     },
35357
35358     // private
35359     render : function(bulkRender){
35360         this.ui.render(bulkRender);
35361         if(!this.rendered){
35362             this.rendered = true;
35363             if(this.expanded){
35364                 this.expanded = false;
35365                 this.expand(false, false);
35366             }
35367         }
35368     },
35369
35370     // private
35371     renderIndent : function(deep, refresh){
35372         if(refresh){
35373             this.ui.childIndent = null;
35374         }
35375         this.ui.renderIndent();
35376         if(deep === true && this.childrenRendered){
35377             var cs = this.childNodes;
35378             for(var i = 0, len = cs.length; i < len; i++){
35379                 cs[i].renderIndent(true, refresh);
35380             }
35381         }
35382     }
35383 });/*
35384  * Based on:
35385  * Ext JS Library 1.1.1
35386  * Copyright(c) 2006-2007, Ext JS, LLC.
35387  *
35388  * Originally Released Under LGPL - original licence link has changed is not relivant.
35389  *
35390  * Fork - LGPL
35391  * <script type="text/javascript">
35392  */
35393  
35394 /**
35395  * @class Roo.tree.AsyncTreeNode
35396  * @extends Roo.tree.TreeNode
35397  * @cfg {TreeLoader} loader A TreeLoader to be used by this node (defaults to the loader defined on the tree)
35398  * @constructor
35399  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node 
35400  */
35401  Roo.tree.AsyncTreeNode = function(config){
35402     this.loaded = false;
35403     this.loading = false;
35404     Roo.tree.AsyncTreeNode.superclass.constructor.apply(this, arguments);
35405     /**
35406     * @event beforeload
35407     * Fires before this node is loaded, return false to cancel
35408     * @param {Node} this This node
35409     */
35410     this.addEvents({'beforeload':true, 'load': true});
35411     /**
35412     * @event load
35413     * Fires when this node is loaded
35414     * @param {Node} this This node
35415     */
35416     /**
35417      * The loader used by this node (defaults to using the tree's defined loader)
35418      * @type TreeLoader
35419      * @property loader
35420      */
35421 };
35422 Roo.extend(Roo.tree.AsyncTreeNode, Roo.tree.TreeNode, {
35423     expand : function(deep, anim, callback){
35424         if(this.loading){ // if an async load is already running, waiting til it's done
35425             var timer;
35426             var f = function(){
35427                 if(!this.loading){ // done loading
35428                     clearInterval(timer);
35429                     this.expand(deep, anim, callback);
35430                 }
35431             }.createDelegate(this);
35432             timer = setInterval(f, 200);
35433             return;
35434         }
35435         if(!this.loaded){
35436             if(this.fireEvent("beforeload", this) === false){
35437                 return;
35438             }
35439             this.loading = true;
35440             this.ui.beforeLoad(this);
35441             var loader = this.loader || this.attributes.loader || this.getOwnerTree().getLoader();
35442             if(loader){
35443                 loader.load(this, this.loadComplete.createDelegate(this, [deep, anim, callback]));
35444                 return;
35445             }
35446         }
35447         Roo.tree.AsyncTreeNode.superclass.expand.call(this, deep, anim, callback);
35448     },
35449     
35450     /**
35451      * Returns true if this node is currently loading
35452      * @return {Boolean}
35453      */
35454     isLoading : function(){
35455         return this.loading;  
35456     },
35457     
35458     loadComplete : function(deep, anim, callback){
35459         this.loading = false;
35460         this.loaded = true;
35461         this.ui.afterLoad(this);
35462         this.fireEvent("load", this);
35463         this.expand(deep, anim, callback);
35464     },
35465     
35466     /**
35467      * Returns true if this node has been loaded
35468      * @return {Boolean}
35469      */
35470     isLoaded : function(){
35471         return this.loaded;
35472     },
35473     
35474     hasChildNodes : function(){
35475         if(!this.isLeaf() && !this.loaded){
35476             return true;
35477         }else{
35478             return Roo.tree.AsyncTreeNode.superclass.hasChildNodes.call(this);
35479         }
35480     },
35481
35482     /**
35483      * Trigger a reload for this node
35484      * @param {Function} callback
35485      */
35486     reload : function(callback){
35487         this.collapse(false, false);
35488         while(this.firstChild){
35489             this.removeChild(this.firstChild);
35490         }
35491         this.childrenRendered = false;
35492         this.loaded = false;
35493         if(this.isHiddenRoot()){
35494             this.expanded = false;
35495         }
35496         this.expand(false, false, callback);
35497     }
35498 });/*
35499  * Based on:
35500  * Ext JS Library 1.1.1
35501  * Copyright(c) 2006-2007, Ext JS, LLC.
35502  *
35503  * Originally Released Under LGPL - original licence link has changed is not relivant.
35504  *
35505  * Fork - LGPL
35506  * <script type="text/javascript">
35507  */
35508  
35509 /**
35510  * @class Roo.tree.TreeNodeUI
35511  * @constructor
35512  * @param {Object} node The node to render
35513  * The TreeNode UI implementation is separate from the
35514  * tree implementation. Unless you are customizing the tree UI,
35515  * you should never have to use this directly.
35516  */
35517 Roo.tree.TreeNodeUI = function(node){
35518     this.node = node;
35519     this.rendered = false;
35520     this.animating = false;
35521     this.emptyIcon = Roo.BLANK_IMAGE_URL;
35522 };
35523
35524 Roo.tree.TreeNodeUI.prototype = {
35525     removeChild : function(node){
35526         if(this.rendered){
35527             this.ctNode.removeChild(node.ui.getEl());
35528         }
35529     },
35530
35531     beforeLoad : function(){
35532          this.addClass("x-tree-node-loading");
35533     },
35534
35535     afterLoad : function(){
35536          this.removeClass("x-tree-node-loading");
35537     },
35538
35539     onTextChange : function(node, text, oldText){
35540         if(this.rendered){
35541             this.textNode.innerHTML = text;
35542         }
35543     },
35544
35545     onDisableChange : function(node, state){
35546         this.disabled = state;
35547         if(state){
35548             this.addClass("x-tree-node-disabled");
35549         }else{
35550             this.removeClass("x-tree-node-disabled");
35551         }
35552     },
35553
35554     onSelectedChange : function(state){
35555         if(state){
35556             this.focus();
35557             this.addClass("x-tree-selected");
35558         }else{
35559             //this.blur();
35560             this.removeClass("x-tree-selected");
35561         }
35562     },
35563
35564     onMove : function(tree, node, oldParent, newParent, index, refNode){
35565         this.childIndent = null;
35566         if(this.rendered){
35567             var targetNode = newParent.ui.getContainer();
35568             if(!targetNode){//target not rendered
35569                 this.holder = document.createElement("div");
35570                 this.holder.appendChild(this.wrap);
35571                 return;
35572             }
35573             var insertBefore = refNode ? refNode.ui.getEl() : null;
35574             if(insertBefore){
35575                 targetNode.insertBefore(this.wrap, insertBefore);
35576             }else{
35577                 targetNode.appendChild(this.wrap);
35578             }
35579             this.node.renderIndent(true);
35580         }
35581     },
35582
35583     addClass : function(cls){
35584         if(this.elNode){
35585             Roo.fly(this.elNode).addClass(cls);
35586         }
35587     },
35588
35589     removeClass : function(cls){
35590         if(this.elNode){
35591             Roo.fly(this.elNode).removeClass(cls);
35592         }
35593     },
35594
35595     remove : function(){
35596         if(this.rendered){
35597             this.holder = document.createElement("div");
35598             this.holder.appendChild(this.wrap);
35599         }
35600     },
35601
35602     fireEvent : function(){
35603         return this.node.fireEvent.apply(this.node, arguments);
35604     },
35605
35606     initEvents : function(){
35607         this.node.on("move", this.onMove, this);
35608         var E = Roo.EventManager;
35609         var a = this.anchor;
35610
35611         var el = Roo.fly(a, '_treeui');
35612
35613         if(Roo.isOpera){ // opera render bug ignores the CSS
35614             el.setStyle("text-decoration", "none");
35615         }
35616
35617         el.on("click", this.onClick, this);
35618         el.on("dblclick", this.onDblClick, this);
35619
35620         if(this.checkbox){
35621             Roo.EventManager.on(this.checkbox,
35622                     Roo.isIE ? 'click' : 'change', this.onCheckChange, this);
35623         }
35624
35625         el.on("contextmenu", this.onContextMenu, this);
35626
35627         var icon = Roo.fly(this.iconNode);
35628         icon.on("click", this.onClick, this);
35629         icon.on("dblclick", this.onDblClick, this);
35630         icon.on("contextmenu", this.onContextMenu, this);
35631         E.on(this.ecNode, "click", this.ecClick, this, true);
35632
35633         if(this.node.disabled){
35634             this.addClass("x-tree-node-disabled");
35635         }
35636         if(this.node.hidden){
35637             this.addClass("x-tree-node-disabled");
35638         }
35639         var ot = this.node.getOwnerTree();
35640         var dd = ot ? (ot.enableDD || ot.enableDrag || ot.enableDrop) : false;
35641         if(dd && (!this.node.isRoot || ot.rootVisible)){
35642             Roo.dd.Registry.register(this.elNode, {
35643                 node: this.node,
35644                 handles: this.getDDHandles(),
35645                 isHandle: false
35646             });
35647         }
35648     },
35649
35650     getDDHandles : function(){
35651         return [this.iconNode, this.textNode];
35652     },
35653
35654     hide : function(){
35655         if(this.rendered){
35656             this.wrap.style.display = "none";
35657         }
35658     },
35659
35660     show : function(){
35661         if(this.rendered){
35662             this.wrap.style.display = "";
35663         }
35664     },
35665
35666     onContextMenu : function(e){
35667         if (this.node.hasListener("contextmenu") || this.node.getOwnerTree().hasListener("contextmenu")) {
35668             e.preventDefault();
35669             this.focus();
35670             this.fireEvent("contextmenu", this.node, e);
35671         }
35672     },
35673
35674     onClick : function(e){
35675         if(this.dropping){
35676             e.stopEvent();
35677             return;
35678         }
35679         if(this.fireEvent("beforeclick", this.node, e) !== false){
35680             if(!this.disabled && this.node.attributes.href){
35681                 this.fireEvent("click", this.node, e);
35682                 return;
35683             }
35684             e.preventDefault();
35685             if(this.disabled){
35686                 return;
35687             }
35688
35689             if(this.node.attributes.singleClickExpand && !this.animating && this.node.hasChildNodes()){
35690                 this.node.toggle();
35691             }
35692
35693             this.fireEvent("click", this.node, e);
35694         }else{
35695             e.stopEvent();
35696         }
35697     },
35698
35699     onDblClick : function(e){
35700         e.preventDefault();
35701         if(this.disabled){
35702             return;
35703         }
35704         if(this.checkbox){
35705             this.toggleCheck();
35706         }
35707         if(!this.animating && this.node.hasChildNodes()){
35708             this.node.toggle();
35709         }
35710         this.fireEvent("dblclick", this.node, e);
35711     },
35712
35713     onCheckChange : function(){
35714         var checked = this.checkbox.checked;
35715         this.node.attributes.checked = checked;
35716         this.fireEvent('checkchange', this.node, checked);
35717     },
35718
35719     ecClick : function(e){
35720         if(!this.animating && this.node.hasChildNodes()){
35721             this.node.toggle();
35722         }
35723     },
35724
35725     startDrop : function(){
35726         this.dropping = true;
35727     },
35728
35729     // delayed drop so the click event doesn't get fired on a drop
35730     endDrop : function(){
35731        setTimeout(function(){
35732            this.dropping = false;
35733        }.createDelegate(this), 50);
35734     },
35735
35736     expand : function(){
35737         this.updateExpandIcon();
35738         this.ctNode.style.display = "";
35739     },
35740
35741     focus : function(){
35742         if(!this.node.preventHScroll){
35743             try{this.anchor.focus();
35744             }catch(e){}
35745         }else if(!Roo.isIE){
35746             try{
35747                 var noscroll = this.node.getOwnerTree().getTreeEl().dom;
35748                 var l = noscroll.scrollLeft;
35749                 this.anchor.focus();
35750                 noscroll.scrollLeft = l;
35751             }catch(e){}
35752         }
35753     },
35754
35755     toggleCheck : function(value){
35756         var cb = this.checkbox;
35757         if(cb){
35758             cb.checked = (value === undefined ? !cb.checked : value);
35759         }
35760     },
35761
35762     blur : function(){
35763         try{
35764             this.anchor.blur();
35765         }catch(e){}
35766     },
35767
35768     animExpand : function(callback){
35769         var ct = Roo.get(this.ctNode);
35770         ct.stopFx();
35771         if(!this.node.hasChildNodes()){
35772             this.updateExpandIcon();
35773             this.ctNode.style.display = "";
35774             Roo.callback(callback);
35775             return;
35776         }
35777         this.animating = true;
35778         this.updateExpandIcon();
35779
35780         ct.slideIn('t', {
35781            callback : function(){
35782                this.animating = false;
35783                Roo.callback(callback);
35784             },
35785             scope: this,
35786             duration: this.node.ownerTree.duration || .25
35787         });
35788     },
35789
35790     highlight : function(){
35791         var tree = this.node.getOwnerTree();
35792         Roo.fly(this.wrap).highlight(
35793             tree.hlColor || "C3DAF9",
35794             {endColor: tree.hlBaseColor}
35795         );
35796     },
35797
35798     collapse : function(){
35799         this.updateExpandIcon();
35800         this.ctNode.style.display = "none";
35801     },
35802
35803     animCollapse : function(callback){
35804         var ct = Roo.get(this.ctNode);
35805         ct.enableDisplayMode('block');
35806         ct.stopFx();
35807
35808         this.animating = true;
35809         this.updateExpandIcon();
35810
35811         ct.slideOut('t', {
35812             callback : function(){
35813                this.animating = false;
35814                Roo.callback(callback);
35815             },
35816             scope: this,
35817             duration: this.node.ownerTree.duration || .25
35818         });
35819     },
35820
35821     getContainer : function(){
35822         return this.ctNode;
35823     },
35824
35825     getEl : function(){
35826         return this.wrap;
35827     },
35828
35829     appendDDGhost : function(ghostNode){
35830         ghostNode.appendChild(this.elNode.cloneNode(true));
35831     },
35832
35833     getDDRepairXY : function(){
35834         return Roo.lib.Dom.getXY(this.iconNode);
35835     },
35836
35837     onRender : function(){
35838         this.render();
35839     },
35840
35841     render : function(bulkRender){
35842         var n = this.node, a = n.attributes;
35843         var targetNode = n.parentNode ?
35844               n.parentNode.ui.getContainer() : n.ownerTree.innerCt.dom;
35845
35846         if(!this.rendered){
35847             this.rendered = true;
35848
35849             this.renderElements(n, a, targetNode, bulkRender);
35850
35851             if(a.qtip){
35852                if(this.textNode.setAttributeNS){
35853                    this.textNode.setAttributeNS("ext", "qtip", a.qtip);
35854                    if(a.qtipTitle){
35855                        this.textNode.setAttributeNS("ext", "qtitle", a.qtipTitle);
35856                    }
35857                }else{
35858                    this.textNode.setAttribute("ext:qtip", a.qtip);
35859                    if(a.qtipTitle){
35860                        this.textNode.setAttribute("ext:qtitle", a.qtipTitle);
35861                    }
35862                }
35863             }else if(a.qtipCfg){
35864                 a.qtipCfg.target = Roo.id(this.textNode);
35865                 Roo.QuickTips.register(a.qtipCfg);
35866             }
35867             this.initEvents();
35868             if(!this.node.expanded){
35869                 this.updateExpandIcon();
35870             }
35871         }else{
35872             if(bulkRender === true) {
35873                 targetNode.appendChild(this.wrap);
35874             }
35875         }
35876     },
35877
35878     renderElements : function(n, a, targetNode, bulkRender)
35879     {
35880         // add some indent caching, this helps performance when rendering a large tree
35881         this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
35882         var t = n.getOwnerTree();
35883         var txt = t && t.renderer ? t.renderer(n.attributes) : Roo.util.Format.htmlEncode(n.text);
35884         if (typeof(n.attributes.html) != 'undefined') {
35885             txt = n.attributes.html;
35886         }
35887         var tip = t && t.rendererTip ? t.rendererTip(n.attributes) : txt;
35888         var cb = typeof a.checked == 'boolean';
35889         var href = a.href ? a.href : Roo.isGecko ? "" : "#";
35890         var buf = ['<li class="x-tree-node"><div class="x-tree-node-el ', a.cls,'">',
35891             '<span class="x-tree-node-indent">',this.indentMarkup,"</span>",
35892             '<img src="', this.emptyIcon, '" class="x-tree-ec-icon" />',
35893             '<img src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',(a.icon ? " x-tree-node-inline-icon" : ""),(a.iconCls ? " "+a.iconCls : ""),'" unselectable="on" />',
35894             cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + (a.checked ? 'checked="checked" />' : ' />')) : '',
35895             '<a hidefocus="on" href="',href,'" tabIndex="1" ',
35896              a.hrefTarget ? ' target="'+a.hrefTarget+'"' : "", 
35897                 '><span unselectable="on" qtip="' , tip ,'">',txt,"</span></a></div>",
35898             '<ul class="x-tree-node-ct" style="display:none;"></ul>',
35899             "</li>"];
35900
35901         if(bulkRender !== true && n.nextSibling && n.nextSibling.ui.getEl()){
35902             this.wrap = Roo.DomHelper.insertHtml("beforeBegin",
35903                                 n.nextSibling.ui.getEl(), buf.join(""));
35904         }else{
35905             this.wrap = Roo.DomHelper.insertHtml("beforeEnd", targetNode, buf.join(""));
35906         }
35907
35908         this.elNode = this.wrap.childNodes[0];
35909         this.ctNode = this.wrap.childNodes[1];
35910         var cs = this.elNode.childNodes;
35911         this.indentNode = cs[0];
35912         this.ecNode = cs[1];
35913         this.iconNode = cs[2];
35914         var index = 3;
35915         if(cb){
35916             this.checkbox = cs[3];
35917             index++;
35918         }
35919         this.anchor = cs[index];
35920         this.textNode = cs[index].firstChild;
35921     },
35922
35923     getAnchor : function(){
35924         return this.anchor;
35925     },
35926
35927     getTextEl : function(){
35928         return this.textNode;
35929     },
35930
35931     getIconEl : function(){
35932         return this.iconNode;
35933     },
35934
35935     isChecked : function(){
35936         return this.checkbox ? this.checkbox.checked : false;
35937     },
35938
35939     updateExpandIcon : function(){
35940         if(this.rendered){
35941             var n = this.node, c1, c2;
35942             var cls = n.isLast() ? "x-tree-elbow-end" : "x-tree-elbow";
35943             var hasChild = n.hasChildNodes();
35944             if(hasChild){
35945                 if(n.expanded){
35946                     cls += "-minus";
35947                     c1 = "x-tree-node-collapsed";
35948                     c2 = "x-tree-node-expanded";
35949                 }else{
35950                     cls += "-plus";
35951                     c1 = "x-tree-node-expanded";
35952                     c2 = "x-tree-node-collapsed";
35953                 }
35954                 if(this.wasLeaf){
35955                     this.removeClass("x-tree-node-leaf");
35956                     this.wasLeaf = false;
35957                 }
35958                 if(this.c1 != c1 || this.c2 != c2){
35959                     Roo.fly(this.elNode).replaceClass(c1, c2);
35960                     this.c1 = c1; this.c2 = c2;
35961                 }
35962             }else{
35963                 // this changes non-leafs into leafs if they have no children.
35964                 // it's not very rational behaviour..
35965                 
35966                 if(!this.wasLeaf && this.node.leaf){
35967                     Roo.fly(this.elNode).replaceClass("x-tree-node-expanded", "x-tree-node-leaf");
35968                     delete this.c1;
35969                     delete this.c2;
35970                     this.wasLeaf = true;
35971                 }
35972             }
35973             var ecc = "x-tree-ec-icon "+cls;
35974             if(this.ecc != ecc){
35975                 this.ecNode.className = ecc;
35976                 this.ecc = ecc;
35977             }
35978         }
35979     },
35980
35981     getChildIndent : function(){
35982         if(!this.childIndent){
35983             var buf = [];
35984             var p = this.node;
35985             while(p){
35986                 if(!p.isRoot || (p.isRoot && p.ownerTree.rootVisible)){
35987                     if(!p.isLast()) {
35988                         buf.unshift('<img src="'+this.emptyIcon+'" class="x-tree-elbow-line" />');
35989                     } else {
35990                         buf.unshift('<img src="'+this.emptyIcon+'" class="x-tree-icon" />');
35991                     }
35992                 }
35993                 p = p.parentNode;
35994             }
35995             this.childIndent = buf.join("");
35996         }
35997         return this.childIndent;
35998     },
35999
36000     renderIndent : function(){
36001         if(this.rendered){
36002             var indent = "";
36003             var p = this.node.parentNode;
36004             if(p){
36005                 indent = p.ui.getChildIndent();
36006             }
36007             if(this.indentMarkup != indent){ // don't rerender if not required
36008                 this.indentNode.innerHTML = indent;
36009                 this.indentMarkup = indent;
36010             }
36011             this.updateExpandIcon();
36012         }
36013     }
36014 };
36015
36016 Roo.tree.RootTreeNodeUI = function(){
36017     Roo.tree.RootTreeNodeUI.superclass.constructor.apply(this, arguments);
36018 };
36019 Roo.extend(Roo.tree.RootTreeNodeUI, Roo.tree.TreeNodeUI, {
36020     render : function(){
36021         if(!this.rendered){
36022             var targetNode = this.node.ownerTree.innerCt.dom;
36023             this.node.expanded = true;
36024             targetNode.innerHTML = '<div class="x-tree-root-node"></div>';
36025             this.wrap = this.ctNode = targetNode.firstChild;
36026         }
36027     },
36028     collapse : function(){
36029     },
36030     expand : function(){
36031     }
36032 });/*
36033  * Based on:
36034  * Ext JS Library 1.1.1
36035  * Copyright(c) 2006-2007, Ext JS, LLC.
36036  *
36037  * Originally Released Under LGPL - original licence link has changed is not relivant.
36038  *
36039  * Fork - LGPL
36040  * <script type="text/javascript">
36041  */
36042 /**
36043  * @class Roo.tree.TreeLoader
36044  * @extends Roo.util.Observable
36045  * A TreeLoader provides for lazy loading of an {@link Roo.tree.TreeNode}'s child
36046  * nodes from a specified URL. The response must be a javascript Array definition
36047  * who's elements are node definition objects. eg:
36048  * <pre><code>
36049 {  success : true,
36050    data :      [
36051    
36052     { 'id': 1, 'text': 'A folder Node', 'leaf': false },
36053     { 'id': 2, 'text': 'A leaf Node', 'leaf': true }
36054     ]
36055 }
36056
36057
36058 </code></pre>
36059  * <br><br>
36060  * The old style respose with just an array is still supported, but not recommended.
36061  * <br><br>
36062  *
36063  * A server request is sent, and child nodes are loaded only when a node is expanded.
36064  * The loading node's id is passed to the server under the parameter name "node" to
36065  * enable the server to produce the correct child nodes.
36066  * <br><br>
36067  * To pass extra parameters, an event handler may be attached to the "beforeload"
36068  * event, and the parameters specified in the TreeLoader's baseParams property:
36069  * <pre><code>
36070     myTreeLoader.on("beforeload", function(treeLoader, node) {
36071         this.baseParams.category = node.attributes.category;
36072     }, this);
36073     
36074 </code></pre>
36075  *
36076  * This would pass an HTTP parameter called "category" to the server containing
36077  * the value of the Node's "category" attribute.
36078  * @constructor
36079  * Creates a new Treeloader.
36080  * @param {Object} config A config object containing config properties.
36081  */
36082 Roo.tree.TreeLoader = function(config){
36083     this.baseParams = {};
36084     this.requestMethod = "POST";
36085     Roo.apply(this, config);
36086
36087     this.addEvents({
36088     
36089         /**
36090          * @event beforeload
36091          * Fires before a network request is made to retrieve the Json text which specifies a node's children.
36092          * @param {Object} This TreeLoader object.
36093          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36094          * @param {Object} callback The callback function specified in the {@link #load} call.
36095          */
36096         beforeload : true,
36097         /**
36098          * @event load
36099          * Fires when the node has been successfuly loaded.
36100          * @param {Object} This TreeLoader object.
36101          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36102          * @param {Object} response The response object containing the data from the server.
36103          */
36104         load : true,
36105         /**
36106          * @event loadexception
36107          * Fires if the network request failed.
36108          * @param {Object} This TreeLoader object.
36109          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36110          * @param {Object} response The response object containing the data from the server.
36111          */
36112         loadexception : true,
36113         /**
36114          * @event create
36115          * Fires before a node is created, enabling you to return custom Node types 
36116          * @param {Object} This TreeLoader object.
36117          * @param {Object} attr - the data returned from the AJAX call (modify it to suit)
36118          */
36119         create : true
36120     });
36121
36122     Roo.tree.TreeLoader.superclass.constructor.call(this);
36123 };
36124
36125 Roo.extend(Roo.tree.TreeLoader, Roo.util.Observable, {
36126     /**
36127     * @cfg {String} dataUrl The URL from which to request a Json string which
36128     * specifies an array of node definition object representing the child nodes
36129     * to be loaded.
36130     */
36131     /**
36132     * @cfg {String} requestMethod either GET or POST
36133     * defaults to POST (due to BC)
36134     * to be loaded.
36135     */
36136     /**
36137     * @cfg {Object} baseParams (optional) An object containing properties which
36138     * specify HTTP parameters to be passed to each request for child nodes.
36139     */
36140     /**
36141     * @cfg {Object} baseAttrs (optional) An object containing attributes to be added to all nodes
36142     * created by this loader. If the attributes sent by the server have an attribute in this object,
36143     * they take priority.
36144     */
36145     /**
36146     * @cfg {Object} uiProviders (optional) An object containing properties which
36147     * 
36148     * DEPRECATED - use 'create' event handler to modify attributes - which affect creation.
36149     * specify custom {@link Roo.tree.TreeNodeUI} implementations. If the optional
36150     * <i>uiProvider</i> attribute of a returned child node is a string rather
36151     * than a reference to a TreeNodeUI implementation, this that string value
36152     * is used as a property name in the uiProviders object. You can define the provider named
36153     * 'default' , and this will be used for all nodes (if no uiProvider is delivered by the node data)
36154     */
36155     uiProviders : {},
36156
36157     /**
36158     * @cfg {Boolean} clearOnLoad (optional) Default to true. Remove previously existing
36159     * child nodes before loading.
36160     */
36161     clearOnLoad : true,
36162
36163     /**
36164     * @cfg {String} root (optional) Default to false. Use this to read data from an object 
36165     * property on loading, rather than expecting an array. (eg. more compatible to a standard
36166     * Grid query { data : [ .....] }
36167     */
36168     
36169     root : false,
36170      /**
36171     * @cfg {String} queryParam (optional) 
36172     * Name of the query as it will be passed on the querystring (defaults to 'node')
36173     * eg. the request will be ?node=[id]
36174     */
36175     
36176     
36177     queryParam: false,
36178     
36179     /**
36180      * Load an {@link Roo.tree.TreeNode} from the URL specified in the constructor.
36181      * This is called automatically when a node is expanded, but may be used to reload
36182      * a node (or append new children if the {@link #clearOnLoad} option is false.)
36183      * @param {Roo.tree.TreeNode} node
36184      * @param {Function} callback
36185      */
36186     load : function(node, callback){
36187         if(this.clearOnLoad){
36188             while(node.firstChild){
36189                 node.removeChild(node.firstChild);
36190             }
36191         }
36192         if(node.attributes.children){ // preloaded json children
36193             var cs = node.attributes.children;
36194             for(var i = 0, len = cs.length; i < len; i++){
36195                 node.appendChild(this.createNode(cs[i]));
36196             }
36197             if(typeof callback == "function"){
36198                 callback();
36199             }
36200         }else if(this.dataUrl){
36201             this.requestData(node, callback);
36202         }
36203     },
36204
36205     getParams: function(node){
36206         var buf = [], bp = this.baseParams;
36207         for(var key in bp){
36208             if(typeof bp[key] != "function"){
36209                 buf.push(encodeURIComponent(key), "=", encodeURIComponent(bp[key]), "&");
36210             }
36211         }
36212         var n = this.queryParam === false ? 'node' : this.queryParam;
36213         buf.push(n + "=", encodeURIComponent(node.id));
36214         return buf.join("");
36215     },
36216
36217     requestData : function(node, callback){
36218         if(this.fireEvent("beforeload", this, node, callback) !== false){
36219             this.transId = Roo.Ajax.request({
36220                 method:this.requestMethod,
36221                 url: this.dataUrl||this.url,
36222                 success: this.handleResponse,
36223                 failure: this.handleFailure,
36224                 scope: this,
36225                 argument: {callback: callback, node: node},
36226                 params: this.getParams(node)
36227             });
36228         }else{
36229             // if the load is cancelled, make sure we notify
36230             // the node that we are done
36231             if(typeof callback == "function"){
36232                 callback();
36233             }
36234         }
36235     },
36236
36237     isLoading : function(){
36238         return this.transId ? true : false;
36239     },
36240
36241     abort : function(){
36242         if(this.isLoading()){
36243             Roo.Ajax.abort(this.transId);
36244         }
36245     },
36246
36247     // private
36248     createNode : function(attr)
36249     {
36250         // apply baseAttrs, nice idea Corey!
36251         if(this.baseAttrs){
36252             Roo.applyIf(attr, this.baseAttrs);
36253         }
36254         if(this.applyLoader !== false){
36255             attr.loader = this;
36256         }
36257         // uiProvider = depreciated..
36258         
36259         if(typeof(attr.uiProvider) == 'string'){
36260            attr.uiProvider = this.uiProviders[attr.uiProvider] || 
36261                 /**  eval:var:attr */ eval(attr.uiProvider);
36262         }
36263         if(typeof(this.uiProviders['default']) != 'undefined') {
36264             attr.uiProvider = this.uiProviders['default'];
36265         }
36266         
36267         this.fireEvent('create', this, attr);
36268         
36269         attr.leaf  = typeof(attr.leaf) == 'string' ? attr.leaf * 1 : attr.leaf;
36270         return(attr.leaf ?
36271                         new Roo.tree.TreeNode(attr) :
36272                         new Roo.tree.AsyncTreeNode(attr));
36273     },
36274
36275     processResponse : function(response, node, callback)
36276     {
36277         var json = response.responseText;
36278         try {
36279             
36280             var o = Roo.decode(json);
36281             
36282             if (this.root === false && typeof(o.success) != undefined) {
36283                 this.root = 'data'; // the default behaviour for list like data..
36284                 }
36285                 
36286             if (this.root !== false &&  !o.success) {
36287                 // it's a failure condition.
36288                 var a = response.argument;
36289                 this.fireEvent("loadexception", this, a.node, response);
36290                 Roo.log("Load failed - should have a handler really");
36291                 return;
36292             }
36293             
36294             
36295             
36296             if (this.root !== false) {
36297                  o = o[this.root];
36298             }
36299             
36300             for(var i = 0, len = o.length; i < len; i++){
36301                 var n = this.createNode(o[i]);
36302                 if(n){
36303                     node.appendChild(n);
36304                 }
36305             }
36306             if(typeof callback == "function"){
36307                 callback(this, node);
36308             }
36309         }catch(e){
36310             this.handleFailure(response);
36311         }
36312     },
36313
36314     handleResponse : function(response){
36315         this.transId = false;
36316         var a = response.argument;
36317         this.processResponse(response, a.node, a.callback);
36318         this.fireEvent("load", this, a.node, response);
36319     },
36320
36321     handleFailure : function(response)
36322     {
36323         // should handle failure better..
36324         this.transId = false;
36325         var a = response.argument;
36326         this.fireEvent("loadexception", this, a.node, response);
36327         if(typeof a.callback == "function"){
36328             a.callback(this, a.node);
36329         }
36330     }
36331 });/*
36332  * Based on:
36333  * Ext JS Library 1.1.1
36334  * Copyright(c) 2006-2007, Ext JS, LLC.
36335  *
36336  * Originally Released Under LGPL - original licence link has changed is not relivant.
36337  *
36338  * Fork - LGPL
36339  * <script type="text/javascript">
36340  */
36341
36342 /**
36343 * @class Roo.tree.TreeFilter
36344 * Note this class is experimental and doesn't update the indent (lines) or expand collapse icons of the nodes
36345 * @param {TreePanel} tree
36346 * @param {Object} config (optional)
36347  */
36348 Roo.tree.TreeFilter = function(tree, config){
36349     this.tree = tree;
36350     this.filtered = {};
36351     Roo.apply(this, config);
36352 };
36353
36354 Roo.tree.TreeFilter.prototype = {
36355     clearBlank:false,
36356     reverse:false,
36357     autoClear:false,
36358     remove:false,
36359
36360      /**
36361      * Filter the data by a specific attribute.
36362      * @param {String/RegExp} value Either string that the attribute value
36363      * should start with or a RegExp to test against the attribute
36364      * @param {String} attr (optional) The attribute passed in your node's attributes collection. Defaults to "text".
36365      * @param {TreeNode} startNode (optional) The node to start the filter at.
36366      */
36367     filter : function(value, attr, startNode){
36368         attr = attr || "text";
36369         var f;
36370         if(typeof value == "string"){
36371             var vlen = value.length;
36372             // auto clear empty filter
36373             if(vlen == 0 && this.clearBlank){
36374                 this.clear();
36375                 return;
36376             }
36377             value = value.toLowerCase();
36378             f = function(n){
36379                 return n.attributes[attr].substr(0, vlen).toLowerCase() == value;
36380             };
36381         }else if(value.exec){ // regex?
36382             f = function(n){
36383                 return value.test(n.attributes[attr]);
36384             };
36385         }else{
36386             throw 'Illegal filter type, must be string or regex';
36387         }
36388         this.filterBy(f, null, startNode);
36389         },
36390
36391     /**
36392      * Filter by a function. The passed function will be called with each
36393      * node in the tree (or from the startNode). If the function returns true, the node is kept
36394      * otherwise it is filtered. If a node is filtered, its children are also filtered.
36395      * @param {Function} fn The filter function
36396      * @param {Object} scope (optional) The scope of the function (defaults to the current node)
36397      */
36398     filterBy : function(fn, scope, startNode){
36399         startNode = startNode || this.tree.root;
36400         if(this.autoClear){
36401             this.clear();
36402         }
36403         var af = this.filtered, rv = this.reverse;
36404         var f = function(n){
36405             if(n == startNode){
36406                 return true;
36407             }
36408             if(af[n.id]){
36409                 return false;
36410             }
36411             var m = fn.call(scope || n, n);
36412             if(!m || rv){
36413                 af[n.id] = n;
36414                 n.ui.hide();
36415                 return false;
36416             }
36417             return true;
36418         };
36419         startNode.cascade(f);
36420         if(this.remove){
36421            for(var id in af){
36422                if(typeof id != "function"){
36423                    var n = af[id];
36424                    if(n && n.parentNode){
36425                        n.parentNode.removeChild(n);
36426                    }
36427                }
36428            }
36429         }
36430     },
36431
36432     /**
36433      * Clears the current filter. Note: with the "remove" option
36434      * set a filter cannot be cleared.
36435      */
36436     clear : function(){
36437         var t = this.tree;
36438         var af = this.filtered;
36439         for(var id in af){
36440             if(typeof id != "function"){
36441                 var n = af[id];
36442                 if(n){
36443                     n.ui.show();
36444                 }
36445             }
36446         }
36447         this.filtered = {};
36448     }
36449 };
36450 /*
36451  * Based on:
36452  * Ext JS Library 1.1.1
36453  * Copyright(c) 2006-2007, Ext JS, LLC.
36454  *
36455  * Originally Released Under LGPL - original licence link has changed is not relivant.
36456  *
36457  * Fork - LGPL
36458  * <script type="text/javascript">
36459  */
36460  
36461
36462 /**
36463  * @class Roo.tree.TreeSorter
36464  * Provides sorting of nodes in a TreePanel
36465  * 
36466  * @cfg {Boolean} folderSort True to sort leaf nodes under non leaf nodes
36467  * @cfg {String} property The named attribute on the node to sort by (defaults to text)
36468  * @cfg {String} dir The direction to sort (asc or desc) (defaults to asc)
36469  * @cfg {String} leafAttr The attribute used to determine leaf nodes in folder sort (defaults to "leaf")
36470  * @cfg {Boolean} caseSensitive true for case sensitive sort (defaults to false)
36471  * @cfg {Function} sortType A custom "casting" function used to convert node values before sorting
36472  * @constructor
36473  * @param {TreePanel} tree
36474  * @param {Object} config
36475  */
36476 Roo.tree.TreeSorter = function(tree, config){
36477     Roo.apply(this, config);
36478     tree.on("beforechildrenrendered", this.doSort, this);
36479     tree.on("append", this.updateSort, this);
36480     tree.on("insert", this.updateSort, this);
36481     
36482     var dsc = this.dir && this.dir.toLowerCase() == "desc";
36483     var p = this.property || "text";
36484     var sortType = this.sortType;
36485     var fs = this.folderSort;
36486     var cs = this.caseSensitive === true;
36487     var leafAttr = this.leafAttr || 'leaf';
36488
36489     this.sortFn = function(n1, n2){
36490         if(fs){
36491             if(n1.attributes[leafAttr] && !n2.attributes[leafAttr]){
36492                 return 1;
36493             }
36494             if(!n1.attributes[leafAttr] && n2.attributes[leafAttr]){
36495                 return -1;
36496             }
36497         }
36498         var v1 = sortType ? sortType(n1) : (cs ? n1.attributes[p] : n1.attributes[p].toUpperCase());
36499         var v2 = sortType ? sortType(n2) : (cs ? n2.attributes[p] : n2.attributes[p].toUpperCase());
36500         if(v1 < v2){
36501                         return dsc ? +1 : -1;
36502                 }else if(v1 > v2){
36503                         return dsc ? -1 : +1;
36504         }else{
36505                 return 0;
36506         }
36507     };
36508 };
36509
36510 Roo.tree.TreeSorter.prototype = {
36511     doSort : function(node){
36512         node.sort(this.sortFn);
36513     },
36514     
36515     compareNodes : function(n1, n2){
36516         return (n1.text.toUpperCase() > n2.text.toUpperCase() ? 1 : -1);
36517     },
36518     
36519     updateSort : function(tree, node){
36520         if(node.childrenRendered){
36521             this.doSort.defer(1, this, [node]);
36522         }
36523     }
36524 };/*
36525  * Based on:
36526  * Ext JS Library 1.1.1
36527  * Copyright(c) 2006-2007, Ext JS, LLC.
36528  *
36529  * Originally Released Under LGPL - original licence link has changed is not relivant.
36530  *
36531  * Fork - LGPL
36532  * <script type="text/javascript">
36533  */
36534
36535 if(Roo.dd.DropZone){
36536     
36537 Roo.tree.TreeDropZone = function(tree, config){
36538     this.allowParentInsert = false;
36539     this.allowContainerDrop = false;
36540     this.appendOnly = false;
36541     Roo.tree.TreeDropZone.superclass.constructor.call(this, tree.innerCt, config);
36542     this.tree = tree;
36543     this.lastInsertClass = "x-tree-no-status";
36544     this.dragOverData = {};
36545 };
36546
36547 Roo.extend(Roo.tree.TreeDropZone, Roo.dd.DropZone, {
36548     ddGroup : "TreeDD",
36549     scroll:  true,
36550     
36551     expandDelay : 1000,
36552     
36553     expandNode : function(node){
36554         if(node.hasChildNodes() && !node.isExpanded()){
36555             node.expand(false, null, this.triggerCacheRefresh.createDelegate(this));
36556         }
36557     },
36558     
36559     queueExpand : function(node){
36560         this.expandProcId = this.expandNode.defer(this.expandDelay, this, [node]);
36561     },
36562     
36563     cancelExpand : function(){
36564         if(this.expandProcId){
36565             clearTimeout(this.expandProcId);
36566             this.expandProcId = false;
36567         }
36568     },
36569     
36570     isValidDropPoint : function(n, pt, dd, e, data){
36571         if(!n || !data){ return false; }
36572         var targetNode = n.node;
36573         var dropNode = data.node;
36574         // default drop rules
36575         if(!(targetNode && targetNode.isTarget && pt)){
36576             return false;
36577         }
36578         if(pt == "append" && targetNode.allowChildren === false){
36579             return false;
36580         }
36581         if((pt == "above" || pt == "below") && (targetNode.parentNode && targetNode.parentNode.allowChildren === false)){
36582             return false;
36583         }
36584         if(dropNode && (targetNode == dropNode || dropNode.contains(targetNode))){
36585             return false;
36586         }
36587         // reuse the object
36588         var overEvent = this.dragOverData;
36589         overEvent.tree = this.tree;
36590         overEvent.target = targetNode;
36591         overEvent.data = data;
36592         overEvent.point = pt;
36593         overEvent.source = dd;
36594         overEvent.rawEvent = e;
36595         overEvent.dropNode = dropNode;
36596         overEvent.cancel = false;  
36597         var result = this.tree.fireEvent("nodedragover", overEvent);
36598         return overEvent.cancel === false && result !== false;
36599     },
36600     
36601     getDropPoint : function(e, n, dd)
36602     {
36603         var tn = n.node;
36604         if(tn.isRoot){
36605             return tn.allowChildren !== false ? "append" : false; // always append for root
36606         }
36607         var dragEl = n.ddel;
36608         var t = Roo.lib.Dom.getY(dragEl), b = t + dragEl.offsetHeight;
36609         var y = Roo.lib.Event.getPageY(e);
36610         //var noAppend = tn.allowChildren === false || tn.isLeaf();
36611         
36612         // we may drop nodes anywhere, as long as allowChildren has not been set to false..
36613         var noAppend = tn.allowChildren === false;
36614         if(this.appendOnly || tn.parentNode.allowChildren === false){
36615             return noAppend ? false : "append";
36616         }
36617         var noBelow = false;
36618         if(!this.allowParentInsert){
36619             noBelow = tn.hasChildNodes() && tn.isExpanded();
36620         }
36621         var q = (b - t) / (noAppend ? 2 : 3);
36622         if(y >= t && y < (t + q)){
36623             return "above";
36624         }else if(!noBelow && (noAppend || y >= b-q && y <= b)){
36625             return "below";
36626         }else{
36627             return "append";
36628         }
36629     },
36630     
36631     onNodeEnter : function(n, dd, e, data)
36632     {
36633         this.cancelExpand();
36634     },
36635     
36636     onNodeOver : function(n, dd, e, data)
36637     {
36638        
36639         var pt = this.getDropPoint(e, n, dd);
36640         var node = n.node;
36641         
36642         // auto node expand check
36643         if(!this.expandProcId && pt == "append" && node.hasChildNodes() && !n.node.isExpanded()){
36644             this.queueExpand(node);
36645         }else if(pt != "append"){
36646             this.cancelExpand();
36647         }
36648         
36649         // set the insert point style on the target node
36650         var returnCls = this.dropNotAllowed;
36651         if(this.isValidDropPoint(n, pt, dd, e, data)){
36652            if(pt){
36653                var el = n.ddel;
36654                var cls;
36655                if(pt == "above"){
36656                    returnCls = n.node.isFirst() ? "x-tree-drop-ok-above" : "x-tree-drop-ok-between";
36657                    cls = "x-tree-drag-insert-above";
36658                }else if(pt == "below"){
36659                    returnCls = n.node.isLast() ? "x-tree-drop-ok-below" : "x-tree-drop-ok-between";
36660                    cls = "x-tree-drag-insert-below";
36661                }else{
36662                    returnCls = "x-tree-drop-ok-append";
36663                    cls = "x-tree-drag-append";
36664                }
36665                if(this.lastInsertClass != cls){
36666                    Roo.fly(el).replaceClass(this.lastInsertClass, cls);
36667                    this.lastInsertClass = cls;
36668                }
36669            }
36670        }
36671        return returnCls;
36672     },
36673     
36674     onNodeOut : function(n, dd, e, data){
36675         
36676         this.cancelExpand();
36677         this.removeDropIndicators(n);
36678     },
36679     
36680     onNodeDrop : function(n, dd, e, data){
36681         var point = this.getDropPoint(e, n, dd);
36682         var targetNode = n.node;
36683         targetNode.ui.startDrop();
36684         if(!this.isValidDropPoint(n, point, dd, e, data)){
36685             targetNode.ui.endDrop();
36686             return false;
36687         }
36688         // first try to find the drop node
36689         var dropNode = data.node || (dd.getTreeNode ? dd.getTreeNode(data, targetNode, point, e) : null);
36690         var dropEvent = {
36691             tree : this.tree,
36692             target: targetNode,
36693             data: data,
36694             point: point,
36695             source: dd,
36696             rawEvent: e,
36697             dropNode: dropNode,
36698             cancel: !dropNode   
36699         };
36700         var retval = this.tree.fireEvent("beforenodedrop", dropEvent);
36701         if(retval === false || dropEvent.cancel === true || !dropEvent.dropNode){
36702             targetNode.ui.endDrop();
36703             return false;
36704         }
36705         // allow target changing
36706         targetNode = dropEvent.target;
36707         if(point == "append" && !targetNode.isExpanded()){
36708             targetNode.expand(false, null, function(){
36709                 this.completeDrop(dropEvent);
36710             }.createDelegate(this));
36711         }else{
36712             this.completeDrop(dropEvent);
36713         }
36714         return true;
36715     },
36716     
36717     completeDrop : function(de){
36718         var ns = de.dropNode, p = de.point, t = de.target;
36719         if(!(ns instanceof Array)){
36720             ns = [ns];
36721         }
36722         var n;
36723         for(var i = 0, len = ns.length; i < len; i++){
36724             n = ns[i];
36725             if(p == "above"){
36726                 t.parentNode.insertBefore(n, t);
36727             }else if(p == "below"){
36728                 t.parentNode.insertBefore(n, t.nextSibling);
36729             }else{
36730                 t.appendChild(n);
36731             }
36732         }
36733         n.ui.focus();
36734         if(this.tree.hlDrop){
36735             n.ui.highlight();
36736         }
36737         t.ui.endDrop();
36738         this.tree.fireEvent("nodedrop", de);
36739     },
36740     
36741     afterNodeMoved : function(dd, data, e, targetNode, dropNode){
36742         if(this.tree.hlDrop){
36743             dropNode.ui.focus();
36744             dropNode.ui.highlight();
36745         }
36746         this.tree.fireEvent("nodedrop", this.tree, targetNode, data, dd, e);
36747     },
36748     
36749     getTree : function(){
36750         return this.tree;
36751     },
36752     
36753     removeDropIndicators : function(n){
36754         if(n && n.ddel){
36755             var el = n.ddel;
36756             Roo.fly(el).removeClass([
36757                     "x-tree-drag-insert-above",
36758                     "x-tree-drag-insert-below",
36759                     "x-tree-drag-append"]);
36760             this.lastInsertClass = "_noclass";
36761         }
36762     },
36763     
36764     beforeDragDrop : function(target, e, id){
36765         this.cancelExpand();
36766         return true;
36767     },
36768     
36769     afterRepair : function(data){
36770         if(data && Roo.enableFx){
36771             data.node.ui.highlight();
36772         }
36773         this.hideProxy();
36774     } 
36775     
36776 });
36777
36778 }
36779 /*
36780  * Based on:
36781  * Ext JS Library 1.1.1
36782  * Copyright(c) 2006-2007, Ext JS, LLC.
36783  *
36784  * Originally Released Under LGPL - original licence link has changed is not relivant.
36785  *
36786  * Fork - LGPL
36787  * <script type="text/javascript">
36788  */
36789  
36790
36791 if(Roo.dd.DragZone){
36792 Roo.tree.TreeDragZone = function(tree, config){
36793     Roo.tree.TreeDragZone.superclass.constructor.call(this, tree.getTreeEl(), config);
36794     this.tree = tree;
36795 };
36796
36797 Roo.extend(Roo.tree.TreeDragZone, Roo.dd.DragZone, {
36798     ddGroup : "TreeDD",
36799    
36800     onBeforeDrag : function(data, e){
36801         var n = data.node;
36802         return n && n.draggable && !n.disabled;
36803     },
36804      
36805     
36806     onInitDrag : function(e){
36807         var data = this.dragData;
36808         this.tree.getSelectionModel().select(data.node);
36809         this.proxy.update("");
36810         data.node.ui.appendDDGhost(this.proxy.ghost.dom);
36811         this.tree.fireEvent("startdrag", this.tree, data.node, e);
36812     },
36813     
36814     getRepairXY : function(e, data){
36815         return data.node.ui.getDDRepairXY();
36816     },
36817     
36818     onEndDrag : function(data, e){
36819         this.tree.fireEvent("enddrag", this.tree, data.node, e);
36820         
36821         
36822     },
36823     
36824     onValidDrop : function(dd, e, id){
36825         this.tree.fireEvent("dragdrop", this.tree, this.dragData.node, dd, e);
36826         this.hideProxy();
36827     },
36828     
36829     beforeInvalidDrop : function(e, id){
36830         // this scrolls the original position back into view
36831         var sm = this.tree.getSelectionModel();
36832         sm.clearSelections();
36833         sm.select(this.dragData.node);
36834     }
36835 });
36836 }/*
36837  * Based on:
36838  * Ext JS Library 1.1.1
36839  * Copyright(c) 2006-2007, Ext JS, LLC.
36840  *
36841  * Originally Released Under LGPL - original licence link has changed is not relivant.
36842  *
36843  * Fork - LGPL
36844  * <script type="text/javascript">
36845  */
36846 /**
36847  * @class Roo.tree.TreeEditor
36848  * @extends Roo.Editor
36849  * Provides editor functionality for inline tree node editing.  Any valid {@link Roo.form.Field} can be used
36850  * as the editor field.
36851  * @constructor
36852  * @param {Object} config (used to be the tree panel.)
36853  * @param {Object} oldconfig DEPRECIATED Either a prebuilt {@link Roo.form.Field} instance or a Field config object
36854  * 
36855  * @cfg {Roo.tree.TreePanel} tree The tree to bind to.
36856  * @cfg {Roo.form.TextField|Object} field The field configuration
36857  *
36858  * 
36859  */
36860 Roo.tree.TreeEditor = function(config, oldconfig) { // was -- (tree, config){
36861     var tree = config;
36862     var field;
36863     if (oldconfig) { // old style..
36864         field = oldconfig.events ? oldconfig : new Roo.form.TextField(oldconfig);
36865     } else {
36866         // new style..
36867         tree = config.tree;
36868         config.field = config.field  || {};
36869         config.field.xtype = 'TextField';
36870         field = Roo.factory(config.field, Roo.form);
36871     }
36872     config = config || {};
36873     
36874     
36875     this.addEvents({
36876         /**
36877          * @event beforenodeedit
36878          * Fires when editing is initiated, but before the value changes.  Editing can be canceled by returning
36879          * false from the handler of this event.
36880          * @param {Editor} this
36881          * @param {Roo.tree.Node} node 
36882          */
36883         "beforenodeedit" : true
36884     });
36885     
36886     //Roo.log(config);
36887     Roo.tree.TreeEditor.superclass.constructor.call(this, field, config);
36888
36889     this.tree = tree;
36890
36891     tree.on('beforeclick', this.beforeNodeClick, this);
36892     tree.getTreeEl().on('mousedown', this.hide, this);
36893     this.on('complete', this.updateNode, this);
36894     this.on('beforestartedit', this.fitToTree, this);
36895     this.on('startedit', this.bindScroll, this, {delay:10});
36896     this.on('specialkey', this.onSpecialKey, this);
36897 };
36898
36899 Roo.extend(Roo.tree.TreeEditor, Roo.Editor, {
36900     /**
36901      * @cfg {String} alignment
36902      * The position to align to (see {@link Roo.Element#alignTo} for more details, defaults to "l-l").
36903      */
36904     alignment: "l-l",
36905     // inherit
36906     autoSize: false,
36907     /**
36908      * @cfg {Boolean} hideEl
36909      * True to hide the bound element while the editor is displayed (defaults to false)
36910      */
36911     hideEl : false,
36912     /**
36913      * @cfg {String} cls
36914      * CSS class to apply to the editor (defaults to "x-small-editor x-tree-editor")
36915      */
36916     cls: "x-small-editor x-tree-editor",
36917     /**
36918      * @cfg {Boolean} shim
36919      * True to shim the editor if selects/iframes could be displayed beneath it (defaults to false)
36920      */
36921     shim:false,
36922     // inherit
36923     shadow:"frame",
36924     /**
36925      * @cfg {Number} maxWidth
36926      * The maximum width in pixels of the editor field (defaults to 250).  Note that if the maxWidth would exceed
36927      * the containing tree element's size, it will be automatically limited for you to the container width, taking
36928      * scroll and client offsets into account prior to each edit.
36929      */
36930     maxWidth: 250,
36931
36932     editDelay : 350,
36933
36934     // private
36935     fitToTree : function(ed, el){
36936         var td = this.tree.getTreeEl().dom, nd = el.dom;
36937         if(td.scrollLeft >  nd.offsetLeft){ // ensure the node left point is visible
36938             td.scrollLeft = nd.offsetLeft;
36939         }
36940         var w = Math.min(
36941                 this.maxWidth,
36942                 (td.clientWidth > 20 ? td.clientWidth : td.offsetWidth) - Math.max(0, nd.offsetLeft-td.scrollLeft) - /*cushion*/5);
36943         this.setSize(w, '');
36944         
36945         return this.fireEvent('beforenodeedit', this, this.editNode);
36946         
36947     },
36948
36949     // private
36950     triggerEdit : function(node){
36951         this.completeEdit();
36952         this.editNode = node;
36953         this.startEdit(node.ui.textNode, node.text);
36954     },
36955
36956     // private
36957     bindScroll : function(){
36958         this.tree.getTreeEl().on('scroll', this.cancelEdit, this);
36959     },
36960
36961     // private
36962     beforeNodeClick : function(node, e){
36963         var sinceLast = (this.lastClick ? this.lastClick.getElapsed() : 0);
36964         this.lastClick = new Date();
36965         if(sinceLast > this.editDelay && this.tree.getSelectionModel().isSelected(node)){
36966             e.stopEvent();
36967             this.triggerEdit(node);
36968             return false;
36969         }
36970         return true;
36971     },
36972
36973     // private
36974     updateNode : function(ed, value){
36975         this.tree.getTreeEl().un('scroll', this.cancelEdit, this);
36976         this.editNode.setText(value);
36977     },
36978
36979     // private
36980     onHide : function(){
36981         Roo.tree.TreeEditor.superclass.onHide.call(this);
36982         if(this.editNode){
36983             this.editNode.ui.focus();
36984         }
36985     },
36986
36987     // private
36988     onSpecialKey : function(field, e){
36989         var k = e.getKey();
36990         if(k == e.ESC){
36991             e.stopEvent();
36992             this.cancelEdit();
36993         }else if(k == e.ENTER && !e.hasModifier()){
36994             e.stopEvent();
36995             this.completeEdit();
36996         }
36997     }
36998 });//<Script type="text/javascript">
36999 /*
37000  * Based on:
37001  * Ext JS Library 1.1.1
37002  * Copyright(c) 2006-2007, Ext JS, LLC.
37003  *
37004  * Originally Released Under LGPL - original licence link has changed is not relivant.
37005  *
37006  * Fork - LGPL
37007  * <script type="text/javascript">
37008  */
37009  
37010 /**
37011  * Not documented??? - probably should be...
37012  */
37013
37014 Roo.tree.ColumnNodeUI = Roo.extend(Roo.tree.TreeNodeUI, {
37015     //focus: Roo.emptyFn, // prevent odd scrolling behavior
37016     
37017     renderElements : function(n, a, targetNode, bulkRender){
37018         //consel.log("renderElements?");
37019         this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
37020
37021         var t = n.getOwnerTree();
37022         var tid = Pman.Tab.Document_TypesTree.tree.el.id;
37023         
37024         var cols = t.columns;
37025         var bw = t.borderWidth;
37026         var c = cols[0];
37027         var href = a.href ? a.href : Roo.isGecko ? "" : "#";
37028          var cb = typeof a.checked == "boolean";
37029         var tx = String.format('{0}',n.text || (c.renderer ? c.renderer(a[c.dataIndex], n, a) : a[c.dataIndex]));
37030         var colcls = 'x-t-' + tid + '-c0';
37031         var buf = [
37032             '<li class="x-tree-node">',
37033             
37034                 
37035                 '<div class="x-tree-node-el ', a.cls,'">',
37036                     // extran...
37037                     '<div class="x-tree-col ', colcls, '" style="width:', c.width-bw, 'px;">',
37038                 
37039                 
37040                         '<span class="x-tree-node-indent">',this.indentMarkup,'</span>',
37041                         '<img src="', this.emptyIcon, '" class="x-tree-ec-icon  " />',
37042                         '<img src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',
37043                            (a.icon ? ' x-tree-node-inline-icon' : ''),
37044                            (a.iconCls ? ' '+a.iconCls : ''),
37045                            '" unselectable="on" />',
37046                         (cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + 
37047                              (a.checked ? 'checked="checked" />' : ' />')) : ''),
37048                              
37049                         '<a class="x-tree-node-anchor" hidefocus="on" href="',href,'" tabIndex="1" ',
37050                             (a.hrefTarget ? ' target="' +a.hrefTarget + '"' : ''), '>',
37051                             '<span unselectable="on" qtip="' + tx + '">',
37052                              tx,
37053                              '</span></a>' ,
37054                     '</div>',
37055                      '<a class="x-tree-node-anchor" hidefocus="on" href="',href,'" tabIndex="1" ',
37056                             (a.hrefTarget ? ' target="' +a.hrefTarget + '"' : ''), '>'
37057                  ];
37058         for(var i = 1, len = cols.length; i < len; i++){
37059             c = cols[i];
37060             colcls = 'x-t-' + tid + '-c' +i;
37061             tx = String.format('{0}', (c.renderer ? c.renderer(a[c.dataIndex], n, a) : a[c.dataIndex]));
37062             buf.push('<div class="x-tree-col ', colcls, ' ' ,(c.cls?c.cls:''),'" style="width:',c.width-bw,'px;">',
37063                         '<div class="x-tree-col-text" qtip="' + tx +'">',tx,"</div>",
37064                       "</div>");
37065          }
37066          
37067          buf.push(
37068             '</a>',
37069             '<div class="x-clear"></div></div>',
37070             '<ul class="x-tree-node-ct" style="display:none;"></ul>',
37071             "</li>");
37072         
37073         if(bulkRender !== true && n.nextSibling && n.nextSibling.ui.getEl()){
37074             this.wrap = Roo.DomHelper.insertHtml("beforeBegin",
37075                                 n.nextSibling.ui.getEl(), buf.join(""));
37076         }else{
37077             this.wrap = Roo.DomHelper.insertHtml("beforeEnd", targetNode, buf.join(""));
37078         }
37079         var el = this.wrap.firstChild;
37080         this.elRow = el;
37081         this.elNode = el.firstChild;
37082         this.ranchor = el.childNodes[1];
37083         this.ctNode = this.wrap.childNodes[1];
37084         var cs = el.firstChild.childNodes;
37085         this.indentNode = cs[0];
37086         this.ecNode = cs[1];
37087         this.iconNode = cs[2];
37088         var index = 3;
37089         if(cb){
37090             this.checkbox = cs[3];
37091             index++;
37092         }
37093         this.anchor = cs[index];
37094         
37095         this.textNode = cs[index].firstChild;
37096         
37097         //el.on("click", this.onClick, this);
37098         //el.on("dblclick", this.onDblClick, this);
37099         
37100         
37101        // console.log(this);
37102     },
37103     initEvents : function(){
37104         Roo.tree.ColumnNodeUI.superclass.initEvents.call(this);
37105         
37106             
37107         var a = this.ranchor;
37108
37109         var el = Roo.get(a);
37110
37111         if(Roo.isOpera){ // opera render bug ignores the CSS
37112             el.setStyle("text-decoration", "none");
37113         }
37114
37115         el.on("click", this.onClick, this);
37116         el.on("dblclick", this.onDblClick, this);
37117         el.on("contextmenu", this.onContextMenu, this);
37118         
37119     },
37120     
37121     /*onSelectedChange : function(state){
37122         if(state){
37123             this.focus();
37124             this.addClass("x-tree-selected");
37125         }else{
37126             //this.blur();
37127             this.removeClass("x-tree-selected");
37128         }
37129     },*/
37130     addClass : function(cls){
37131         if(this.elRow){
37132             Roo.fly(this.elRow).addClass(cls);
37133         }
37134         
37135     },
37136     
37137     
37138     removeClass : function(cls){
37139         if(this.elRow){
37140             Roo.fly(this.elRow).removeClass(cls);
37141         }
37142     }
37143
37144     
37145     
37146 });//<Script type="text/javascript">
37147
37148 /*
37149  * Based on:
37150  * Ext JS Library 1.1.1
37151  * Copyright(c) 2006-2007, Ext JS, LLC.
37152  *
37153  * Originally Released Under LGPL - original licence link has changed is not relivant.
37154  *
37155  * Fork - LGPL
37156  * <script type="text/javascript">
37157  */
37158  
37159
37160 /**
37161  * @class Roo.tree.ColumnTree
37162  * @extends Roo.data.TreePanel
37163  * @cfg {Object} columns  Including width, header, renderer, cls, dataIndex 
37164  * @cfg {int} borderWidth  compined right/left border allowance
37165  * @constructor
37166  * @param {String/HTMLElement/Element} el The container element
37167  * @param {Object} config
37168  */
37169 Roo.tree.ColumnTree =  function(el, config)
37170 {
37171    Roo.tree.ColumnTree.superclass.constructor.call(this, el , config);
37172    this.addEvents({
37173         /**
37174         * @event resize
37175         * Fire this event on a container when it resizes
37176         * @param {int} w Width
37177         * @param {int} h Height
37178         */
37179        "resize" : true
37180     });
37181     this.on('resize', this.onResize, this);
37182 };
37183
37184 Roo.extend(Roo.tree.ColumnTree, Roo.tree.TreePanel, {
37185     //lines:false,
37186     
37187     
37188     borderWidth: Roo.isBorderBox ? 0 : 2, 
37189     headEls : false,
37190     
37191     render : function(){
37192         // add the header.....
37193        
37194         Roo.tree.ColumnTree.superclass.render.apply(this);
37195         
37196         this.el.addClass('x-column-tree');
37197         
37198         this.headers = this.el.createChild(
37199             {cls:'x-tree-headers'},this.innerCt.dom);
37200    
37201         var cols = this.columns, c;
37202         var totalWidth = 0;
37203         this.headEls = [];
37204         var  len = cols.length;
37205         for(var i = 0; i < len; i++){
37206              c = cols[i];
37207              totalWidth += c.width;
37208             this.headEls.push(this.headers.createChild({
37209                  cls:'x-tree-hd ' + (c.cls?c.cls+'-hd':''),
37210                  cn: {
37211                      cls:'x-tree-hd-text',
37212                      html: c.header
37213                  },
37214                  style:'width:'+(c.width-this.borderWidth)+'px;'
37215              }));
37216         }
37217         this.headers.createChild({cls:'x-clear'});
37218         // prevent floats from wrapping when clipped
37219         this.headers.setWidth(totalWidth);
37220         //this.innerCt.setWidth(totalWidth);
37221         this.innerCt.setStyle({ overflow: 'auto' });
37222         this.onResize(this.width, this.height);
37223              
37224         
37225     },
37226     onResize : function(w,h)
37227     {
37228         this.height = h;
37229         this.width = w;
37230         // resize cols..
37231         this.innerCt.setWidth(this.width);
37232         this.innerCt.setHeight(this.height-20);
37233         
37234         // headers...
37235         var cols = this.columns, c;
37236         var totalWidth = 0;
37237         var expEl = false;
37238         var len = cols.length;
37239         for(var i = 0; i < len; i++){
37240             c = cols[i];
37241             if (this.autoExpandColumn !== false && c.dataIndex == this.autoExpandColumn) {
37242                 // it's the expander..
37243                 expEl  = this.headEls[i];
37244                 continue;
37245             }
37246             totalWidth += c.width;
37247             
37248         }
37249         if (expEl) {
37250             expEl.setWidth(  ((w - totalWidth)-this.borderWidth - 20));
37251         }
37252         this.headers.setWidth(w-20);
37253
37254         
37255         
37256         
37257     }
37258 });
37259 /*
37260  * Based on:
37261  * Ext JS Library 1.1.1
37262  * Copyright(c) 2006-2007, Ext JS, LLC.
37263  *
37264  * Originally Released Under LGPL - original licence link has changed is not relivant.
37265  *
37266  * Fork - LGPL
37267  * <script type="text/javascript">
37268  */
37269  
37270 /**
37271  * @class Roo.menu.Menu
37272  * @extends Roo.util.Observable
37273  * A menu object.  This is the container to which you add all other menu items.  Menu can also serve a as a base class
37274  * when you want a specialzed menu based off of another component (like {@link Roo.menu.DateMenu} for example).
37275  * @constructor
37276  * Creates a new Menu
37277  * @param {Object} config Configuration options
37278  */
37279 Roo.menu.Menu = function(config){
37280     
37281     Roo.menu.Menu.superclass.constructor.call(this, config);
37282     
37283     this.id = this.id || Roo.id();
37284     this.addEvents({
37285         /**
37286          * @event beforeshow
37287          * Fires before this menu is displayed
37288          * @param {Roo.menu.Menu} this
37289          */
37290         beforeshow : true,
37291         /**
37292          * @event beforehide
37293          * Fires before this menu is hidden
37294          * @param {Roo.menu.Menu} this
37295          */
37296         beforehide : true,
37297         /**
37298          * @event show
37299          * Fires after this menu is displayed
37300          * @param {Roo.menu.Menu} this
37301          */
37302         show : true,
37303         /**
37304          * @event hide
37305          * Fires after this menu is hidden
37306          * @param {Roo.menu.Menu} this
37307          */
37308         hide : true,
37309         /**
37310          * @event click
37311          * Fires when this menu is clicked (or when the enter key is pressed while it is active)
37312          * @param {Roo.menu.Menu} this
37313          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37314          * @param {Roo.EventObject} e
37315          */
37316         click : true,
37317         /**
37318          * @event mouseover
37319          * Fires when the mouse is hovering over this menu
37320          * @param {Roo.menu.Menu} this
37321          * @param {Roo.EventObject} e
37322          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37323          */
37324         mouseover : true,
37325         /**
37326          * @event mouseout
37327          * Fires when the mouse exits this menu
37328          * @param {Roo.menu.Menu} this
37329          * @param {Roo.EventObject} e
37330          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37331          */
37332         mouseout : true,
37333         /**
37334          * @event itemclick
37335          * Fires when a menu item contained in this menu is clicked
37336          * @param {Roo.menu.BaseItem} baseItem The BaseItem that was clicked
37337          * @param {Roo.EventObject} e
37338          */
37339         itemclick: true
37340     });
37341     if (this.registerMenu) {
37342         Roo.menu.MenuMgr.register(this);
37343     }
37344     
37345     var mis = this.items;
37346     this.items = new Roo.util.MixedCollection();
37347     if(mis){
37348         this.add.apply(this, mis);
37349     }
37350 };
37351
37352 Roo.extend(Roo.menu.Menu, Roo.util.Observable, {
37353     /**
37354      * @cfg {Number} minWidth The minimum width of the menu in pixels (defaults to 120)
37355      */
37356     minWidth : 120,
37357     /**
37358      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop"
37359      * for bottom-right shadow (defaults to "sides")
37360      */
37361     shadow : "sides",
37362     /**
37363      * @cfg {String} subMenuAlign The {@link Roo.Element#alignTo} anchor position value to use for submenus of
37364      * this menu (defaults to "tl-tr?")
37365      */
37366     subMenuAlign : "tl-tr?",
37367     /**
37368      * @cfg {String} defaultAlign The default {@link Roo.Element#alignTo) anchor position value for this menu
37369      * relative to its element of origin (defaults to "tl-bl?")
37370      */
37371     defaultAlign : "tl-bl?",
37372     /**
37373      * @cfg {Boolean} allowOtherMenus True to allow multiple menus to be displayed at the same time (defaults to false)
37374      */
37375     allowOtherMenus : false,
37376     /**
37377      * @cfg {Boolean} registerMenu True (default) - means that clicking on screen etc. hides it.
37378      */
37379     registerMenu : true,
37380
37381     hidden:true,
37382
37383     // private
37384     render : function(){
37385         if(this.el){
37386             return;
37387         }
37388         var el = this.el = new Roo.Layer({
37389             cls: "x-menu",
37390             shadow:this.shadow,
37391             constrain: false,
37392             parentEl: this.parentEl || document.body,
37393             zindex:15000
37394         });
37395
37396         this.keyNav = new Roo.menu.MenuNav(this);
37397
37398         if(this.plain){
37399             el.addClass("x-menu-plain");
37400         }
37401         if(this.cls){
37402             el.addClass(this.cls);
37403         }
37404         // generic focus element
37405         this.focusEl = el.createChild({
37406             tag: "a", cls: "x-menu-focus", href: "#", onclick: "return false;", tabIndex:"-1"
37407         });
37408         var ul = el.createChild({tag: "ul", cls: "x-menu-list"});
37409         //disabling touch- as it's causing issues ..
37410         //ul.on(Roo.isTouch ? 'touchstart' : 'click'   , this.onClick, this);
37411         ul.on('click'   , this.onClick, this);
37412         
37413         
37414         ul.on("mouseover", this.onMouseOver, this);
37415         ul.on("mouseout", this.onMouseOut, this);
37416         this.items.each(function(item){
37417             if (item.hidden) {
37418                 return;
37419             }
37420             
37421             var li = document.createElement("li");
37422             li.className = "x-menu-list-item";
37423             ul.dom.appendChild(li);
37424             item.render(li, this);
37425         }, this);
37426         this.ul = ul;
37427         this.autoWidth();
37428     },
37429
37430     // private
37431     autoWidth : function(){
37432         var el = this.el, ul = this.ul;
37433         if(!el){
37434             return;
37435         }
37436         var w = this.width;
37437         if(w){
37438             el.setWidth(w);
37439         }else if(Roo.isIE){
37440             el.setWidth(this.minWidth);
37441             var t = el.dom.offsetWidth; // force recalc
37442             el.setWidth(ul.getWidth()+el.getFrameWidth("lr"));
37443         }
37444     },
37445
37446     // private
37447     delayAutoWidth : function(){
37448         if(this.rendered){
37449             if(!this.awTask){
37450                 this.awTask = new Roo.util.DelayedTask(this.autoWidth, this);
37451             }
37452             this.awTask.delay(20);
37453         }
37454     },
37455
37456     // private
37457     findTargetItem : function(e){
37458         var t = e.getTarget(".x-menu-list-item", this.ul,  true);
37459         if(t && t.menuItemId){
37460             return this.items.get(t.menuItemId);
37461         }
37462     },
37463
37464     // private
37465     onClick : function(e){
37466         Roo.log("menu.onClick");
37467         var t = this.findTargetItem(e);
37468         if(!t){
37469             return;
37470         }
37471         Roo.log(e);
37472         if (Roo.isTouch && e.type == 'touchstart' && t.menu  && !t.disabled) {
37473             if(t == this.activeItem && t.shouldDeactivate(e)){
37474                 this.activeItem.deactivate();
37475                 delete this.activeItem;
37476                 return;
37477             }
37478             if(t.canActivate){
37479                 this.setActiveItem(t, true);
37480             }
37481             return;
37482             
37483             
37484         }
37485         
37486         t.onClick(e);
37487         this.fireEvent("click", this, t, e);
37488     },
37489
37490     // private
37491     setActiveItem : function(item, autoExpand){
37492         if(item != this.activeItem){
37493             if(this.activeItem){
37494                 this.activeItem.deactivate();
37495             }
37496             this.activeItem = item;
37497             item.activate(autoExpand);
37498         }else if(autoExpand){
37499             item.expandMenu();
37500         }
37501     },
37502
37503     // private
37504     tryActivate : function(start, step){
37505         var items = this.items;
37506         for(var i = start, len = items.length; i >= 0 && i < len; i+= step){
37507             var item = items.get(i);
37508             if(!item.disabled && item.canActivate){
37509                 this.setActiveItem(item, false);
37510                 return item;
37511             }
37512         }
37513         return false;
37514     },
37515
37516     // private
37517     onMouseOver : function(e){
37518         var t;
37519         if(t = this.findTargetItem(e)){
37520             if(t.canActivate && !t.disabled){
37521                 this.setActiveItem(t, true);
37522             }
37523         }
37524         this.fireEvent("mouseover", this, e, t);
37525     },
37526
37527     // private
37528     onMouseOut : function(e){
37529         var t;
37530         if(t = this.findTargetItem(e)){
37531             if(t == this.activeItem && t.shouldDeactivate(e)){
37532                 this.activeItem.deactivate();
37533                 delete this.activeItem;
37534             }
37535         }
37536         this.fireEvent("mouseout", this, e, t);
37537     },
37538
37539     /**
37540      * Read-only.  Returns true if the menu is currently displayed, else false.
37541      * @type Boolean
37542      */
37543     isVisible : function(){
37544         return this.el && !this.hidden;
37545     },
37546
37547     /**
37548      * Displays this menu relative to another element
37549      * @param {String/HTMLElement/Roo.Element} element The element to align to
37550      * @param {String} position (optional) The {@link Roo.Element#alignTo} anchor position to use in aligning to
37551      * the element (defaults to this.defaultAlign)
37552      * @param {Roo.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
37553      */
37554     show : function(el, pos, parentMenu){
37555         this.parentMenu = parentMenu;
37556         if(!this.el){
37557             this.render();
37558         }
37559         this.fireEvent("beforeshow", this);
37560         this.showAt(this.el.getAlignToXY(el, pos || this.defaultAlign), parentMenu, false);
37561     },
37562
37563     /**
37564      * Displays this menu at a specific xy position
37565      * @param {Array} xyPosition Contains X & Y [x, y] values for the position at which to show the menu (coordinates are page-based)
37566      * @param {Roo.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
37567      */
37568     showAt : function(xy, parentMenu, /* private: */_e){
37569         this.parentMenu = parentMenu;
37570         if(!this.el){
37571             this.render();
37572         }
37573         if(_e !== false){
37574             this.fireEvent("beforeshow", this);
37575             xy = this.el.adjustForConstraints(xy);
37576         }
37577         this.el.setXY(xy);
37578         this.el.show();
37579         this.hidden = false;
37580         this.focus();
37581         this.fireEvent("show", this);
37582     },
37583
37584     focus : function(){
37585         if(!this.hidden){
37586             this.doFocus.defer(50, this);
37587         }
37588     },
37589
37590     doFocus : function(){
37591         if(!this.hidden){
37592             this.focusEl.focus();
37593         }
37594     },
37595
37596     /**
37597      * Hides this menu and optionally all parent menus
37598      * @param {Boolean} deep (optional) True to hide all parent menus recursively, if any (defaults to false)
37599      */
37600     hide : function(deep){
37601         if(this.el && this.isVisible()){
37602             this.fireEvent("beforehide", this);
37603             if(this.activeItem){
37604                 this.activeItem.deactivate();
37605                 this.activeItem = null;
37606             }
37607             this.el.hide();
37608             this.hidden = true;
37609             this.fireEvent("hide", this);
37610         }
37611         if(deep === true && this.parentMenu){
37612             this.parentMenu.hide(true);
37613         }
37614     },
37615
37616     /**
37617      * Addds one or more items of any type supported by the Menu class, or that can be converted into menu items.
37618      * Any of the following are valid:
37619      * <ul>
37620      * <li>Any menu item object based on {@link Roo.menu.Item}</li>
37621      * <li>An HTMLElement object which will be converted to a menu item</li>
37622      * <li>A menu item config object that will be created as a new menu item</li>
37623      * <li>A string, which can either be '-' or 'separator' to add a menu separator, otherwise
37624      * it will be converted into a {@link Roo.menu.TextItem} and added</li>
37625      * </ul>
37626      * Usage:
37627      * <pre><code>
37628 // Create the menu
37629 var menu = new Roo.menu.Menu();
37630
37631 // Create a menu item to add by reference
37632 var menuItem = new Roo.menu.Item({ text: 'New Item!' });
37633
37634 // Add a bunch of items at once using different methods.
37635 // Only the last item added will be returned.
37636 var item = menu.add(
37637     menuItem,                // add existing item by ref
37638     'Dynamic Item',          // new TextItem
37639     '-',                     // new separator
37640     { text: 'Config Item' }  // new item by config
37641 );
37642 </code></pre>
37643      * @param {Mixed} args One or more menu items, menu item configs or other objects that can be converted to menu items
37644      * @return {Roo.menu.Item} The menu item that was added, or the last one if multiple items were added
37645      */
37646     add : function(){
37647         var a = arguments, l = a.length, item;
37648         for(var i = 0; i < l; i++){
37649             var el = a[i];
37650             if ((typeof(el) == "object") && el.xtype && el.xns) {
37651                 el = Roo.factory(el, Roo.menu);
37652             }
37653             
37654             if(el.render){ // some kind of Item
37655                 item = this.addItem(el);
37656             }else if(typeof el == "string"){ // string
37657                 if(el == "separator" || el == "-"){
37658                     item = this.addSeparator();
37659                 }else{
37660                     item = this.addText(el);
37661                 }
37662             }else if(el.tagName || el.el){ // element
37663                 item = this.addElement(el);
37664             }else if(typeof el == "object"){ // must be menu item config?
37665                 item = this.addMenuItem(el);
37666             }
37667         }
37668         return item;
37669     },
37670
37671     /**
37672      * Returns this menu's underlying {@link Roo.Element} object
37673      * @return {Roo.Element} The element
37674      */
37675     getEl : function(){
37676         if(!this.el){
37677             this.render();
37678         }
37679         return this.el;
37680     },
37681
37682     /**
37683      * Adds a separator bar to the menu
37684      * @return {Roo.menu.Item} The menu item that was added
37685      */
37686     addSeparator : function(){
37687         return this.addItem(new Roo.menu.Separator());
37688     },
37689
37690     /**
37691      * Adds an {@link Roo.Element} object to the menu
37692      * @param {String/HTMLElement/Roo.Element} el The element or DOM node to add, or its id
37693      * @return {Roo.menu.Item} The menu item that was added
37694      */
37695     addElement : function(el){
37696         return this.addItem(new Roo.menu.BaseItem(el));
37697     },
37698
37699     /**
37700      * Adds an existing object based on {@link Roo.menu.Item} to the menu
37701      * @param {Roo.menu.Item} item The menu item to add
37702      * @return {Roo.menu.Item} The menu item that was added
37703      */
37704     addItem : function(item){
37705         this.items.add(item);
37706         if(this.ul){
37707             var li = document.createElement("li");
37708             li.className = "x-menu-list-item";
37709             this.ul.dom.appendChild(li);
37710             item.render(li, this);
37711             this.delayAutoWidth();
37712         }
37713         return item;
37714     },
37715
37716     /**
37717      * Creates a new {@link Roo.menu.Item} based an the supplied config object and adds it to the menu
37718      * @param {Object} config A MenuItem config object
37719      * @return {Roo.menu.Item} The menu item that was added
37720      */
37721     addMenuItem : function(config){
37722         if(!(config instanceof Roo.menu.Item)){
37723             if(typeof config.checked == "boolean"){ // must be check menu item config?
37724                 config = new Roo.menu.CheckItem(config);
37725             }else{
37726                 config = new Roo.menu.Item(config);
37727             }
37728         }
37729         return this.addItem(config);
37730     },
37731
37732     /**
37733      * Creates a new {@link Roo.menu.TextItem} with the supplied text and adds it to the menu
37734      * @param {String} text The text to display in the menu item
37735      * @return {Roo.menu.Item} The menu item that was added
37736      */
37737     addText : function(text){
37738         return this.addItem(new Roo.menu.TextItem({ text : text }));
37739     },
37740
37741     /**
37742      * Inserts an existing object based on {@link Roo.menu.Item} to the menu at a specified index
37743      * @param {Number} index The index in the menu's list of current items where the new item should be inserted
37744      * @param {Roo.menu.Item} item The menu item to add
37745      * @return {Roo.menu.Item} The menu item that was added
37746      */
37747     insert : function(index, item){
37748         this.items.insert(index, item);
37749         if(this.ul){
37750             var li = document.createElement("li");
37751             li.className = "x-menu-list-item";
37752             this.ul.dom.insertBefore(li, this.ul.dom.childNodes[index]);
37753             item.render(li, this);
37754             this.delayAutoWidth();
37755         }
37756         return item;
37757     },
37758
37759     /**
37760      * Removes an {@link Roo.menu.Item} from the menu and destroys the object
37761      * @param {Roo.menu.Item} item The menu item to remove
37762      */
37763     remove : function(item){
37764         this.items.removeKey(item.id);
37765         item.destroy();
37766     },
37767
37768     /**
37769      * Removes and destroys all items in the menu
37770      */
37771     removeAll : function(){
37772         var f;
37773         while(f = this.items.first()){
37774             this.remove(f);
37775         }
37776     }
37777 });
37778
37779 // MenuNav is a private utility class used internally by the Menu
37780 Roo.menu.MenuNav = function(menu){
37781     Roo.menu.MenuNav.superclass.constructor.call(this, menu.el);
37782     this.scope = this.menu = menu;
37783 };
37784
37785 Roo.extend(Roo.menu.MenuNav, Roo.KeyNav, {
37786     doRelay : function(e, h){
37787         var k = e.getKey();
37788         if(!this.menu.activeItem && e.isNavKeyPress() && k != e.SPACE && k != e.RETURN){
37789             this.menu.tryActivate(0, 1);
37790             return false;
37791         }
37792         return h.call(this.scope || this, e, this.menu);
37793     },
37794
37795     up : function(e, m){
37796         if(!m.tryActivate(m.items.indexOf(m.activeItem)-1, -1)){
37797             m.tryActivate(m.items.length-1, -1);
37798         }
37799     },
37800
37801     down : function(e, m){
37802         if(!m.tryActivate(m.items.indexOf(m.activeItem)+1, 1)){
37803             m.tryActivate(0, 1);
37804         }
37805     },
37806
37807     right : function(e, m){
37808         if(m.activeItem){
37809             m.activeItem.expandMenu(true);
37810         }
37811     },
37812
37813     left : function(e, m){
37814         m.hide();
37815         if(m.parentMenu && m.parentMenu.activeItem){
37816             m.parentMenu.activeItem.activate();
37817         }
37818     },
37819
37820     enter : function(e, m){
37821         if(m.activeItem){
37822             e.stopPropagation();
37823             m.activeItem.onClick(e);
37824             m.fireEvent("click", this, m.activeItem);
37825             return true;
37826         }
37827     }
37828 });/*
37829  * Based on:
37830  * Ext JS Library 1.1.1
37831  * Copyright(c) 2006-2007, Ext JS, LLC.
37832  *
37833  * Originally Released Under LGPL - original licence link has changed is not relivant.
37834  *
37835  * Fork - LGPL
37836  * <script type="text/javascript">
37837  */
37838  
37839 /**
37840  * @class Roo.menu.MenuMgr
37841  * Provides a common registry of all menu items on a page so that they can be easily accessed by id.
37842  * @singleton
37843  */
37844 Roo.menu.MenuMgr = function(){
37845    var menus, active, groups = {}, attached = false, lastShow = new Date();
37846
37847    // private - called when first menu is created
37848    function init(){
37849        menus = {};
37850        active = new Roo.util.MixedCollection();
37851        Roo.get(document).addKeyListener(27, function(){
37852            if(active.length > 0){
37853                hideAll();
37854            }
37855        });
37856    }
37857
37858    // private
37859    function hideAll(){
37860        if(active && active.length > 0){
37861            var c = active.clone();
37862            c.each(function(m){
37863                m.hide();
37864            });
37865        }
37866    }
37867
37868    // private
37869    function onHide(m){
37870        active.remove(m);
37871        if(active.length < 1){
37872            Roo.get(document).un("mousedown", onMouseDown);
37873            attached = false;
37874        }
37875    }
37876
37877    // private
37878    function onShow(m){
37879        var last = active.last();
37880        lastShow = new Date();
37881        active.add(m);
37882        if(!attached){
37883            Roo.get(document).on("mousedown", onMouseDown);
37884            attached = true;
37885        }
37886        if(m.parentMenu){
37887           m.getEl().setZIndex(parseInt(m.parentMenu.getEl().getStyle("z-index"), 10) + 3);
37888           m.parentMenu.activeChild = m;
37889        }else if(last && last.isVisible()){
37890           m.getEl().setZIndex(parseInt(last.getEl().getStyle("z-index"), 10) + 3);
37891        }
37892    }
37893
37894    // private
37895    function onBeforeHide(m){
37896        if(m.activeChild){
37897            m.activeChild.hide();
37898        }
37899        if(m.autoHideTimer){
37900            clearTimeout(m.autoHideTimer);
37901            delete m.autoHideTimer;
37902        }
37903    }
37904
37905    // private
37906    function onBeforeShow(m){
37907        var pm = m.parentMenu;
37908        if(!pm && !m.allowOtherMenus){
37909            hideAll();
37910        }else if(pm && pm.activeChild && active != m){
37911            pm.activeChild.hide();
37912        }
37913    }
37914
37915    // private
37916    function onMouseDown(e){
37917        if(lastShow.getElapsed() > 50 && active.length > 0 && !e.getTarget(".x-menu")){
37918            hideAll();
37919        }
37920    }
37921
37922    // private
37923    function onBeforeCheck(mi, state){
37924        if(state){
37925            var g = groups[mi.group];
37926            for(var i = 0, l = g.length; i < l; i++){
37927                if(g[i] != mi){
37928                    g[i].setChecked(false);
37929                }
37930            }
37931        }
37932    }
37933
37934    return {
37935
37936        /**
37937         * Hides all menus that are currently visible
37938         */
37939        hideAll : function(){
37940             hideAll();  
37941        },
37942
37943        // private
37944        register : function(menu){
37945            if(!menus){
37946                init();
37947            }
37948            menus[menu.id] = menu;
37949            menu.on("beforehide", onBeforeHide);
37950            menu.on("hide", onHide);
37951            menu.on("beforeshow", onBeforeShow);
37952            menu.on("show", onShow);
37953            var g = menu.group;
37954            if(g && menu.events["checkchange"]){
37955                if(!groups[g]){
37956                    groups[g] = [];
37957                }
37958                groups[g].push(menu);
37959                menu.on("checkchange", onCheck);
37960            }
37961        },
37962
37963         /**
37964          * Returns a {@link Roo.menu.Menu} object
37965          * @param {String/Object} menu The string menu id, an existing menu object reference, or a Menu config that will
37966          * be used to generate and return a new Menu instance.
37967          */
37968        get : function(menu){
37969            if(typeof menu == "string"){ // menu id
37970                return menus[menu];
37971            }else if(menu.events){  // menu instance
37972                return menu;
37973            }else if(typeof menu.length == 'number'){ // array of menu items?
37974                return new Roo.menu.Menu({items:menu});
37975            }else{ // otherwise, must be a config
37976                return new Roo.menu.Menu(menu);
37977            }
37978        },
37979
37980        // private
37981        unregister : function(menu){
37982            delete menus[menu.id];
37983            menu.un("beforehide", onBeforeHide);
37984            menu.un("hide", onHide);
37985            menu.un("beforeshow", onBeforeShow);
37986            menu.un("show", onShow);
37987            var g = menu.group;
37988            if(g && menu.events["checkchange"]){
37989                groups[g].remove(menu);
37990                menu.un("checkchange", onCheck);
37991            }
37992        },
37993
37994        // private
37995        registerCheckable : function(menuItem){
37996            var g = menuItem.group;
37997            if(g){
37998                if(!groups[g]){
37999                    groups[g] = [];
38000                }
38001                groups[g].push(menuItem);
38002                menuItem.on("beforecheckchange", onBeforeCheck);
38003            }
38004        },
38005
38006        // private
38007        unregisterCheckable : function(menuItem){
38008            var g = menuItem.group;
38009            if(g){
38010                groups[g].remove(menuItem);
38011                menuItem.un("beforecheckchange", onBeforeCheck);
38012            }
38013        }
38014    };
38015 }();/*
38016  * Based on:
38017  * Ext JS Library 1.1.1
38018  * Copyright(c) 2006-2007, Ext JS, LLC.
38019  *
38020  * Originally Released Under LGPL - original licence link has changed is not relivant.
38021  *
38022  * Fork - LGPL
38023  * <script type="text/javascript">
38024  */
38025  
38026
38027 /**
38028  * @class Roo.menu.BaseItem
38029  * @extends Roo.Component
38030  * The base class for all items that render into menus.  BaseItem provides default rendering, activated state
38031  * management and base configuration options shared by all menu components.
38032  * @constructor
38033  * Creates a new BaseItem
38034  * @param {Object} config Configuration options
38035  */
38036 Roo.menu.BaseItem = function(config){
38037     Roo.menu.BaseItem.superclass.constructor.call(this, config);
38038
38039     this.addEvents({
38040         /**
38041          * @event click
38042          * Fires when this item is clicked
38043          * @param {Roo.menu.BaseItem} this
38044          * @param {Roo.EventObject} e
38045          */
38046         click: true,
38047         /**
38048          * @event activate
38049          * Fires when this item is activated
38050          * @param {Roo.menu.BaseItem} this
38051          */
38052         activate : true,
38053         /**
38054          * @event deactivate
38055          * Fires when this item is deactivated
38056          * @param {Roo.menu.BaseItem} this
38057          */
38058         deactivate : true
38059     });
38060
38061     if(this.handler){
38062         this.on("click", this.handler, this.scope, true);
38063     }
38064 };
38065
38066 Roo.extend(Roo.menu.BaseItem, Roo.Component, {
38067     /**
38068      * @cfg {Function} handler
38069      * A function that will handle the click event of this menu item (defaults to undefined)
38070      */
38071     /**
38072      * @cfg {Boolean} canActivate True if this item can be visually activated (defaults to false)
38073      */
38074     canActivate : false,
38075     
38076      /**
38077      * @cfg {Boolean} hidden True to prevent creation of this menu item (defaults to false)
38078      */
38079     hidden: false,
38080     
38081     /**
38082      * @cfg {String} activeClass The CSS class to use when the item becomes activated (defaults to "x-menu-item-active")
38083      */
38084     activeClass : "x-menu-item-active",
38085     /**
38086      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to true)
38087      */
38088     hideOnClick : true,
38089     /**
38090      * @cfg {Number} hideDelay Length of time in milliseconds to wait before hiding after a click (defaults to 100)
38091      */
38092     hideDelay : 100,
38093
38094     // private
38095     ctype: "Roo.menu.BaseItem",
38096
38097     // private
38098     actionMode : "container",
38099
38100     // private
38101     render : function(container, parentMenu){
38102         this.parentMenu = parentMenu;
38103         Roo.menu.BaseItem.superclass.render.call(this, container);
38104         this.container.menuItemId = this.id;
38105     },
38106
38107     // private
38108     onRender : function(container, position){
38109         this.el = Roo.get(this.el);
38110         container.dom.appendChild(this.el.dom);
38111     },
38112
38113     // private
38114     onClick : function(e){
38115         if(!this.disabled && this.fireEvent("click", this, e) !== false
38116                 && this.parentMenu.fireEvent("itemclick", this, e) !== false){
38117             this.handleClick(e);
38118         }else{
38119             e.stopEvent();
38120         }
38121     },
38122
38123     // private
38124     activate : function(){
38125         if(this.disabled){
38126             return false;
38127         }
38128         var li = this.container;
38129         li.addClass(this.activeClass);
38130         this.region = li.getRegion().adjust(2, 2, -2, -2);
38131         this.fireEvent("activate", this);
38132         return true;
38133     },
38134
38135     // private
38136     deactivate : function(){
38137         this.container.removeClass(this.activeClass);
38138         this.fireEvent("deactivate", this);
38139     },
38140
38141     // private
38142     shouldDeactivate : function(e){
38143         return !this.region || !this.region.contains(e.getPoint());
38144     },
38145
38146     // private
38147     handleClick : function(e){
38148         if(this.hideOnClick){
38149             this.parentMenu.hide.defer(this.hideDelay, this.parentMenu, [true]);
38150         }
38151     },
38152
38153     // private
38154     expandMenu : function(autoActivate){
38155         // do nothing
38156     },
38157
38158     // private
38159     hideMenu : function(){
38160         // do nothing
38161     }
38162 });/*
38163  * Based on:
38164  * Ext JS Library 1.1.1
38165  * Copyright(c) 2006-2007, Ext JS, LLC.
38166  *
38167  * Originally Released Under LGPL - original licence link has changed is not relivant.
38168  *
38169  * Fork - LGPL
38170  * <script type="text/javascript">
38171  */
38172  
38173 /**
38174  * @class Roo.menu.Adapter
38175  * @extends Roo.menu.BaseItem
38176  * 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.
38177  * It provides basic rendering, activation management and enable/disable logic required to work in menus.
38178  * @constructor
38179  * Creates a new Adapter
38180  * @param {Object} config Configuration options
38181  */
38182 Roo.menu.Adapter = function(component, config){
38183     Roo.menu.Adapter.superclass.constructor.call(this, config);
38184     this.component = component;
38185 };
38186 Roo.extend(Roo.menu.Adapter, Roo.menu.BaseItem, {
38187     // private
38188     canActivate : true,
38189
38190     // private
38191     onRender : function(container, position){
38192         this.component.render(container);
38193         this.el = this.component.getEl();
38194     },
38195
38196     // private
38197     activate : function(){
38198         if(this.disabled){
38199             return false;
38200         }
38201         this.component.focus();
38202         this.fireEvent("activate", this);
38203         return true;
38204     },
38205
38206     // private
38207     deactivate : function(){
38208         this.fireEvent("deactivate", this);
38209     },
38210
38211     // private
38212     disable : function(){
38213         this.component.disable();
38214         Roo.menu.Adapter.superclass.disable.call(this);
38215     },
38216
38217     // private
38218     enable : function(){
38219         this.component.enable();
38220         Roo.menu.Adapter.superclass.enable.call(this);
38221     }
38222 });/*
38223  * Based on:
38224  * Ext JS Library 1.1.1
38225  * Copyright(c) 2006-2007, Ext JS, LLC.
38226  *
38227  * Originally Released Under LGPL - original licence link has changed is not relivant.
38228  *
38229  * Fork - LGPL
38230  * <script type="text/javascript">
38231  */
38232
38233 /**
38234  * @class Roo.menu.TextItem
38235  * @extends Roo.menu.BaseItem
38236  * Adds a static text string to a menu, usually used as either a heading or group separator.
38237  * Note: old style constructor with text is still supported.
38238  * 
38239  * @constructor
38240  * Creates a new TextItem
38241  * @param {Object} cfg Configuration
38242  */
38243 Roo.menu.TextItem = function(cfg){
38244     if (typeof(cfg) == 'string') {
38245         this.text = cfg;
38246     } else {
38247         Roo.apply(this,cfg);
38248     }
38249     
38250     Roo.menu.TextItem.superclass.constructor.call(this);
38251 };
38252
38253 Roo.extend(Roo.menu.TextItem, Roo.menu.BaseItem, {
38254     /**
38255      * @cfg {Boolean} text Text to show on item.
38256      */
38257     text : '',
38258     
38259     /**
38260      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to false)
38261      */
38262     hideOnClick : false,
38263     /**
38264      * @cfg {String} itemCls The default CSS class to use for text items (defaults to "x-menu-text")
38265      */
38266     itemCls : "x-menu-text",
38267
38268     // private
38269     onRender : function(){
38270         var s = document.createElement("span");
38271         s.className = this.itemCls;
38272         s.innerHTML = this.text;
38273         this.el = s;
38274         Roo.menu.TextItem.superclass.onRender.apply(this, arguments);
38275     }
38276 });/*
38277  * Based on:
38278  * Ext JS Library 1.1.1
38279  * Copyright(c) 2006-2007, Ext JS, LLC.
38280  *
38281  * Originally Released Under LGPL - original licence link has changed is not relivant.
38282  *
38283  * Fork - LGPL
38284  * <script type="text/javascript">
38285  */
38286
38287 /**
38288  * @class Roo.menu.Separator
38289  * @extends Roo.menu.BaseItem
38290  * Adds a separator bar to a menu, used to divide logical groups of menu items. Generally you will
38291  * add one of these by using "-" in you call to add() or in your items config rather than creating one directly.
38292  * @constructor
38293  * @param {Object} config Configuration options
38294  */
38295 Roo.menu.Separator = function(config){
38296     Roo.menu.Separator.superclass.constructor.call(this, config);
38297 };
38298
38299 Roo.extend(Roo.menu.Separator, Roo.menu.BaseItem, {
38300     /**
38301      * @cfg {String} itemCls The default CSS class to use for separators (defaults to "x-menu-sep")
38302      */
38303     itemCls : "x-menu-sep",
38304     /**
38305      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to false)
38306      */
38307     hideOnClick : false,
38308
38309     // private
38310     onRender : function(li){
38311         var s = document.createElement("span");
38312         s.className = this.itemCls;
38313         s.innerHTML = "&#160;";
38314         this.el = s;
38315         li.addClass("x-menu-sep-li");
38316         Roo.menu.Separator.superclass.onRender.apply(this, arguments);
38317     }
38318 });/*
38319  * Based on:
38320  * Ext JS Library 1.1.1
38321  * Copyright(c) 2006-2007, Ext JS, LLC.
38322  *
38323  * Originally Released Under LGPL - original licence link has changed is not relivant.
38324  *
38325  * Fork - LGPL
38326  * <script type="text/javascript">
38327  */
38328 /**
38329  * @class Roo.menu.Item
38330  * @extends Roo.menu.BaseItem
38331  * A base class for all menu items that require menu-related functionality (like sub-menus) and are not static
38332  * display items.  Item extends the base functionality of {@link Roo.menu.BaseItem} by adding menu-specific
38333  * activation and click handling.
38334  * @constructor
38335  * Creates a new Item
38336  * @param {Object} config Configuration options
38337  */
38338 Roo.menu.Item = function(config){
38339     Roo.menu.Item.superclass.constructor.call(this, config);
38340     if(this.menu){
38341         this.menu = Roo.menu.MenuMgr.get(this.menu);
38342     }
38343 };
38344 Roo.extend(Roo.menu.Item, Roo.menu.BaseItem, {
38345     
38346     /**
38347      * @cfg {String} text
38348      * The text to show on the menu item.
38349      */
38350     text: '',
38351      /**
38352      * @cfg {String} HTML to render in menu
38353      * The text to show on the menu item (HTML version).
38354      */
38355     html: '',
38356     /**
38357      * @cfg {String} icon
38358      * The path to an icon to display in this menu item (defaults to Roo.BLANK_IMAGE_URL)
38359      */
38360     icon: undefined,
38361     /**
38362      * @cfg {String} itemCls The default CSS class to use for menu items (defaults to "x-menu-item")
38363      */
38364     itemCls : "x-menu-item",
38365     /**
38366      * @cfg {Boolean} canActivate True if this item can be visually activated (defaults to true)
38367      */
38368     canActivate : true,
38369     /**
38370      * @cfg {Number} showDelay Length of time in milliseconds to wait before showing this item (defaults to 200)
38371      */
38372     showDelay: 200,
38373     // doc'd in BaseItem
38374     hideDelay: 200,
38375
38376     // private
38377     ctype: "Roo.menu.Item",
38378     
38379     // private
38380     onRender : function(container, position){
38381         var el = document.createElement("a");
38382         el.hideFocus = true;
38383         el.unselectable = "on";
38384         el.href = this.href || "#";
38385         if(this.hrefTarget){
38386             el.target = this.hrefTarget;
38387         }
38388         el.className = this.itemCls + (this.menu ?  " x-menu-item-arrow" : "") + (this.cls ?  " " + this.cls : "");
38389         
38390         var html = this.html.length ? this.html  : String.format('{0}',this.text);
38391         
38392         el.innerHTML = String.format(
38393                 '<img src="{0}" class="x-menu-item-icon {1}" />' + html,
38394                 this.icon || Roo.BLANK_IMAGE_URL, this.iconCls || '');
38395         this.el = el;
38396         Roo.menu.Item.superclass.onRender.call(this, container, position);
38397     },
38398
38399     /**
38400      * Sets the text to display in this menu item
38401      * @param {String} text The text to display
38402      * @param {Boolean} isHTML true to indicate text is pure html.
38403      */
38404     setText : function(text, isHTML){
38405         if (isHTML) {
38406             this.html = text;
38407         } else {
38408             this.text = text;
38409             this.html = '';
38410         }
38411         if(this.rendered){
38412             var html = this.html.length ? this.html  : String.format('{0}',this.text);
38413      
38414             this.el.update(String.format(
38415                 '<img src="{0}" class="x-menu-item-icon {2}">' + html,
38416                 this.icon || Roo.BLANK_IMAGE_URL, this.text, this.iconCls || ''));
38417             this.parentMenu.autoWidth();
38418         }
38419     },
38420
38421     // private
38422     handleClick : function(e){
38423         if(!this.href){ // if no link defined, stop the event automatically
38424             e.stopEvent();
38425         }
38426         Roo.menu.Item.superclass.handleClick.apply(this, arguments);
38427     },
38428
38429     // private
38430     activate : function(autoExpand){
38431         if(Roo.menu.Item.superclass.activate.apply(this, arguments)){
38432             this.focus();
38433             if(autoExpand){
38434                 this.expandMenu();
38435             }
38436         }
38437         return true;
38438     },
38439
38440     // private
38441     shouldDeactivate : function(e){
38442         if(Roo.menu.Item.superclass.shouldDeactivate.call(this, e)){
38443             if(this.menu && this.menu.isVisible()){
38444                 return !this.menu.getEl().getRegion().contains(e.getPoint());
38445             }
38446             return true;
38447         }
38448         return false;
38449     },
38450
38451     // private
38452     deactivate : function(){
38453         Roo.menu.Item.superclass.deactivate.apply(this, arguments);
38454         this.hideMenu();
38455     },
38456
38457     // private
38458     expandMenu : function(autoActivate){
38459         if(!this.disabled && this.menu){
38460             clearTimeout(this.hideTimer);
38461             delete this.hideTimer;
38462             if(!this.menu.isVisible() && !this.showTimer){
38463                 this.showTimer = this.deferExpand.defer(this.showDelay, this, [autoActivate]);
38464             }else if (this.menu.isVisible() && autoActivate){
38465                 this.menu.tryActivate(0, 1);
38466             }
38467         }
38468     },
38469
38470     // private
38471     deferExpand : function(autoActivate){
38472         delete this.showTimer;
38473         this.menu.show(this.container, this.parentMenu.subMenuAlign || "tl-tr?", this.parentMenu);
38474         if(autoActivate){
38475             this.menu.tryActivate(0, 1);
38476         }
38477     },
38478
38479     // private
38480     hideMenu : function(){
38481         clearTimeout(this.showTimer);
38482         delete this.showTimer;
38483         if(!this.hideTimer && this.menu && this.menu.isVisible()){
38484             this.hideTimer = this.deferHide.defer(this.hideDelay, this);
38485         }
38486     },
38487
38488     // private
38489     deferHide : function(){
38490         delete this.hideTimer;
38491         this.menu.hide();
38492     }
38493 });/*
38494  * Based on:
38495  * Ext JS Library 1.1.1
38496  * Copyright(c) 2006-2007, Ext JS, LLC.
38497  *
38498  * Originally Released Under LGPL - original licence link has changed is not relivant.
38499  *
38500  * Fork - LGPL
38501  * <script type="text/javascript">
38502  */
38503  
38504 /**
38505  * @class Roo.menu.CheckItem
38506  * @extends Roo.menu.Item
38507  * Adds a menu item that contains a checkbox by default, but can also be part of a radio group.
38508  * @constructor
38509  * Creates a new CheckItem
38510  * @param {Object} config Configuration options
38511  */
38512 Roo.menu.CheckItem = function(config){
38513     Roo.menu.CheckItem.superclass.constructor.call(this, config);
38514     this.addEvents({
38515         /**
38516          * @event beforecheckchange
38517          * Fires before the checked value is set, providing an opportunity to cancel if needed
38518          * @param {Roo.menu.CheckItem} this
38519          * @param {Boolean} checked The new checked value that will be set
38520          */
38521         "beforecheckchange" : true,
38522         /**
38523          * @event checkchange
38524          * Fires after the checked value has been set
38525          * @param {Roo.menu.CheckItem} this
38526          * @param {Boolean} checked The checked value that was set
38527          */
38528         "checkchange" : true
38529     });
38530     if(this.checkHandler){
38531         this.on('checkchange', this.checkHandler, this.scope);
38532     }
38533 };
38534 Roo.extend(Roo.menu.CheckItem, Roo.menu.Item, {
38535     /**
38536      * @cfg {String} group
38537      * All check items with the same group name will automatically be grouped into a single-select
38538      * radio button group (defaults to '')
38539      */
38540     /**
38541      * @cfg {String} itemCls The default CSS class to use for check items (defaults to "x-menu-item x-menu-check-item")
38542      */
38543     itemCls : "x-menu-item x-menu-check-item",
38544     /**
38545      * @cfg {String} groupClass The default CSS class to use for radio group check items (defaults to "x-menu-group-item")
38546      */
38547     groupClass : "x-menu-group-item",
38548
38549     /**
38550      * @cfg {Boolean} checked True to initialize this checkbox as checked (defaults to false).  Note that
38551      * if this checkbox is part of a radio group (group = true) only the last item in the group that is
38552      * initialized with checked = true will be rendered as checked.
38553      */
38554     checked: false,
38555
38556     // private
38557     ctype: "Roo.menu.CheckItem",
38558
38559     // private
38560     onRender : function(c){
38561         Roo.menu.CheckItem.superclass.onRender.apply(this, arguments);
38562         if(this.group){
38563             this.el.addClass(this.groupClass);
38564         }
38565         Roo.menu.MenuMgr.registerCheckable(this);
38566         if(this.checked){
38567             this.checked = false;
38568             this.setChecked(true, true);
38569         }
38570     },
38571
38572     // private
38573     destroy : function(){
38574         if(this.rendered){
38575             Roo.menu.MenuMgr.unregisterCheckable(this);
38576         }
38577         Roo.menu.CheckItem.superclass.destroy.apply(this, arguments);
38578     },
38579
38580     /**
38581      * Set the checked state of this item
38582      * @param {Boolean} checked The new checked value
38583      * @param {Boolean} suppressEvent (optional) True to prevent the checkchange event from firing (defaults to false)
38584      */
38585     setChecked : function(state, suppressEvent){
38586         if(this.checked != state && this.fireEvent("beforecheckchange", this, state) !== false){
38587             if(this.container){
38588                 this.container[state ? "addClass" : "removeClass"]("x-menu-item-checked");
38589             }
38590             this.checked = state;
38591             if(suppressEvent !== true){
38592                 this.fireEvent("checkchange", this, state);
38593             }
38594         }
38595     },
38596
38597     // private
38598     handleClick : function(e){
38599        if(!this.disabled && !(this.checked && this.group)){// disable unselect on radio item
38600            this.setChecked(!this.checked);
38601        }
38602        Roo.menu.CheckItem.superclass.handleClick.apply(this, arguments);
38603     }
38604 });/*
38605  * Based on:
38606  * Ext JS Library 1.1.1
38607  * Copyright(c) 2006-2007, Ext JS, LLC.
38608  *
38609  * Originally Released Under LGPL - original licence link has changed is not relivant.
38610  *
38611  * Fork - LGPL
38612  * <script type="text/javascript">
38613  */
38614  
38615 /**
38616  * @class Roo.menu.DateItem
38617  * @extends Roo.menu.Adapter
38618  * A menu item that wraps the {@link Roo.DatPicker} component.
38619  * @constructor
38620  * Creates a new DateItem
38621  * @param {Object} config Configuration options
38622  */
38623 Roo.menu.DateItem = function(config){
38624     Roo.menu.DateItem.superclass.constructor.call(this, new Roo.DatePicker(config), config);
38625     /** The Roo.DatePicker object @type Roo.DatePicker */
38626     this.picker = this.component;
38627     this.addEvents({select: true});
38628     
38629     this.picker.on("render", function(picker){
38630         picker.getEl().swallowEvent("click");
38631         picker.container.addClass("x-menu-date-item");
38632     });
38633
38634     this.picker.on("select", this.onSelect, this);
38635 };
38636
38637 Roo.extend(Roo.menu.DateItem, Roo.menu.Adapter, {
38638     // private
38639     onSelect : function(picker, date){
38640         this.fireEvent("select", this, date, picker);
38641         Roo.menu.DateItem.superclass.handleClick.call(this);
38642     }
38643 });/*
38644  * Based on:
38645  * Ext JS Library 1.1.1
38646  * Copyright(c) 2006-2007, Ext JS, LLC.
38647  *
38648  * Originally Released Under LGPL - original licence link has changed is not relivant.
38649  *
38650  * Fork - LGPL
38651  * <script type="text/javascript">
38652  */
38653  
38654 /**
38655  * @class Roo.menu.ColorItem
38656  * @extends Roo.menu.Adapter
38657  * A menu item that wraps the {@link Roo.ColorPalette} component.
38658  * @constructor
38659  * Creates a new ColorItem
38660  * @param {Object} config Configuration options
38661  */
38662 Roo.menu.ColorItem = function(config){
38663     Roo.menu.ColorItem.superclass.constructor.call(this, new Roo.ColorPalette(config), config);
38664     /** The Roo.ColorPalette object @type Roo.ColorPalette */
38665     this.palette = this.component;
38666     this.relayEvents(this.palette, ["select"]);
38667     if(this.selectHandler){
38668         this.on('select', this.selectHandler, this.scope);
38669     }
38670 };
38671 Roo.extend(Roo.menu.ColorItem, Roo.menu.Adapter);/*
38672  * Based on:
38673  * Ext JS Library 1.1.1
38674  * Copyright(c) 2006-2007, Ext JS, LLC.
38675  *
38676  * Originally Released Under LGPL - original licence link has changed is not relivant.
38677  *
38678  * Fork - LGPL
38679  * <script type="text/javascript">
38680  */
38681  
38682
38683 /**
38684  * @class Roo.menu.DateMenu
38685  * @extends Roo.menu.Menu
38686  * A menu containing a {@link Roo.menu.DateItem} component (which provides a date picker).
38687  * @constructor
38688  * Creates a new DateMenu
38689  * @param {Object} config Configuration options
38690  */
38691 Roo.menu.DateMenu = function(config){
38692     Roo.menu.DateMenu.superclass.constructor.call(this, config);
38693     this.plain = true;
38694     var di = new Roo.menu.DateItem(config);
38695     this.add(di);
38696     /**
38697      * The {@link Roo.DatePicker} instance for this DateMenu
38698      * @type DatePicker
38699      */
38700     this.picker = di.picker;
38701     /**
38702      * @event select
38703      * @param {DatePicker} picker
38704      * @param {Date} date
38705      */
38706     this.relayEvents(di, ["select"]);
38707     this.on('beforeshow', function(){
38708         if(this.picker){
38709             this.picker.hideMonthPicker(false);
38710         }
38711     }, this);
38712 };
38713 Roo.extend(Roo.menu.DateMenu, Roo.menu.Menu, {
38714     cls:'x-date-menu'
38715 });/*
38716  * Based on:
38717  * Ext JS Library 1.1.1
38718  * Copyright(c) 2006-2007, Ext JS, LLC.
38719  *
38720  * Originally Released Under LGPL - original licence link has changed is not relivant.
38721  *
38722  * Fork - LGPL
38723  * <script type="text/javascript">
38724  */
38725  
38726
38727 /**
38728  * @class Roo.menu.ColorMenu
38729  * @extends Roo.menu.Menu
38730  * A menu containing a {@link Roo.menu.ColorItem} component (which provides a basic color picker).
38731  * @constructor
38732  * Creates a new ColorMenu
38733  * @param {Object} config Configuration options
38734  */
38735 Roo.menu.ColorMenu = function(config){
38736     Roo.menu.ColorMenu.superclass.constructor.call(this, config);
38737     this.plain = true;
38738     var ci = new Roo.menu.ColorItem(config);
38739     this.add(ci);
38740     /**
38741      * The {@link Roo.ColorPalette} instance for this ColorMenu
38742      * @type ColorPalette
38743      */
38744     this.palette = ci.palette;
38745     /**
38746      * @event select
38747      * @param {ColorPalette} palette
38748      * @param {String} color
38749      */
38750     this.relayEvents(ci, ["select"]);
38751 };
38752 Roo.extend(Roo.menu.ColorMenu, Roo.menu.Menu);/*
38753  * Based on:
38754  * Ext JS Library 1.1.1
38755  * Copyright(c) 2006-2007, Ext JS, LLC.
38756  *
38757  * Originally Released Under LGPL - original licence link has changed is not relivant.
38758  *
38759  * Fork - LGPL
38760  * <script type="text/javascript">
38761  */
38762  
38763 /**
38764  * @class Roo.form.TextItem
38765  * @extends Roo.BoxComponent
38766  * Base class for form fields that provides default event handling, sizing, value handling and other functionality.
38767  * @constructor
38768  * Creates a new TextItem
38769  * @param {Object} config Configuration options
38770  */
38771 Roo.form.TextItem = function(config){
38772     Roo.form.TextItem.superclass.constructor.call(this, config);
38773 };
38774
38775 Roo.extend(Roo.form.TextItem, Roo.BoxComponent,  {
38776     
38777     /**
38778      * @cfg {String} tag the tag for this item (default div)
38779      */
38780     tag : 'div',
38781     /**
38782      * @cfg {String} html the content for this item
38783      */
38784     html : '',
38785     
38786     getAutoCreate : function()
38787     {
38788         var cfg = {
38789             id: this.id,
38790             tag: this.tag,
38791             html: this.html,
38792             cls: 'x-form-item'
38793         };
38794         
38795         return cfg;
38796         
38797     },
38798     
38799     onRender : function(ct, position)
38800     {
38801         Roo.form.TextItem.superclass.onRender.call(this, ct, position);
38802         
38803         if(!this.el){
38804             var cfg = this.getAutoCreate();
38805             if(!cfg.name){
38806                 cfg.name = typeof(this.name) == 'undefined' ? this.id : this.name;
38807             }
38808             if (!cfg.name.length) {
38809                 delete cfg.name;
38810             }
38811             this.el = ct.createChild(cfg, position);
38812         }
38813     }
38814     
38815 });/*
38816  * Based on:
38817  * Ext JS Library 1.1.1
38818  * Copyright(c) 2006-2007, Ext JS, LLC.
38819  *
38820  * Originally Released Under LGPL - original licence link has changed is not relivant.
38821  *
38822  * Fork - LGPL
38823  * <script type="text/javascript">
38824  */
38825  
38826 /**
38827  * @class Roo.form.Field
38828  * @extends Roo.BoxComponent
38829  * Base class for form fields that provides default event handling, sizing, value handling and other functionality.
38830  * @constructor
38831  * Creates a new Field
38832  * @param {Object} config Configuration options
38833  */
38834 Roo.form.Field = function(config){
38835     Roo.form.Field.superclass.constructor.call(this, config);
38836 };
38837
38838 Roo.extend(Roo.form.Field, Roo.BoxComponent,  {
38839     /**
38840      * @cfg {String} fieldLabel Label to use when rendering a form.
38841      */
38842        /**
38843      * @cfg {String} qtip Mouse over tip
38844      */
38845      
38846     /**
38847      * @cfg {String} invalidClass The CSS class to use when marking a field invalid (defaults to "x-form-invalid")
38848      */
38849     invalidClass : "x-form-invalid",
38850     /**
38851      * @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")
38852      */
38853     invalidText : "The value in this field is invalid",
38854     /**
38855      * @cfg {String} focusClass The CSS class to use when the field receives focus (defaults to "x-form-focus")
38856      */
38857     focusClass : "x-form-focus",
38858     /**
38859      * @cfg {String/Boolean} validationEvent The event that should initiate field validation. Set to false to disable
38860       automatic validation (defaults to "keyup").
38861      */
38862     validationEvent : "keyup",
38863     /**
38864      * @cfg {Boolean} validateOnBlur Whether the field should validate when it loses focus (defaults to true).
38865      */
38866     validateOnBlur : true,
38867     /**
38868      * @cfg {Number} validationDelay The length of time in milliseconds after user input begins until validation is initiated (defaults to 250)
38869      */
38870     validationDelay : 250,
38871     /**
38872      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
38873      * {tag: "input", type: "text", size: "20", autocomplete: "off"})
38874      */
38875     defaultAutoCreate : {tag: "input", type: "text", size: "20", autocomplete: "new-password"},
38876     /**
38877      * @cfg {String} fieldClass The default CSS class for the field (defaults to "x-form-field")
38878      */
38879     fieldClass : "x-form-field",
38880     /**
38881      * @cfg {String} msgTarget The location where error text should display.  Should be one of the following values (defaults to 'qtip'):
38882      *<pre>
38883 Value         Description
38884 -----------   ----------------------------------------------------------------------
38885 qtip          Display a quick tip when the user hovers over the field
38886 title         Display a default browser title attribute popup
38887 under         Add a block div beneath the field containing the error text
38888 side          Add an error icon to the right of the field with a popup on hover
38889 [element id]  Add the error text directly to the innerHTML of the specified element
38890 </pre>
38891      */
38892     msgTarget : 'qtip',
38893     /**
38894      * @cfg {String} msgFx <b>Experimental</b> The effect used when displaying a validation message under the field (defaults to 'normal').
38895      */
38896     msgFx : 'normal',
38897
38898     /**
38899      * @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.
38900      */
38901     readOnly : false,
38902
38903     /**
38904      * @cfg {Boolean} disabled True to disable the field (defaults to false).
38905      */
38906     disabled : false,
38907
38908     /**
38909      * @cfg {String} inputType The type attribute for input fields -- e.g. radio, text, password (defaults to "text").
38910      */
38911     inputType : undefined,
38912     
38913     /**
38914      * @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).
38915          */
38916         tabIndex : undefined,
38917         
38918     // private
38919     isFormField : true,
38920
38921     // private
38922     hasFocus : false,
38923     /**
38924      * @property {Roo.Element} fieldEl
38925      * Element Containing the rendered Field (with label etc.)
38926      */
38927     /**
38928      * @cfg {Mixed} value A value to initialize this field with.
38929      */
38930     value : undefined,
38931
38932     /**
38933      * @cfg {String} name The field's HTML name attribute.
38934      */
38935     /**
38936      * @cfg {String} cls A CSS class to apply to the field's underlying element.
38937      */
38938     // private
38939     loadedValue : false,
38940      
38941      
38942         // private ??
38943         initComponent : function(){
38944         Roo.form.Field.superclass.initComponent.call(this);
38945         this.addEvents({
38946             /**
38947              * @event focus
38948              * Fires when this field receives input focus.
38949              * @param {Roo.form.Field} this
38950              */
38951             focus : true,
38952             /**
38953              * @event blur
38954              * Fires when this field loses input focus.
38955              * @param {Roo.form.Field} this
38956              */
38957             blur : true,
38958             /**
38959              * @event specialkey
38960              * Fires when any key related to navigation (arrows, tab, enter, esc, etc.) is pressed.  You can check
38961              * {@link Roo.EventObject#getKey} to determine which key was pressed.
38962              * @param {Roo.form.Field} this
38963              * @param {Roo.EventObject} e The event object
38964              */
38965             specialkey : true,
38966             /**
38967              * @event change
38968              * Fires just before the field blurs if the field value has changed.
38969              * @param {Roo.form.Field} this
38970              * @param {Mixed} newValue The new value
38971              * @param {Mixed} oldValue The original value
38972              */
38973             change : true,
38974             /**
38975              * @event invalid
38976              * Fires after the field has been marked as invalid.
38977              * @param {Roo.form.Field} this
38978              * @param {String} msg The validation message
38979              */
38980             invalid : true,
38981             /**
38982              * @event valid
38983              * Fires after the field has been validated with no errors.
38984              * @param {Roo.form.Field} this
38985              */
38986             valid : true,
38987              /**
38988              * @event keyup
38989              * Fires after the key up
38990              * @param {Roo.form.Field} this
38991              * @param {Roo.EventObject}  e The event Object
38992              */
38993             keyup : true
38994         });
38995     },
38996
38997     /**
38998      * Returns the name attribute of the field if available
38999      * @return {String} name The field name
39000      */
39001     getName: function(){
39002          return this.rendered && this.el.dom.name ? this.el.dom.name : (this.hiddenName || '');
39003     },
39004
39005     // private
39006     onRender : function(ct, position){
39007         Roo.form.Field.superclass.onRender.call(this, ct, position);
39008         if(!this.el){
39009             var cfg = this.getAutoCreate();
39010             if(!cfg.name){
39011                 cfg.name = typeof(this.name) == 'undefined' ? this.id : this.name;
39012             }
39013             if (!cfg.name.length) {
39014                 delete cfg.name;
39015             }
39016             if(this.inputType){
39017                 cfg.type = this.inputType;
39018             }
39019             this.el = ct.createChild(cfg, position);
39020         }
39021         var type = this.el.dom.type;
39022         if(type){
39023             if(type == 'password'){
39024                 type = 'text';
39025             }
39026             this.el.addClass('x-form-'+type);
39027         }
39028         if(this.readOnly){
39029             this.el.dom.readOnly = true;
39030         }
39031         if(this.tabIndex !== undefined){
39032             this.el.dom.setAttribute('tabIndex', this.tabIndex);
39033         }
39034
39035         this.el.addClass([this.fieldClass, this.cls]);
39036         this.initValue();
39037     },
39038
39039     /**
39040      * Apply the behaviors of this component to an existing element. <b>This is used instead of render().</b>
39041      * @param {String/HTMLElement/Element} el The id of the node, a DOM node or an existing Element
39042      * @return {Roo.form.Field} this
39043      */
39044     applyTo : function(target){
39045         this.allowDomMove = false;
39046         this.el = Roo.get(target);
39047         this.render(this.el.dom.parentNode);
39048         return this;
39049     },
39050
39051     // private
39052     initValue : function(){
39053         if(this.value !== undefined){
39054             this.setValue(this.value);
39055         }else if(this.el.dom.value.length > 0){
39056             this.setValue(this.el.dom.value);
39057         }
39058     },
39059
39060     /**
39061      * Returns true if this field has been changed since it was originally loaded and is not disabled.
39062      * DEPRICATED  - it never worked well - use hasChanged/resetHasChanged.
39063      */
39064     isDirty : function() {
39065         if(this.disabled) {
39066             return false;
39067         }
39068         return String(this.getValue()) !== String(this.originalValue);
39069     },
39070
39071     /**
39072      * stores the current value in loadedValue
39073      */
39074     resetHasChanged : function()
39075     {
39076         this.loadedValue = String(this.getValue());
39077     },
39078     /**
39079      * checks the current value against the 'loaded' value.
39080      * Note - will return false if 'resetHasChanged' has not been called first.
39081      */
39082     hasChanged : function()
39083     {
39084         if(this.disabled || this.readOnly) {
39085             return false;
39086         }
39087         return this.loadedValue !== false && String(this.getValue()) !== this.loadedValue;
39088     },
39089     
39090     
39091     
39092     // private
39093     afterRender : function(){
39094         Roo.form.Field.superclass.afterRender.call(this);
39095         this.initEvents();
39096     },
39097
39098     // private
39099     fireKey : function(e){
39100         //Roo.log('field ' + e.getKey());
39101         if(e.isNavKeyPress()){
39102             this.fireEvent("specialkey", this, e);
39103         }
39104     },
39105
39106     /**
39107      * Resets the current field value to the originally loaded value and clears any validation messages
39108      */
39109     reset : function(){
39110         this.setValue(this.resetValue);
39111         this.originalValue = this.getValue();
39112         this.clearInvalid();
39113     },
39114
39115     // private
39116     initEvents : function(){
39117         // safari killled keypress - so keydown is now used..
39118         this.el.on("keydown" , this.fireKey,  this);
39119         this.el.on("focus", this.onFocus,  this);
39120         this.el.on("blur", this.onBlur,  this);
39121         this.el.relayEvent('keyup', this);
39122
39123         // reference to original value for reset
39124         this.originalValue = this.getValue();
39125         this.resetValue =  this.getValue();
39126     },
39127
39128     // private
39129     onFocus : function(){
39130         if(!Roo.isOpera && this.focusClass){ // don't touch in Opera
39131             this.el.addClass(this.focusClass);
39132         }
39133         if(!this.hasFocus){
39134             this.hasFocus = true;
39135             this.startValue = this.getValue();
39136             this.fireEvent("focus", this);
39137         }
39138     },
39139
39140     beforeBlur : Roo.emptyFn,
39141
39142     // private
39143     onBlur : function(){
39144         this.beforeBlur();
39145         if(!Roo.isOpera && this.focusClass){ // don't touch in Opera
39146             this.el.removeClass(this.focusClass);
39147         }
39148         this.hasFocus = false;
39149         if(this.validationEvent !== false && this.validateOnBlur && this.validationEvent != "blur"){
39150             this.validate();
39151         }
39152         var v = this.getValue();
39153         if(String(v) !== String(this.startValue)){
39154             this.fireEvent('change', this, v, this.startValue);
39155         }
39156         this.fireEvent("blur", this);
39157     },
39158
39159     /**
39160      * Returns whether or not the field value is currently valid
39161      * @param {Boolean} preventMark True to disable marking the field invalid
39162      * @return {Boolean} True if the value is valid, else false
39163      */
39164     isValid : function(preventMark){
39165         if(this.disabled){
39166             return true;
39167         }
39168         var restore = this.preventMark;
39169         this.preventMark = preventMark === true;
39170         var v = this.validateValue(this.processValue(this.getRawValue()));
39171         this.preventMark = restore;
39172         return v;
39173     },
39174
39175     /**
39176      * Validates the field value
39177      * @return {Boolean} True if the value is valid, else false
39178      */
39179     validate : function(){
39180         if(this.disabled || this.validateValue(this.processValue(this.getRawValue()))){
39181             this.clearInvalid();
39182             return true;
39183         }
39184         return false;
39185     },
39186
39187     processValue : function(value){
39188         return value;
39189     },
39190
39191     // private
39192     // Subclasses should provide the validation implementation by overriding this
39193     validateValue : function(value){
39194         return true;
39195     },
39196
39197     /**
39198      * Mark this field as invalid
39199      * @param {String} msg The validation message
39200      */
39201     markInvalid : function(msg){
39202         if(!this.rendered || this.preventMark){ // not rendered
39203             return;
39204         }
39205         
39206         var obj = (typeof(this.combo) != 'undefined') ? this.combo : this; // fix the combox array!!
39207         
39208         obj.el.addClass(this.invalidClass);
39209         msg = msg || this.invalidText;
39210         switch(this.msgTarget){
39211             case 'qtip':
39212                 obj.el.dom.qtip = msg;
39213                 obj.el.dom.qclass = 'x-form-invalid-tip';
39214                 if(Roo.QuickTips){ // fix for floating editors interacting with DND
39215                     Roo.QuickTips.enable();
39216                 }
39217                 break;
39218             case 'title':
39219                 this.el.dom.title = msg;
39220                 break;
39221             case 'under':
39222                 if(!this.errorEl){
39223                     var elp = this.el.findParent('.x-form-element', 5, true);
39224                     this.errorEl = elp.createChild({cls:'x-form-invalid-msg'});
39225                     this.errorEl.setWidth(elp.getWidth(true)-20);
39226                 }
39227                 this.errorEl.update(msg);
39228                 Roo.form.Field.msgFx[this.msgFx].show(this.errorEl, this);
39229                 break;
39230             case 'side':
39231                 if(!this.errorIcon){
39232                     var elp = this.el.findParent('.x-form-element', 5, true);
39233                     this.errorIcon = elp.createChild({cls:'x-form-invalid-icon'});
39234                 }
39235                 this.alignErrorIcon();
39236                 this.errorIcon.dom.qtip = msg;
39237                 this.errorIcon.dom.qclass = 'x-form-invalid-tip';
39238                 this.errorIcon.show();
39239                 this.on('resize', this.alignErrorIcon, this);
39240                 break;
39241             default:
39242                 var t = Roo.getDom(this.msgTarget);
39243                 t.innerHTML = msg;
39244                 t.style.display = this.msgDisplay;
39245                 break;
39246         }
39247         this.fireEvent('invalid', this, msg);
39248     },
39249
39250     // private
39251     alignErrorIcon : function(){
39252         this.errorIcon.alignTo(this.el, 'tl-tr', [2, 0]);
39253     },
39254
39255     /**
39256      * Clear any invalid styles/messages for this field
39257      */
39258     clearInvalid : function(){
39259         if(!this.rendered || this.preventMark){ // not rendered
39260             return;
39261         }
39262         var obj = (typeof(this.combo) != 'undefined') ? this.combo : this; // fix the combox array!!
39263         
39264         obj.el.removeClass(this.invalidClass);
39265         switch(this.msgTarget){
39266             case 'qtip':
39267                 obj.el.dom.qtip = '';
39268                 break;
39269             case 'title':
39270                 this.el.dom.title = '';
39271                 break;
39272             case 'under':
39273                 if(this.errorEl){
39274                     Roo.form.Field.msgFx[this.msgFx].hide(this.errorEl, this);
39275                 }
39276                 break;
39277             case 'side':
39278                 if(this.errorIcon){
39279                     this.errorIcon.dom.qtip = '';
39280                     this.errorIcon.hide();
39281                     this.un('resize', this.alignErrorIcon, this);
39282                 }
39283                 break;
39284             default:
39285                 var t = Roo.getDom(this.msgTarget);
39286                 t.innerHTML = '';
39287                 t.style.display = 'none';
39288                 break;
39289         }
39290         this.fireEvent('valid', this);
39291     },
39292
39293     /**
39294      * Returns the raw data value which may or may not be a valid, defined value.  To return a normalized value see {@link #getValue}.
39295      * @return {Mixed} value The field value
39296      */
39297     getRawValue : function(){
39298         var v = this.el.getValue();
39299         
39300         return v;
39301     },
39302
39303     /**
39304      * Returns the normalized data value (undefined or emptyText will be returned as '').  To return the raw value see {@link #getRawValue}.
39305      * @return {Mixed} value The field value
39306      */
39307     getValue : function(){
39308         var v = this.el.getValue();
39309          
39310         return v;
39311     },
39312
39313     /**
39314      * Sets the underlying DOM field's value directly, bypassing validation.  To set the value with validation see {@link #setValue}.
39315      * @param {Mixed} value The value to set
39316      */
39317     setRawValue : function(v){
39318         return this.el.dom.value = (v === null || v === undefined ? '' : v);
39319     },
39320
39321     /**
39322      * Sets a data value into the field and validates it.  To set the value directly without validation see {@link #setRawValue}.
39323      * @param {Mixed} value The value to set
39324      */
39325     setValue : function(v){
39326         this.value = v;
39327         if(this.rendered){
39328             this.el.dom.value = (v === null || v === undefined ? '' : v);
39329              this.validate();
39330         }
39331     },
39332
39333     adjustSize : function(w, h){
39334         var s = Roo.form.Field.superclass.adjustSize.call(this, w, h);
39335         s.width = this.adjustWidth(this.el.dom.tagName, s.width);
39336         return s;
39337     },
39338
39339     adjustWidth : function(tag, w){
39340         tag = tag.toLowerCase();
39341         if(typeof w == 'number' && Roo.isStrict && !Roo.isSafari){
39342             if(Roo.isIE && (tag == 'input' || tag == 'textarea')){
39343                 if(tag == 'input'){
39344                     return w + 2;
39345                 }
39346                 if(tag == 'textarea'){
39347                     return w-2;
39348                 }
39349             }else if(Roo.isOpera){
39350                 if(tag == 'input'){
39351                     return w + 2;
39352                 }
39353                 if(tag == 'textarea'){
39354                     return w-2;
39355                 }
39356             }
39357         }
39358         return w;
39359     }
39360 });
39361
39362
39363 // anything other than normal should be considered experimental
39364 Roo.form.Field.msgFx = {
39365     normal : {
39366         show: function(msgEl, f){
39367             msgEl.setDisplayed('block');
39368         },
39369
39370         hide : function(msgEl, f){
39371             msgEl.setDisplayed(false).update('');
39372         }
39373     },
39374
39375     slide : {
39376         show: function(msgEl, f){
39377             msgEl.slideIn('t', {stopFx:true});
39378         },
39379
39380         hide : function(msgEl, f){
39381             msgEl.slideOut('t', {stopFx:true,useDisplay:true});
39382         }
39383     },
39384
39385     slideRight : {
39386         show: function(msgEl, f){
39387             msgEl.fixDisplay();
39388             msgEl.alignTo(f.el, 'tl-tr');
39389             msgEl.slideIn('l', {stopFx:true});
39390         },
39391
39392         hide : function(msgEl, f){
39393             msgEl.slideOut('l', {stopFx:true,useDisplay:true});
39394         }
39395     }
39396 };/*
39397  * Based on:
39398  * Ext JS Library 1.1.1
39399  * Copyright(c) 2006-2007, Ext JS, LLC.
39400  *
39401  * Originally Released Under LGPL - original licence link has changed is not relivant.
39402  *
39403  * Fork - LGPL
39404  * <script type="text/javascript">
39405  */
39406  
39407
39408 /**
39409  * @class Roo.form.TextField
39410  * @extends Roo.form.Field
39411  * Basic text field.  Can be used as a direct replacement for traditional text inputs, or as the base
39412  * class for more sophisticated input controls (like {@link Roo.form.TextArea} and {@link Roo.form.ComboBox}).
39413  * @constructor
39414  * Creates a new TextField
39415  * @param {Object} config Configuration options
39416  */
39417 Roo.form.TextField = function(config){
39418     Roo.form.TextField.superclass.constructor.call(this, config);
39419     this.addEvents({
39420         /**
39421          * @event autosize
39422          * Fires when the autosize function is triggered.  The field may or may not have actually changed size
39423          * according to the default logic, but this event provides a hook for the developer to apply additional
39424          * logic at runtime to resize the field if needed.
39425              * @param {Roo.form.Field} this This text field
39426              * @param {Number} width The new field width
39427              */
39428         autosize : true
39429     });
39430 };
39431
39432 Roo.extend(Roo.form.TextField, Roo.form.Field,  {
39433     /**
39434      * @cfg {Boolean} grow True if this field should automatically grow and shrink to its content
39435      */
39436     grow : false,
39437     /**
39438      * @cfg {Number} growMin The minimum width to allow when grow = true (defaults to 30)
39439      */
39440     growMin : 30,
39441     /**
39442      * @cfg {Number} growMax The maximum width to allow when grow = true (defaults to 800)
39443      */
39444     growMax : 800,
39445     /**
39446      * @cfg {String} vtype A validation type name as defined in {@link Roo.form.VTypes} (defaults to null)
39447      */
39448     vtype : null,
39449     /**
39450      * @cfg {String} maskRe An input mask regular expression that will be used to filter keystrokes that don't match (defaults to null)
39451      */
39452     maskRe : null,
39453     /**
39454      * @cfg {Boolean} disableKeyFilter True to disable input keystroke filtering (defaults to false)
39455      */
39456     disableKeyFilter : false,
39457     /**
39458      * @cfg {Boolean} allowBlank False to validate that the value length > 0 (defaults to true)
39459      */
39460     allowBlank : true,
39461     /**
39462      * @cfg {Number} minLength Minimum input field length required (defaults to 0)
39463      */
39464     minLength : 0,
39465     /**
39466      * @cfg {Number} maxLength Maximum input field length allowed (defaults to Number.MAX_VALUE)
39467      */
39468     maxLength : Number.MAX_VALUE,
39469     /**
39470      * @cfg {String} minLengthText Error text to display if the minimum length validation fails (defaults to "The minimum length for this field is {minLength}")
39471      */
39472     minLengthText : "The minimum length for this field is {0}",
39473     /**
39474      * @cfg {String} maxLengthText Error text to display if the maximum length validation fails (defaults to "The maximum length for this field is {maxLength}")
39475      */
39476     maxLengthText : "The maximum length for this field is {0}",
39477     /**
39478      * @cfg {Boolean} selectOnFocus True to automatically select any existing field text when the field receives input focus (defaults to false)
39479      */
39480     selectOnFocus : false,
39481     /**
39482      * @cfg {Boolean} allowLeadingSpace True to prevent the stripping of leading white space 
39483      */    
39484     allowLeadingSpace : false,
39485     /**
39486      * @cfg {String} blankText Error text to display if the allow blank validation fails (defaults to "This field is required")
39487      */
39488     blankText : "This field is required",
39489     /**
39490      * @cfg {Function} validator A custom validation function to be called during field validation (defaults to null).
39491      * If available, this function will be called only after the basic validators all return true, and will be passed the
39492      * current field value and expected to return boolean true if the value is valid or a string error message if invalid.
39493      */
39494     validator : null,
39495     /**
39496      * @cfg {RegExp} regex A JavaScript RegExp object to be tested against the field value during validation (defaults to null).
39497      * If available, this regex will be evaluated only after the basic validators all return true, and will be passed the
39498      * current field value.  If the test fails, the field will be marked invalid using {@link #regexText}.
39499      */
39500     regex : null,
39501     /**
39502      * @cfg {String} regexText The error text to display if {@link #regex} is used and the test fails during validation (defaults to "")
39503      */
39504     regexText : "",
39505     /**
39506      * @cfg {String} emptyText The default text to display in an empty field - placeholder... (defaults to null).
39507      */
39508     emptyText : null,
39509    
39510
39511     // private
39512     initEvents : function()
39513     {
39514         if (this.emptyText) {
39515             this.el.attr('placeholder', this.emptyText);
39516         }
39517         
39518         Roo.form.TextField.superclass.initEvents.call(this);
39519         if(this.validationEvent == 'keyup'){
39520             this.validationTask = new Roo.util.DelayedTask(this.validate, this);
39521             this.el.on('keyup', this.filterValidation, this);
39522         }
39523         else if(this.validationEvent !== false){
39524             this.el.on(this.validationEvent, this.validate, this, {buffer: this.validationDelay});
39525         }
39526         
39527         if(this.selectOnFocus){
39528             this.on("focus", this.preFocus, this);
39529         }
39530         if (!this.allowLeadingSpace) {
39531             this.on('blur', this.cleanLeadingSpace, this);
39532         }
39533         
39534         if(this.maskRe || (this.vtype && this.disableKeyFilter !== true && (this.maskRe = Roo.form.VTypes[this.vtype+'Mask']))){
39535             this.el.on("keypress", this.filterKeys, this);
39536         }
39537         if(this.grow){
39538             this.el.on("keyup", this.onKeyUp,  this, {buffer:50});
39539             this.el.on("click", this.autoSize,  this);
39540         }
39541         if(this.el.is('input[type=password]') && Roo.isSafari){
39542             this.el.on('keydown', this.SafariOnKeyDown, this);
39543         }
39544     },
39545
39546     processValue : function(value){
39547         if(this.stripCharsRe){
39548             var newValue = value.replace(this.stripCharsRe, '');
39549             if(newValue !== value){
39550                 this.setRawValue(newValue);
39551                 return newValue;
39552             }
39553         }
39554         return value;
39555     },
39556
39557     filterValidation : function(e){
39558         if(!e.isNavKeyPress()){
39559             this.validationTask.delay(this.validationDelay);
39560         }
39561     },
39562
39563     // private
39564     onKeyUp : function(e){
39565         if(!e.isNavKeyPress()){
39566             this.autoSize();
39567         }
39568     },
39569     // private - clean the leading white space
39570     cleanLeadingSpace : function(e)
39571     {
39572         if ( this.inputType == 'file') {
39573             return;
39574         }
39575         
39576         this.setValue((this.getValue() + '').replace(/^\s+/,''));
39577     },
39578     /**
39579      * Resets the current field value to the originally-loaded value and clears any validation messages.
39580      *  
39581      */
39582     reset : function(){
39583         Roo.form.TextField.superclass.reset.call(this);
39584        
39585     }, 
39586     // private
39587     preFocus : function(){
39588         
39589         if(this.selectOnFocus){
39590             this.el.dom.select();
39591         }
39592     },
39593
39594     
39595     // private
39596     filterKeys : function(e){
39597         var k = e.getKey();
39598         if(!Roo.isIE && (e.isNavKeyPress() || k == e.BACKSPACE || (k == e.DELETE && e.button == -1))){
39599             return;
39600         }
39601         var c = e.getCharCode(), cc = String.fromCharCode(c);
39602         if(Roo.isIE && (e.isSpecialKey() || !cc)){
39603             return;
39604         }
39605         if(!this.maskRe.test(cc)){
39606             e.stopEvent();
39607         }
39608     },
39609
39610     setValue : function(v){
39611         
39612         Roo.form.TextField.superclass.setValue.apply(this, arguments);
39613         
39614         this.autoSize();
39615     },
39616
39617     /**
39618      * Validates a value according to the field's validation rules and marks the field as invalid
39619      * if the validation fails
39620      * @param {Mixed} value The value to validate
39621      * @return {Boolean} True if the value is valid, else false
39622      */
39623     validateValue : function(value){
39624         if(value.length < 1)  { // if it's blank
39625              if(this.allowBlank){
39626                 this.clearInvalid();
39627                 return true;
39628              }else{
39629                 this.markInvalid(this.blankText);
39630                 return false;
39631              }
39632         }
39633         if(value.length < this.minLength){
39634             this.markInvalid(String.format(this.minLengthText, this.minLength));
39635             return false;
39636         }
39637         if(value.length > this.maxLength){
39638             this.markInvalid(String.format(this.maxLengthText, this.maxLength));
39639             return false;
39640         }
39641         if(this.vtype){
39642             var vt = Roo.form.VTypes;
39643             if(!vt[this.vtype](value, this)){
39644                 this.markInvalid(this.vtypeText || vt[this.vtype +'Text']);
39645                 return false;
39646             }
39647         }
39648         if(typeof this.validator == "function"){
39649             var msg = this.validator(value);
39650             if(msg !== true){
39651                 this.markInvalid(msg);
39652                 return false;
39653             }
39654         }
39655         if(this.regex && !this.regex.test(value)){
39656             this.markInvalid(this.regexText);
39657             return false;
39658         }
39659         return true;
39660     },
39661
39662     /**
39663      * Selects text in this field
39664      * @param {Number} start (optional) The index where the selection should start (defaults to 0)
39665      * @param {Number} end (optional) The index where the selection should end (defaults to the text length)
39666      */
39667     selectText : function(start, end){
39668         var v = this.getRawValue();
39669         if(v.length > 0){
39670             start = start === undefined ? 0 : start;
39671             end = end === undefined ? v.length : end;
39672             var d = this.el.dom;
39673             if(d.setSelectionRange){
39674                 d.setSelectionRange(start, end);
39675             }else if(d.createTextRange){
39676                 var range = d.createTextRange();
39677                 range.moveStart("character", start);
39678                 range.moveEnd("character", v.length-end);
39679                 range.select();
39680             }
39681         }
39682     },
39683
39684     /**
39685      * Automatically grows the field to accomodate the width of the text up to the maximum field width allowed.
39686      * This only takes effect if grow = true, and fires the autosize event.
39687      */
39688     autoSize : function(){
39689         if(!this.grow || !this.rendered){
39690             return;
39691         }
39692         if(!this.metrics){
39693             this.metrics = Roo.util.TextMetrics.createInstance(this.el);
39694         }
39695         var el = this.el;
39696         var v = el.dom.value;
39697         var d = document.createElement('div');
39698         d.appendChild(document.createTextNode(v));
39699         v = d.innerHTML;
39700         d = null;
39701         v += "&#160;";
39702         var w = Math.min(this.growMax, Math.max(this.metrics.getWidth(v) + /* add extra padding */ 10, this.growMin));
39703         this.el.setWidth(w);
39704         this.fireEvent("autosize", this, w);
39705     },
39706     
39707     // private
39708     SafariOnKeyDown : function(event)
39709     {
39710         // this is a workaround for a password hang bug on chrome/ webkit.
39711         
39712         var isSelectAll = false;
39713         
39714         if(this.el.dom.selectionEnd > 0){
39715             isSelectAll = (this.el.dom.selectionEnd - this.el.dom.selectionStart - this.getValue().length == 0) ? true : false;
39716         }
39717         if(((event.getKey() == 8 || event.getKey() == 46) && this.getValue().length ==1)){ // backspace and delete key
39718             event.preventDefault();
39719             this.setValue('');
39720             return;
39721         }
39722         
39723         if(isSelectAll && event.getCharCode() > 31){ // backspace and delete key
39724             
39725             event.preventDefault();
39726             // this is very hacky as keydown always get's upper case.
39727             
39728             var cc = String.fromCharCode(event.getCharCode());
39729             
39730             
39731             this.setValue( event.shiftKey ?  cc : cc.toLowerCase());
39732             
39733         }
39734         
39735         
39736     }
39737 });/*
39738  * Based on:
39739  * Ext JS Library 1.1.1
39740  * Copyright(c) 2006-2007, Ext JS, LLC.
39741  *
39742  * Originally Released Under LGPL - original licence link has changed is not relivant.
39743  *
39744  * Fork - LGPL
39745  * <script type="text/javascript">
39746  */
39747  
39748 /**
39749  * @class Roo.form.Hidden
39750  * @extends Roo.form.TextField
39751  * Simple Hidden element used on forms 
39752  * 
39753  * usage: form.add(new Roo.form.HiddenField({ 'name' : 'test1' }));
39754  * 
39755  * @constructor
39756  * Creates a new Hidden form element.
39757  * @param {Object} config Configuration options
39758  */
39759
39760
39761
39762 // easy hidden field...
39763 Roo.form.Hidden = function(config){
39764     Roo.form.Hidden.superclass.constructor.call(this, config);
39765 };
39766   
39767 Roo.extend(Roo.form.Hidden, Roo.form.TextField, {
39768     fieldLabel:      '',
39769     inputType:      'hidden',
39770     width:          50,
39771     allowBlank:     true,
39772     labelSeparator: '',
39773     hidden:         true,
39774     itemCls :       'x-form-item-display-none'
39775
39776
39777 });
39778
39779
39780 /*
39781  * Based on:
39782  * Ext JS Library 1.1.1
39783  * Copyright(c) 2006-2007, Ext JS, LLC.
39784  *
39785  * Originally Released Under LGPL - original licence link has changed is not relivant.
39786  *
39787  * Fork - LGPL
39788  * <script type="text/javascript">
39789  */
39790  
39791 /**
39792  * @class Roo.form.TriggerField
39793  * @extends Roo.form.TextField
39794  * Provides a convenient wrapper for TextFields that adds a clickable trigger button (looks like a combobox by default).
39795  * The trigger has no default action, so you must assign a function to implement the trigger click handler by
39796  * overriding {@link #onTriggerClick}. You can create a TriggerField directly, as it renders exactly like a combobox
39797  * for which you can provide a custom implementation.  For example:
39798  * <pre><code>
39799 var trigger = new Roo.form.TriggerField();
39800 trigger.onTriggerClick = myTriggerFn;
39801 trigger.applyTo('my-field');
39802 </code></pre>
39803  *
39804  * However, in general you will most likely want to use TriggerField as the base class for a reusable component.
39805  * {@link Roo.form.DateField} and {@link Roo.form.ComboBox} are perfect examples of this.
39806  * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
39807  * class 'x-form-trigger' by default and triggerClass will be <b>appended</b> if specified.
39808  * @constructor
39809  * Create a new TriggerField.
39810  * @param {Object} config Configuration options (valid {@Roo.form.TextField} config options will also be applied
39811  * to the base TextField)
39812  */
39813 Roo.form.TriggerField = function(config){
39814     this.mimicing = false;
39815     Roo.form.TriggerField.superclass.constructor.call(this, config);
39816 };
39817
39818 Roo.extend(Roo.form.TriggerField, Roo.form.TextField,  {
39819     /**
39820      * @cfg {String} triggerClass A CSS class to apply to the trigger
39821      */
39822     /**
39823      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
39824      * {tag: "input", type: "text", size: "16", autocomplete: "off"})
39825      */
39826     defaultAutoCreate : {tag: "input", type: "text", size: "16", autocomplete: "new-password"},
39827     /**
39828      * @cfg {Boolean} hideTrigger True to hide the trigger element and display only the base text field (defaults to false)
39829      */
39830     hideTrigger:false,
39831
39832     /** @cfg {Boolean} grow @hide */
39833     /** @cfg {Number} growMin @hide */
39834     /** @cfg {Number} growMax @hide */
39835
39836     /**
39837      * @hide 
39838      * @method
39839      */
39840     autoSize: Roo.emptyFn,
39841     // private
39842     monitorTab : true,
39843     // private
39844     deferHeight : true,
39845
39846     
39847     actionMode : 'wrap',
39848     // private
39849     onResize : function(w, h){
39850         Roo.form.TriggerField.superclass.onResize.apply(this, arguments);
39851         if(typeof w == 'number'){
39852             var x = w - this.trigger.getWidth();
39853             this.el.setWidth(this.adjustWidth('input', x));
39854             this.trigger.setStyle('left', x+'px');
39855         }
39856     },
39857
39858     // private
39859     adjustSize : Roo.BoxComponent.prototype.adjustSize,
39860
39861     // private
39862     getResizeEl : function(){
39863         return this.wrap;
39864     },
39865
39866     // private
39867     getPositionEl : function(){
39868         return this.wrap;
39869     },
39870
39871     // private
39872     alignErrorIcon : function(){
39873         this.errorIcon.alignTo(this.wrap, 'tl-tr', [2, 0]);
39874     },
39875
39876     // private
39877     onRender : function(ct, position){
39878         Roo.form.TriggerField.superclass.onRender.call(this, ct, position);
39879         this.wrap = this.el.wrap({cls: "x-form-field-wrap"});
39880         this.trigger = this.wrap.createChild(this.triggerConfig ||
39881                 {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.triggerClass});
39882         if(this.hideTrigger){
39883             this.trigger.setDisplayed(false);
39884         }
39885         this.initTrigger();
39886         if(!this.width){
39887             this.wrap.setWidth(this.el.getWidth()+this.trigger.getWidth());
39888         }
39889     },
39890
39891     // private
39892     initTrigger : function(){
39893         this.trigger.on("click", this.onTriggerClick, this, {preventDefault:true});
39894         this.trigger.addClassOnOver('x-form-trigger-over');
39895         this.trigger.addClassOnClick('x-form-trigger-click');
39896     },
39897
39898     // private
39899     onDestroy : function(){
39900         if(this.trigger){
39901             this.trigger.removeAllListeners();
39902             this.trigger.remove();
39903         }
39904         if(this.wrap){
39905             this.wrap.remove();
39906         }
39907         Roo.form.TriggerField.superclass.onDestroy.call(this);
39908     },
39909
39910     // private
39911     onFocus : function(){
39912         Roo.form.TriggerField.superclass.onFocus.call(this);
39913         if(!this.mimicing){
39914             this.wrap.addClass('x-trigger-wrap-focus');
39915             this.mimicing = true;
39916             Roo.get(Roo.isIE ? document.body : document).on("mousedown", this.mimicBlur, this);
39917             if(this.monitorTab){
39918                 this.el.on("keydown", this.checkTab, this);
39919             }
39920         }
39921     },
39922
39923     // private
39924     checkTab : function(e){
39925         if(e.getKey() == e.TAB){
39926             this.triggerBlur();
39927         }
39928     },
39929
39930     // private
39931     onBlur : function(){
39932         // do nothing
39933     },
39934
39935     // private
39936     mimicBlur : function(e, t){
39937         if(!this.wrap.contains(t) && this.validateBlur()){
39938             this.triggerBlur();
39939         }
39940     },
39941
39942     // private
39943     triggerBlur : function(){
39944         this.mimicing = false;
39945         Roo.get(Roo.isIE ? document.body : document).un("mousedown", this.mimicBlur);
39946         if(this.monitorTab){
39947             this.el.un("keydown", this.checkTab, this);
39948         }
39949         this.wrap.removeClass('x-trigger-wrap-focus');
39950         Roo.form.TriggerField.superclass.onBlur.call(this);
39951     },
39952
39953     // private
39954     // This should be overriden by any subclass that needs to check whether or not the field can be blurred.
39955     validateBlur : function(e, t){
39956         return true;
39957     },
39958
39959     // private
39960     onDisable : function(){
39961         Roo.form.TriggerField.superclass.onDisable.call(this);
39962         if(this.wrap){
39963             this.wrap.addClass('x-item-disabled');
39964         }
39965     },
39966
39967     // private
39968     onEnable : function(){
39969         Roo.form.TriggerField.superclass.onEnable.call(this);
39970         if(this.wrap){
39971             this.wrap.removeClass('x-item-disabled');
39972         }
39973     },
39974
39975     // private
39976     onShow : function(){
39977         var ae = this.getActionEl();
39978         
39979         if(ae){
39980             ae.dom.style.display = '';
39981             ae.dom.style.visibility = 'visible';
39982         }
39983     },
39984
39985     // private
39986     
39987     onHide : function(){
39988         var ae = this.getActionEl();
39989         ae.dom.style.display = 'none';
39990     },
39991
39992     /**
39993      * The function that should handle the trigger's click event.  This method does nothing by default until overridden
39994      * by an implementing function.
39995      * @method
39996      * @param {EventObject} e
39997      */
39998     onTriggerClick : Roo.emptyFn
39999 });
40000
40001 // TwinTriggerField is not a public class to be used directly.  It is meant as an abstract base class
40002 // to be extended by an implementing class.  For an example of implementing this class, see the custom
40003 // SearchField implementation here: http://extjs.com/deploy/ext/examples/form/custom.html
40004 Roo.form.TwinTriggerField = Roo.extend(Roo.form.TriggerField, {
40005     initComponent : function(){
40006         Roo.form.TwinTriggerField.superclass.initComponent.call(this);
40007
40008         this.triggerConfig = {
40009             tag:'span', cls:'x-form-twin-triggers', cn:[
40010             {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.trigger1Class},
40011             {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.trigger2Class}
40012         ]};
40013     },
40014
40015     getTrigger : function(index){
40016         return this.triggers[index];
40017     },
40018
40019     initTrigger : function(){
40020         var ts = this.trigger.select('.x-form-trigger', true);
40021         this.wrap.setStyle('overflow', 'hidden');
40022         var triggerField = this;
40023         ts.each(function(t, all, index){
40024             t.hide = function(){
40025                 var w = triggerField.wrap.getWidth();
40026                 this.dom.style.display = 'none';
40027                 triggerField.el.setWidth(w-triggerField.trigger.getWidth());
40028             };
40029             t.show = function(){
40030                 var w = triggerField.wrap.getWidth();
40031                 this.dom.style.display = '';
40032                 triggerField.el.setWidth(w-triggerField.trigger.getWidth());
40033             };
40034             var triggerIndex = 'Trigger'+(index+1);
40035
40036             if(this['hide'+triggerIndex]){
40037                 t.dom.style.display = 'none';
40038             }
40039             t.on("click", this['on'+triggerIndex+'Click'], this, {preventDefault:true});
40040             t.addClassOnOver('x-form-trigger-over');
40041             t.addClassOnClick('x-form-trigger-click');
40042         }, this);
40043         this.triggers = ts.elements;
40044     },
40045
40046     onTrigger1Click : Roo.emptyFn,
40047     onTrigger2Click : Roo.emptyFn
40048 });/*
40049  * Based on:
40050  * Ext JS Library 1.1.1
40051  * Copyright(c) 2006-2007, Ext JS, LLC.
40052  *
40053  * Originally Released Under LGPL - original licence link has changed is not relivant.
40054  *
40055  * Fork - LGPL
40056  * <script type="text/javascript">
40057  */
40058  
40059 /**
40060  * @class Roo.form.TextArea
40061  * @extends Roo.form.TextField
40062  * Multiline text field.  Can be used as a direct replacement for traditional textarea fields, plus adds
40063  * support for auto-sizing.
40064  * @constructor
40065  * Creates a new TextArea
40066  * @param {Object} config Configuration options
40067  */
40068 Roo.form.TextArea = function(config){
40069     Roo.form.TextArea.superclass.constructor.call(this, config);
40070     // these are provided exchanges for backwards compat
40071     // minHeight/maxHeight were replaced by growMin/growMax to be
40072     // compatible with TextField growing config values
40073     if(this.minHeight !== undefined){
40074         this.growMin = this.minHeight;
40075     }
40076     if(this.maxHeight !== undefined){
40077         this.growMax = this.maxHeight;
40078     }
40079 };
40080
40081 Roo.extend(Roo.form.TextArea, Roo.form.TextField,  {
40082     /**
40083      * @cfg {Number} growMin The minimum height to allow when grow = true (defaults to 60)
40084      */
40085     growMin : 60,
40086     /**
40087      * @cfg {Number} growMax The maximum height to allow when grow = true (defaults to 1000)
40088      */
40089     growMax: 1000,
40090     /**
40091      * @cfg {Boolean} preventScrollbars True to prevent scrollbars from appearing regardless of how much text is
40092      * in the field (equivalent to setting overflow: hidden, defaults to false)
40093      */
40094     preventScrollbars: false,
40095     /**
40096      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
40097      * {tag: "textarea", style: "width:300px;height:60px;", autocomplete: "off"})
40098      */
40099
40100     // private
40101     onRender : function(ct, position){
40102         if(!this.el){
40103             this.defaultAutoCreate = {
40104                 tag: "textarea",
40105                 style:"width:300px;height:60px;",
40106                 autocomplete: "new-password"
40107             };
40108         }
40109         Roo.form.TextArea.superclass.onRender.call(this, ct, position);
40110         if(this.grow){
40111             this.textSizeEl = Roo.DomHelper.append(document.body, {
40112                 tag: "pre", cls: "x-form-grow-sizer"
40113             });
40114             if(this.preventScrollbars){
40115                 this.el.setStyle("overflow", "hidden");
40116             }
40117             this.el.setHeight(this.growMin);
40118         }
40119     },
40120
40121     onDestroy : function(){
40122         if(this.textSizeEl){
40123             this.textSizeEl.parentNode.removeChild(this.textSizeEl);
40124         }
40125         Roo.form.TextArea.superclass.onDestroy.call(this);
40126     },
40127
40128     // private
40129     onKeyUp : function(e){
40130         if(!e.isNavKeyPress() || e.getKey() == e.ENTER){
40131             this.autoSize();
40132         }
40133     },
40134
40135     /**
40136      * Automatically grows the field to accomodate the height of the text up to the maximum field height allowed.
40137      * This only takes effect if grow = true, and fires the autosize event if the height changes.
40138      */
40139     autoSize : function(){
40140         if(!this.grow || !this.textSizeEl){
40141             return;
40142         }
40143         var el = this.el;
40144         var v = el.dom.value;
40145         var ts = this.textSizeEl;
40146
40147         ts.innerHTML = '';
40148         ts.appendChild(document.createTextNode(v));
40149         v = ts.innerHTML;
40150
40151         Roo.fly(ts).setWidth(this.el.getWidth());
40152         if(v.length < 1){
40153             v = "&#160;&#160;";
40154         }else{
40155             if(Roo.isIE){
40156                 v = v.replace(/\n/g, '<p>&#160;</p>');
40157             }
40158             v += "&#160;\n&#160;";
40159         }
40160         ts.innerHTML = v;
40161         var h = Math.min(this.growMax, Math.max(ts.offsetHeight, this.growMin));
40162         if(h != this.lastHeight){
40163             this.lastHeight = h;
40164             this.el.setHeight(h);
40165             this.fireEvent("autosize", this, h);
40166         }
40167     }
40168 });/*
40169  * Based on:
40170  * Ext JS Library 1.1.1
40171  * Copyright(c) 2006-2007, Ext JS, LLC.
40172  *
40173  * Originally Released Under LGPL - original licence link has changed is not relivant.
40174  *
40175  * Fork - LGPL
40176  * <script type="text/javascript">
40177  */
40178  
40179
40180 /**
40181  * @class Roo.form.NumberField
40182  * @extends Roo.form.TextField
40183  * Numeric text field that provides automatic keystroke filtering and numeric validation.
40184  * @constructor
40185  * Creates a new NumberField
40186  * @param {Object} config Configuration options
40187  */
40188 Roo.form.NumberField = function(config){
40189     Roo.form.NumberField.superclass.constructor.call(this, config);
40190 };
40191
40192 Roo.extend(Roo.form.NumberField, Roo.form.TextField,  {
40193     /**
40194      * @cfg {String} fieldClass The default CSS class for the field (defaults to "x-form-field x-form-num-field")
40195      */
40196     fieldClass: "x-form-field x-form-num-field",
40197     /**
40198      * @cfg {Boolean} allowDecimals False to disallow decimal values (defaults to true)
40199      */
40200     allowDecimals : true,
40201     /**
40202      * @cfg {String} decimalSeparator Character(s) to allow as the decimal separator (defaults to '.')
40203      */
40204     decimalSeparator : ".",
40205     /**
40206      * @cfg {Number} decimalPrecision The maximum precision to display after the decimal separator (defaults to 2)
40207      */
40208     decimalPrecision : 2,
40209     /**
40210      * @cfg {Boolean} allowNegative False to prevent entering a negative sign (defaults to true)
40211      */
40212     allowNegative : true,
40213     /**
40214      * @cfg {Number} minValue The minimum allowed value (defaults to Number.NEGATIVE_INFINITY)
40215      */
40216     minValue : Number.NEGATIVE_INFINITY,
40217     /**
40218      * @cfg {Number} maxValue The maximum allowed value (defaults to Number.MAX_VALUE)
40219      */
40220     maxValue : Number.MAX_VALUE,
40221     /**
40222      * @cfg {String} minText Error text to display if the minimum value validation fails (defaults to "The minimum value for this field is {minValue}")
40223      */
40224     minText : "The minimum value for this field is {0}",
40225     /**
40226      * @cfg {String} maxText Error text to display if the maximum value validation fails (defaults to "The maximum value for this field is {maxValue}")
40227      */
40228     maxText : "The maximum value for this field is {0}",
40229     /**
40230      * @cfg {String} nanText Error text to display if the value is not a valid number.  For example, this can happen
40231      * if a valid character like '.' or '-' is left in the field with no number (defaults to "{value} is not a valid number")
40232      */
40233     nanText : "{0} is not a valid number",
40234
40235     // private
40236     initEvents : function(){
40237         Roo.form.NumberField.superclass.initEvents.call(this);
40238         var allowed = "0123456789";
40239         if(this.allowDecimals){
40240             allowed += this.decimalSeparator;
40241         }
40242         if(this.allowNegative){
40243             allowed += "-";
40244         }
40245         this.stripCharsRe = new RegExp('[^'+allowed+']', 'gi');
40246         var keyPress = function(e){
40247             var k = e.getKey();
40248             if(!Roo.isIE && (e.isSpecialKey() || k == e.BACKSPACE || k == e.DELETE)){
40249                 return;
40250             }
40251             var c = e.getCharCode();
40252             if(allowed.indexOf(String.fromCharCode(c)) === -1){
40253                 e.stopEvent();
40254             }
40255         };
40256         this.el.on("keypress", keyPress, this);
40257     },
40258
40259     // private
40260     validateValue : function(value){
40261         if(!Roo.form.NumberField.superclass.validateValue.call(this, value)){
40262             return false;
40263         }
40264         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40265              return true;
40266         }
40267         var num = this.parseValue(value);
40268         if(isNaN(num)){
40269             this.markInvalid(String.format(this.nanText, value));
40270             return false;
40271         }
40272         if(num < this.minValue){
40273             this.markInvalid(String.format(this.minText, this.minValue));
40274             return false;
40275         }
40276         if(num > this.maxValue){
40277             this.markInvalid(String.format(this.maxText, this.maxValue));
40278             return false;
40279         }
40280         return true;
40281     },
40282
40283     getValue : function(){
40284         return this.fixPrecision(this.parseValue(Roo.form.NumberField.superclass.getValue.call(this)));
40285     },
40286
40287     // private
40288     parseValue : function(value){
40289         value = parseFloat(String(value).replace(this.decimalSeparator, "."));
40290         return isNaN(value) ? '' : value;
40291     },
40292
40293     // private
40294     fixPrecision : function(value){
40295         var nan = isNaN(value);
40296         if(!this.allowDecimals || this.decimalPrecision == -1 || nan || !value){
40297             return nan ? '' : value;
40298         }
40299         return parseFloat(value).toFixed(this.decimalPrecision);
40300     },
40301
40302     setValue : function(v){
40303         v = this.fixPrecision(v);
40304         Roo.form.NumberField.superclass.setValue.call(this, String(v).replace(".", this.decimalSeparator));
40305     },
40306
40307     // private
40308     decimalPrecisionFcn : function(v){
40309         return Math.floor(v);
40310     },
40311
40312     beforeBlur : function(){
40313         var v = this.parseValue(this.getRawValue());
40314         if(v){
40315             this.setValue(v);
40316         }
40317     }
40318 });/*
40319  * Based on:
40320  * Ext JS Library 1.1.1
40321  * Copyright(c) 2006-2007, Ext JS, LLC.
40322  *
40323  * Originally Released Under LGPL - original licence link has changed is not relivant.
40324  *
40325  * Fork - LGPL
40326  * <script type="text/javascript">
40327  */
40328  
40329 /**
40330  * @class Roo.form.DateField
40331  * @extends Roo.form.TriggerField
40332  * Provides a date input field with a {@link Roo.DatePicker} dropdown and automatic date validation.
40333 * @constructor
40334 * Create a new DateField
40335 * @param {Object} config
40336  */
40337 Roo.form.DateField = function(config)
40338 {
40339     Roo.form.DateField.superclass.constructor.call(this, config);
40340     
40341       this.addEvents({
40342          
40343         /**
40344          * @event select
40345          * Fires when a date is selected
40346              * @param {Roo.form.DateField} combo This combo box
40347              * @param {Date} date The date selected
40348              */
40349         'select' : true
40350          
40351     });
40352     
40353     
40354     if(typeof this.minValue == "string") {
40355         this.minValue = this.parseDate(this.minValue);
40356     }
40357     if(typeof this.maxValue == "string") {
40358         this.maxValue = this.parseDate(this.maxValue);
40359     }
40360     this.ddMatch = null;
40361     if(this.disabledDates){
40362         var dd = this.disabledDates;
40363         var re = "(?:";
40364         for(var i = 0; i < dd.length; i++){
40365             re += dd[i];
40366             if(i != dd.length-1) {
40367                 re += "|";
40368             }
40369         }
40370         this.ddMatch = new RegExp(re + ")");
40371     }
40372 };
40373
40374 Roo.extend(Roo.form.DateField, Roo.form.TriggerField,  {
40375     /**
40376      * @cfg {String} format
40377      * The default date format string which can be overriden for localization support.  The format must be
40378      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
40379      */
40380     format : "m/d/y",
40381     /**
40382      * @cfg {String} altFormats
40383      * Multiple date formats separated by "|" to try when parsing a user input value and it doesn't match the defined
40384      * format (defaults to 'm/d/Y|m-d-y|m-d-Y|m/d|m-d|d').
40385      */
40386     altFormats : "m/d/Y|m-d-y|m-d-Y|m/d|m-d|md|mdy|mdY|d",
40387     /**
40388      * @cfg {Array} disabledDays
40389      * An array of days to disable, 0 based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
40390      */
40391     disabledDays : null,
40392     /**
40393      * @cfg {String} disabledDaysText
40394      * The tooltip to display when the date falls on a disabled day (defaults to 'Disabled')
40395      */
40396     disabledDaysText : "Disabled",
40397     /**
40398      * @cfg {Array} disabledDates
40399      * An array of "dates" to disable, as strings. These strings will be used to build a dynamic regular
40400      * expression so they are very powerful. Some examples:
40401      * <ul>
40402      * <li>["03/08/2003", "09/16/2003"] would disable those exact dates</li>
40403      * <li>["03/08", "09/16"] would disable those days for every year</li>
40404      * <li>["^03/08"] would only match the beginning (useful if you are using short years)</li>
40405      * <li>["03/../2006"] would disable every day in March 2006</li>
40406      * <li>["^03"] would disable every day in every March</li>
40407      * </ul>
40408      * In order to support regular expressions, if you are using a date format that has "." in it, you will have to
40409      * escape the dot when restricting dates. For example: ["03\\.08\\.03"].
40410      */
40411     disabledDates : null,
40412     /**
40413      * @cfg {String} disabledDatesText
40414      * The tooltip text to display when the date falls on a disabled date (defaults to 'Disabled')
40415      */
40416     disabledDatesText : "Disabled",
40417     /**
40418      * @cfg {Date/String} minValue
40419      * The minimum allowed date. Can be either a Javascript date object or a string date in a
40420      * valid format (defaults to null).
40421      */
40422     minValue : null,
40423     /**
40424      * @cfg {Date/String} maxValue
40425      * The maximum allowed date. Can be either a Javascript date object or a string date in a
40426      * valid format (defaults to null).
40427      */
40428     maxValue : null,
40429     /**
40430      * @cfg {String} minText
40431      * The error text to display when the date in the cell is before minValue (defaults to
40432      * 'The date in this field must be after {minValue}').
40433      */
40434     minText : "The date in this field must be equal to or after {0}",
40435     /**
40436      * @cfg {String} maxText
40437      * The error text to display when the date in the cell is after maxValue (defaults to
40438      * 'The date in this field must be before {maxValue}').
40439      */
40440     maxText : "The date in this field must be equal to or before {0}",
40441     /**
40442      * @cfg {String} invalidText
40443      * The error text to display when the date in the field is invalid (defaults to
40444      * '{value} is not a valid date - it must be in the format {format}').
40445      */
40446     invalidText : "{0} is not a valid date - it must be in the format {1}",
40447     /**
40448      * @cfg {String} triggerClass
40449      * An additional CSS class used to style the trigger button.  The trigger will always get the
40450      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-date-trigger'
40451      * which displays a calendar icon).
40452      */
40453     triggerClass : 'x-form-date-trigger',
40454     
40455
40456     /**
40457      * @cfg {Boolean} useIso
40458      * if enabled, then the date field will use a hidden field to store the 
40459      * real value as iso formated date. default (false)
40460      */ 
40461     useIso : false,
40462     /**
40463      * @cfg {String/Object} autoCreate
40464      * A DomHelper element spec, or true for a default element spec (defaults to
40465      * {tag: "input", type: "text", size: "10", autocomplete: "off"})
40466      */ 
40467     // private
40468     defaultAutoCreate : {tag: "input", type: "text", size: "10", autocomplete: "off"},
40469     
40470     // private
40471     hiddenField: false,
40472     
40473     onRender : function(ct, position)
40474     {
40475         Roo.form.DateField.superclass.onRender.call(this, ct, position);
40476         if (this.useIso) {
40477             //this.el.dom.removeAttribute('name'); 
40478             Roo.log("Changing name?");
40479             this.el.dom.setAttribute('name', this.name + '____hidden___' ); 
40480             this.hiddenField = this.el.insertSibling({ tag:'input', type:'hidden', name: this.name },
40481                     'before', true);
40482             this.hiddenField.value = this.value ? this.formatDate(this.value, 'Y-m-d') : '';
40483             // prevent input submission
40484             this.hiddenName = this.name;
40485         }
40486             
40487             
40488     },
40489     
40490     // private
40491     validateValue : function(value)
40492     {
40493         value = this.formatDate(value);
40494         if(!Roo.form.DateField.superclass.validateValue.call(this, value)){
40495             Roo.log('super failed');
40496             return false;
40497         }
40498         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40499              return true;
40500         }
40501         var svalue = value;
40502         value = this.parseDate(value);
40503         if(!value){
40504             Roo.log('parse date failed' + svalue);
40505             this.markInvalid(String.format(this.invalidText, svalue, this.format));
40506             return false;
40507         }
40508         var time = value.getTime();
40509         if(this.minValue && time < this.minValue.getTime()){
40510             this.markInvalid(String.format(this.minText, this.formatDate(this.minValue)));
40511             return false;
40512         }
40513         if(this.maxValue && time > this.maxValue.getTime()){
40514             this.markInvalid(String.format(this.maxText, this.formatDate(this.maxValue)));
40515             return false;
40516         }
40517         if(this.disabledDays){
40518             var day = value.getDay();
40519             for(var i = 0; i < this.disabledDays.length; i++) {
40520                 if(day === this.disabledDays[i]){
40521                     this.markInvalid(this.disabledDaysText);
40522                     return false;
40523                 }
40524             }
40525         }
40526         var fvalue = this.formatDate(value);
40527         if(this.ddMatch && this.ddMatch.test(fvalue)){
40528             this.markInvalid(String.format(this.disabledDatesText, fvalue));
40529             return false;
40530         }
40531         return true;
40532     },
40533
40534     // private
40535     // Provides logic to override the default TriggerField.validateBlur which just returns true
40536     validateBlur : function(){
40537         return !this.menu || !this.menu.isVisible();
40538     },
40539     
40540     getName: function()
40541     {
40542         // returns hidden if it's set..
40543         if (!this.rendered) {return ''};
40544         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
40545         
40546     },
40547
40548     /**
40549      * Returns the current date value of the date field.
40550      * @return {Date} The date value
40551      */
40552     getValue : function(){
40553         
40554         return  this.hiddenField ?
40555                 this.hiddenField.value :
40556                 this.parseDate(Roo.form.DateField.superclass.getValue.call(this)) || "";
40557     },
40558
40559     /**
40560      * Sets the value of the date field.  You can pass a date object or any string that can be parsed into a valid
40561      * date, using DateField.format as the date format, according to the same rules as {@link Date#parseDate}
40562      * (the default format used is "m/d/y").
40563      * <br />Usage:
40564      * <pre><code>
40565 //All of these calls set the same date value (May 4, 2006)
40566
40567 //Pass a date object:
40568 var dt = new Date('5/4/06');
40569 dateField.setValue(dt);
40570
40571 //Pass a date string (default format):
40572 dateField.setValue('5/4/06');
40573
40574 //Pass a date string (custom format):
40575 dateField.format = 'Y-m-d';
40576 dateField.setValue('2006-5-4');
40577 </code></pre>
40578      * @param {String/Date} date The date or valid date string
40579      */
40580     setValue : function(date){
40581         if (this.hiddenField) {
40582             this.hiddenField.value = this.formatDate(this.parseDate(date), 'Y-m-d');
40583         }
40584         Roo.form.DateField.superclass.setValue.call(this, this.formatDate(this.parseDate(date)));
40585         // make sure the value field is always stored as a date..
40586         this.value = this.parseDate(date);
40587         
40588         
40589     },
40590
40591     // private
40592     parseDate : function(value){
40593         if(!value || value instanceof Date){
40594             return value;
40595         }
40596         var v = Date.parseDate(value, this.format);
40597          if (!v && this.useIso) {
40598             v = Date.parseDate(value, 'Y-m-d');
40599         }
40600         if(!v && this.altFormats){
40601             if(!this.altFormatsArray){
40602                 this.altFormatsArray = this.altFormats.split("|");
40603             }
40604             for(var i = 0, len = this.altFormatsArray.length; i < len && !v; i++){
40605                 v = Date.parseDate(value, this.altFormatsArray[i]);
40606             }
40607         }
40608         return v;
40609     },
40610
40611     // private
40612     formatDate : function(date, fmt){
40613         return (!date || !(date instanceof Date)) ?
40614                date : date.dateFormat(fmt || this.format);
40615     },
40616
40617     // private
40618     menuListeners : {
40619         select: function(m, d){
40620             
40621             this.setValue(d);
40622             this.fireEvent('select', this, d);
40623         },
40624         show : function(){ // retain focus styling
40625             this.onFocus();
40626         },
40627         hide : function(){
40628             this.focus.defer(10, this);
40629             var ml = this.menuListeners;
40630             this.menu.un("select", ml.select,  this);
40631             this.menu.un("show", ml.show,  this);
40632             this.menu.un("hide", ml.hide,  this);
40633         }
40634     },
40635
40636     // private
40637     // Implements the default empty TriggerField.onTriggerClick function to display the DatePicker
40638     onTriggerClick : function(){
40639         if(this.disabled){
40640             return;
40641         }
40642         if(this.menu == null){
40643             this.menu = new Roo.menu.DateMenu();
40644         }
40645         Roo.apply(this.menu.picker,  {
40646             showClear: this.allowBlank,
40647             minDate : this.minValue,
40648             maxDate : this.maxValue,
40649             disabledDatesRE : this.ddMatch,
40650             disabledDatesText : this.disabledDatesText,
40651             disabledDays : this.disabledDays,
40652             disabledDaysText : this.disabledDaysText,
40653             format : this.useIso ? 'Y-m-d' : this.format,
40654             minText : String.format(this.minText, this.formatDate(this.minValue)),
40655             maxText : String.format(this.maxText, this.formatDate(this.maxValue))
40656         });
40657         this.menu.on(Roo.apply({}, this.menuListeners, {
40658             scope:this
40659         }));
40660         this.menu.picker.setValue(this.getValue() || new Date());
40661         this.menu.show(this.el, "tl-bl?");
40662     },
40663
40664     beforeBlur : function(){
40665         var v = this.parseDate(this.getRawValue());
40666         if(v){
40667             this.setValue(v);
40668         }
40669     },
40670
40671     /*@
40672      * overide
40673      * 
40674      */
40675     isDirty : function() {
40676         if(this.disabled) {
40677             return false;
40678         }
40679         
40680         if(typeof(this.startValue) === 'undefined'){
40681             return false;
40682         }
40683         
40684         return String(this.getValue()) !== String(this.startValue);
40685         
40686     },
40687     // @overide
40688     cleanLeadingSpace : function(e)
40689     {
40690        return;
40691     }
40692     
40693 });/*
40694  * Based on:
40695  * Ext JS Library 1.1.1
40696  * Copyright(c) 2006-2007, Ext JS, LLC.
40697  *
40698  * Originally Released Under LGPL - original licence link has changed is not relivant.
40699  *
40700  * Fork - LGPL
40701  * <script type="text/javascript">
40702  */
40703  
40704 /**
40705  * @class Roo.form.MonthField
40706  * @extends Roo.form.TriggerField
40707  * Provides a date input field with a {@link Roo.DatePicker} dropdown and automatic date validation.
40708 * @constructor
40709 * Create a new MonthField
40710 * @param {Object} config
40711  */
40712 Roo.form.MonthField = function(config){
40713     
40714     Roo.form.MonthField.superclass.constructor.call(this, config);
40715     
40716       this.addEvents({
40717          
40718         /**
40719          * @event select
40720          * Fires when a date is selected
40721              * @param {Roo.form.MonthFieeld} combo This combo box
40722              * @param {Date} date The date selected
40723              */
40724         'select' : true
40725          
40726     });
40727     
40728     
40729     if(typeof this.minValue == "string") {
40730         this.minValue = this.parseDate(this.minValue);
40731     }
40732     if(typeof this.maxValue == "string") {
40733         this.maxValue = this.parseDate(this.maxValue);
40734     }
40735     this.ddMatch = null;
40736     if(this.disabledDates){
40737         var dd = this.disabledDates;
40738         var re = "(?:";
40739         for(var i = 0; i < dd.length; i++){
40740             re += dd[i];
40741             if(i != dd.length-1) {
40742                 re += "|";
40743             }
40744         }
40745         this.ddMatch = new RegExp(re + ")");
40746     }
40747 };
40748
40749 Roo.extend(Roo.form.MonthField, Roo.form.TriggerField,  {
40750     /**
40751      * @cfg {String} format
40752      * The default date format string which can be overriden for localization support.  The format must be
40753      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
40754      */
40755     format : "M Y",
40756     /**
40757      * @cfg {String} altFormats
40758      * Multiple date formats separated by "|" to try when parsing a user input value and it doesn't match the defined
40759      * format (defaults to 'm/d/Y|m-d-y|m-d-Y|m/d|m-d|d').
40760      */
40761     altFormats : "M Y|m/Y|m-y|m-Y|my|mY",
40762     /**
40763      * @cfg {Array} disabledDays
40764      * An array of days to disable, 0 based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
40765      */
40766     disabledDays : [0,1,2,3,4,5,6],
40767     /**
40768      * @cfg {String} disabledDaysText
40769      * The tooltip to display when the date falls on a disabled day (defaults to 'Disabled')
40770      */
40771     disabledDaysText : "Disabled",
40772     /**
40773      * @cfg {Array} disabledDates
40774      * An array of "dates" to disable, as strings. These strings will be used to build a dynamic regular
40775      * expression so they are very powerful. Some examples:
40776      * <ul>
40777      * <li>["03/08/2003", "09/16/2003"] would disable those exact dates</li>
40778      * <li>["03/08", "09/16"] would disable those days for every year</li>
40779      * <li>["^03/08"] would only match the beginning (useful if you are using short years)</li>
40780      * <li>["03/../2006"] would disable every day in March 2006</li>
40781      * <li>["^03"] would disable every day in every March</li>
40782      * </ul>
40783      * In order to support regular expressions, if you are using a date format that has "." in it, you will have to
40784      * escape the dot when restricting dates. For example: ["03\\.08\\.03"].
40785      */
40786     disabledDates : null,
40787     /**
40788      * @cfg {String} disabledDatesText
40789      * The tooltip text to display when the date falls on a disabled date (defaults to 'Disabled')
40790      */
40791     disabledDatesText : "Disabled",
40792     /**
40793      * @cfg {Date/String} minValue
40794      * The minimum allowed date. Can be either a Javascript date object or a string date in a
40795      * valid format (defaults to null).
40796      */
40797     minValue : null,
40798     /**
40799      * @cfg {Date/String} maxValue
40800      * The maximum allowed date. Can be either a Javascript date object or a string date in a
40801      * valid format (defaults to null).
40802      */
40803     maxValue : null,
40804     /**
40805      * @cfg {String} minText
40806      * The error text to display when the date in the cell is before minValue (defaults to
40807      * 'The date in this field must be after {minValue}').
40808      */
40809     minText : "The date in this field must be equal to or after {0}",
40810     /**
40811      * @cfg {String} maxTextf
40812      * The error text to display when the date in the cell is after maxValue (defaults to
40813      * 'The date in this field must be before {maxValue}').
40814      */
40815     maxText : "The date in this field must be equal to or before {0}",
40816     /**
40817      * @cfg {String} invalidText
40818      * The error text to display when the date in the field is invalid (defaults to
40819      * '{value} is not a valid date - it must be in the format {format}').
40820      */
40821     invalidText : "{0} is not a valid date - it must be in the format {1}",
40822     /**
40823      * @cfg {String} triggerClass
40824      * An additional CSS class used to style the trigger button.  The trigger will always get the
40825      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-date-trigger'
40826      * which displays a calendar icon).
40827      */
40828     triggerClass : 'x-form-date-trigger',
40829     
40830
40831     /**
40832      * @cfg {Boolean} useIso
40833      * if enabled, then the date field will use a hidden field to store the 
40834      * real value as iso formated date. default (true)
40835      */ 
40836     useIso : true,
40837     /**
40838      * @cfg {String/Object} autoCreate
40839      * A DomHelper element spec, or true for a default element spec (defaults to
40840      * {tag: "input", type: "text", size: "10", autocomplete: "off"})
40841      */ 
40842     // private
40843     defaultAutoCreate : {tag: "input", type: "text", size: "10", autocomplete: "new-password"},
40844     
40845     // private
40846     hiddenField: false,
40847     
40848     hideMonthPicker : false,
40849     
40850     onRender : function(ct, position)
40851     {
40852         Roo.form.MonthField.superclass.onRender.call(this, ct, position);
40853         if (this.useIso) {
40854             this.el.dom.removeAttribute('name'); 
40855             this.hiddenField = this.el.insertSibling({ tag:'input', type:'hidden', name: this.name },
40856                     'before', true);
40857             this.hiddenField.value = this.value ? this.formatDate(this.value, 'Y-m-d') : '';
40858             // prevent input submission
40859             this.hiddenName = this.name;
40860         }
40861             
40862             
40863     },
40864     
40865     // private
40866     validateValue : function(value)
40867     {
40868         value = this.formatDate(value);
40869         if(!Roo.form.MonthField.superclass.validateValue.call(this, value)){
40870             return false;
40871         }
40872         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40873              return true;
40874         }
40875         var svalue = value;
40876         value = this.parseDate(value);
40877         if(!value){
40878             this.markInvalid(String.format(this.invalidText, svalue, this.format));
40879             return false;
40880         }
40881         var time = value.getTime();
40882         if(this.minValue && time < this.minValue.getTime()){
40883             this.markInvalid(String.format(this.minText, this.formatDate(this.minValue)));
40884             return false;
40885         }
40886         if(this.maxValue && time > this.maxValue.getTime()){
40887             this.markInvalid(String.format(this.maxText, this.formatDate(this.maxValue)));
40888             return false;
40889         }
40890         /*if(this.disabledDays){
40891             var day = value.getDay();
40892             for(var i = 0; i < this.disabledDays.length; i++) {
40893                 if(day === this.disabledDays[i]){
40894                     this.markInvalid(this.disabledDaysText);
40895                     return false;
40896                 }
40897             }
40898         }
40899         */
40900         var fvalue = this.formatDate(value);
40901         /*if(this.ddMatch && this.ddMatch.test(fvalue)){
40902             this.markInvalid(String.format(this.disabledDatesText, fvalue));
40903             return false;
40904         }
40905         */
40906         return true;
40907     },
40908
40909     // private
40910     // Provides logic to override the default TriggerField.validateBlur which just returns true
40911     validateBlur : function(){
40912         return !this.menu || !this.menu.isVisible();
40913     },
40914
40915     /**
40916      * Returns the current date value of the date field.
40917      * @return {Date} The date value
40918      */
40919     getValue : function(){
40920         
40921         
40922         
40923         return  this.hiddenField ?
40924                 this.hiddenField.value :
40925                 this.parseDate(Roo.form.MonthField.superclass.getValue.call(this)) || "";
40926     },
40927
40928     /**
40929      * Sets the value of the date field.  You can pass a date object or any string that can be parsed into a valid
40930      * date, using MonthField.format as the date format, according to the same rules as {@link Date#parseDate}
40931      * (the default format used is "m/d/y").
40932      * <br />Usage:
40933      * <pre><code>
40934 //All of these calls set the same date value (May 4, 2006)
40935
40936 //Pass a date object:
40937 var dt = new Date('5/4/06');
40938 monthField.setValue(dt);
40939
40940 //Pass a date string (default format):
40941 monthField.setValue('5/4/06');
40942
40943 //Pass a date string (custom format):
40944 monthField.format = 'Y-m-d';
40945 monthField.setValue('2006-5-4');
40946 </code></pre>
40947      * @param {String/Date} date The date or valid date string
40948      */
40949     setValue : function(date){
40950         Roo.log('month setValue' + date);
40951         // can only be first of month..
40952         
40953         var val = this.parseDate(date);
40954         
40955         if (this.hiddenField) {
40956             this.hiddenField.value = this.formatDate(this.parseDate(date), 'Y-m-d');
40957         }
40958         Roo.form.MonthField.superclass.setValue.call(this, this.formatDate(this.parseDate(date)));
40959         this.value = this.parseDate(date);
40960     },
40961
40962     // private
40963     parseDate : function(value){
40964         if(!value || value instanceof Date){
40965             value = value ? Date.parseDate(value.format('Y-m') + '-01', 'Y-m-d') : null;
40966             return value;
40967         }
40968         var v = Date.parseDate(value, this.format);
40969         if (!v && this.useIso) {
40970             v = Date.parseDate(value, 'Y-m-d');
40971         }
40972         if (v) {
40973             // 
40974             v = Date.parseDate(v.format('Y-m') +'-01', 'Y-m-d');
40975         }
40976         
40977         
40978         if(!v && this.altFormats){
40979             if(!this.altFormatsArray){
40980                 this.altFormatsArray = this.altFormats.split("|");
40981             }
40982             for(var i = 0, len = this.altFormatsArray.length; i < len && !v; i++){
40983                 v = Date.parseDate(value, this.altFormatsArray[i]);
40984             }
40985         }
40986         return v;
40987     },
40988
40989     // private
40990     formatDate : function(date, fmt){
40991         return (!date || !(date instanceof Date)) ?
40992                date : date.dateFormat(fmt || this.format);
40993     },
40994
40995     // private
40996     menuListeners : {
40997         select: function(m, d){
40998             this.setValue(d);
40999             this.fireEvent('select', this, d);
41000         },
41001         show : function(){ // retain focus styling
41002             this.onFocus();
41003         },
41004         hide : function(){
41005             this.focus.defer(10, this);
41006             var ml = this.menuListeners;
41007             this.menu.un("select", ml.select,  this);
41008             this.menu.un("show", ml.show,  this);
41009             this.menu.un("hide", ml.hide,  this);
41010         }
41011     },
41012     // private
41013     // Implements the default empty TriggerField.onTriggerClick function to display the DatePicker
41014     onTriggerClick : function(){
41015         if(this.disabled){
41016             return;
41017         }
41018         if(this.menu == null){
41019             this.menu = new Roo.menu.DateMenu();
41020            
41021         }
41022         
41023         Roo.apply(this.menu.picker,  {
41024             
41025             showClear: this.allowBlank,
41026             minDate : this.minValue,
41027             maxDate : this.maxValue,
41028             disabledDatesRE : this.ddMatch,
41029             disabledDatesText : this.disabledDatesText,
41030             
41031             format : this.useIso ? 'Y-m-d' : this.format,
41032             minText : String.format(this.minText, this.formatDate(this.minValue)),
41033             maxText : String.format(this.maxText, this.formatDate(this.maxValue))
41034             
41035         });
41036          this.menu.on(Roo.apply({}, this.menuListeners, {
41037             scope:this
41038         }));
41039        
41040         
41041         var m = this.menu;
41042         var p = m.picker;
41043         
41044         // hide month picker get's called when we called by 'before hide';
41045         
41046         var ignorehide = true;
41047         p.hideMonthPicker  = function(disableAnim){
41048             if (ignorehide) {
41049                 return;
41050             }
41051              if(this.monthPicker){
41052                 Roo.log("hideMonthPicker called");
41053                 if(disableAnim === true){
41054                     this.monthPicker.hide();
41055                 }else{
41056                     this.monthPicker.slideOut('t', {duration:.2});
41057                     p.setValue(new Date(m.picker.mpSelYear, m.picker.mpSelMonth, 1));
41058                     p.fireEvent("select", this, this.value);
41059                     m.hide();
41060                 }
41061             }
41062         }
41063         
41064         Roo.log('picker set value');
41065         Roo.log(this.getValue());
41066         p.setValue(this.getValue() ? this.parseDate(this.getValue()) : new Date());
41067         m.show(this.el, 'tl-bl?');
41068         ignorehide  = false;
41069         // this will trigger hideMonthPicker..
41070         
41071         
41072         // hidden the day picker
41073         Roo.select('.x-date-picker table', true).first().dom.style.visibility = "hidden";
41074         
41075         
41076         
41077       
41078         
41079         p.showMonthPicker.defer(100, p);
41080     
41081         
41082        
41083     },
41084
41085     beforeBlur : function(){
41086         var v = this.parseDate(this.getRawValue());
41087         if(v){
41088             this.setValue(v);
41089         }
41090     }
41091
41092     /** @cfg {Boolean} grow @hide */
41093     /** @cfg {Number} growMin @hide */
41094     /** @cfg {Number} growMax @hide */
41095     /**
41096      * @hide
41097      * @method autoSize
41098      */
41099 });/*
41100  * Based on:
41101  * Ext JS Library 1.1.1
41102  * Copyright(c) 2006-2007, Ext JS, LLC.
41103  *
41104  * Originally Released Under LGPL - original licence link has changed is not relivant.
41105  *
41106  * Fork - LGPL
41107  * <script type="text/javascript">
41108  */
41109  
41110
41111 /**
41112  * @class Roo.form.ComboBox
41113  * @extends Roo.form.TriggerField
41114  * A combobox control with support for autocomplete, remote-loading, paging and many other features.
41115  * @constructor
41116  * Create a new ComboBox.
41117  * @param {Object} config Configuration options
41118  */
41119 Roo.form.ComboBox = function(config){
41120     Roo.form.ComboBox.superclass.constructor.call(this, config);
41121     this.addEvents({
41122         /**
41123          * @event expand
41124          * Fires when the dropdown list is expanded
41125              * @param {Roo.form.ComboBox} combo This combo box
41126              */
41127         'expand' : true,
41128         /**
41129          * @event collapse
41130          * Fires when the dropdown list is collapsed
41131              * @param {Roo.form.ComboBox} combo This combo box
41132              */
41133         'collapse' : true,
41134         /**
41135          * @event beforeselect
41136          * Fires before a list item is selected. Return false to cancel the selection.
41137              * @param {Roo.form.ComboBox} combo This combo box
41138              * @param {Roo.data.Record} record The data record returned from the underlying store
41139              * @param {Number} index The index of the selected item in the dropdown list
41140              */
41141         'beforeselect' : true,
41142         /**
41143          * @event select
41144          * Fires when a list item is selected
41145              * @param {Roo.form.ComboBox} combo This combo box
41146              * @param {Roo.data.Record} record The data record returned from the underlying store (or false on clear)
41147              * @param {Number} index The index of the selected item in the dropdown list
41148              */
41149         'select' : true,
41150         /**
41151          * @event beforequery
41152          * Fires before all queries are processed. Return false to cancel the query or set cancel to true.
41153          * The event object passed has these properties:
41154              * @param {Roo.form.ComboBox} combo This combo box
41155              * @param {String} query The query
41156              * @param {Boolean} forceAll true to force "all" query
41157              * @param {Boolean} cancel true to cancel the query
41158              * @param {Object} e The query event object
41159              */
41160         'beforequery': true,
41161          /**
41162          * @event add
41163          * Fires when the 'add' icon is pressed (add a listener to enable add button)
41164              * @param {Roo.form.ComboBox} combo This combo box
41165              */
41166         'add' : true,
41167         /**
41168          * @event edit
41169          * Fires when the 'edit' icon is pressed (add a listener to enable add button)
41170              * @param {Roo.form.ComboBox} combo This combo box
41171              * @param {Roo.data.Record|false} record The data record returned from the underlying store (or false on nothing selected)
41172              */
41173         'edit' : true
41174         
41175         
41176     });
41177     if(this.transform){
41178         this.allowDomMove = false;
41179         var s = Roo.getDom(this.transform);
41180         if(!this.hiddenName){
41181             this.hiddenName = s.name;
41182         }
41183         if(!this.store){
41184             this.mode = 'local';
41185             var d = [], opts = s.options;
41186             for(var i = 0, len = opts.length;i < len; i++){
41187                 var o = opts[i];
41188                 var value = (Roo.isIE ? o.getAttributeNode('value').specified : o.hasAttribute('value')) ? o.value : o.text;
41189                 if(o.selected) {
41190                     this.value = value;
41191                 }
41192                 d.push([value, o.text]);
41193             }
41194             this.store = new Roo.data.SimpleStore({
41195                 'id': 0,
41196                 fields: ['value', 'text'],
41197                 data : d
41198             });
41199             this.valueField = 'value';
41200             this.displayField = 'text';
41201         }
41202         s.name = Roo.id(); // wipe out the name in case somewhere else they have a reference
41203         if(!this.lazyRender){
41204             this.target = true;
41205             this.el = Roo.DomHelper.insertBefore(s, this.autoCreate || this.defaultAutoCreate);
41206             s.parentNode.removeChild(s); // remove it
41207             this.render(this.el.parentNode);
41208         }else{
41209             s.parentNode.removeChild(s); // remove it
41210         }
41211
41212     }
41213     if (this.store) {
41214         this.store = Roo.factory(this.store, Roo.data);
41215     }
41216     
41217     this.selectedIndex = -1;
41218     if(this.mode == 'local'){
41219         if(config.queryDelay === undefined){
41220             this.queryDelay = 10;
41221         }
41222         if(config.minChars === undefined){
41223             this.minChars = 0;
41224         }
41225     }
41226 };
41227
41228 Roo.extend(Roo.form.ComboBox, Roo.form.TriggerField, {
41229     /**
41230      * @cfg {String/HTMLElement/Element} transform The id, DOM node or element of an existing select to convert to a ComboBox
41231      */
41232     /**
41233      * @cfg {Boolean} lazyRender True to prevent the ComboBox from rendering until requested (should always be used when
41234      * rendering into an Roo.Editor, defaults to false)
41235      */
41236     /**
41237      * @cfg {Boolean/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to:
41238      * {tag: "input", type: "text", size: "24", autocomplete: "off"})
41239      */
41240     /**
41241      * @cfg {Roo.data.Store} store The data store to which this combo is bound (defaults to undefined)
41242      */
41243     /**
41244      * @cfg {String} title If supplied, a header element is created containing this text and added into the top of
41245      * the dropdown list (defaults to undefined, with no header element)
41246      */
41247
41248      /**
41249      * @cfg {String/Roo.Template} tpl The template to use to render the output
41250      */
41251      
41252     // private
41253     defaultAutoCreate : {tag: "input", type: "text", size: "24", autocomplete: "off"},
41254     /**
41255      * @cfg {Number} listWidth The width in pixels of the dropdown list (defaults to the width of the ComboBox field)
41256      */
41257     listWidth: undefined,
41258     /**
41259      * @cfg {String} displayField The underlying data field name to bind to this CombBox (defaults to undefined if
41260      * mode = 'remote' or 'text' if mode = 'local')
41261      */
41262     displayField: undefined,
41263     /**
41264      * @cfg {String} valueField The underlying data value name to bind to this CombBox (defaults to undefined if
41265      * mode = 'remote' or 'value' if mode = 'local'). 
41266      * Note: use of a valueField requires the user make a selection
41267      * in order for a value to be mapped.
41268      */
41269     valueField: undefined,
41270     
41271     
41272     /**
41273      * @cfg {String} hiddenName If specified, a hidden form field with this name is dynamically generated to store the
41274      * field's data value (defaults to the underlying DOM element's name)
41275      */
41276     hiddenName: undefined,
41277     /**
41278      * @cfg {String} listClass CSS class to apply to the dropdown list element (defaults to '')
41279      */
41280     listClass: '',
41281     /**
41282      * @cfg {String} selectedClass CSS class to apply to the selected item in the dropdown list (defaults to 'x-combo-selected')
41283      */
41284     selectedClass: 'x-combo-selected',
41285     /**
41286      * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
41287      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-arrow-trigger'
41288      * which displays a downward arrow icon).
41289      */
41290     triggerClass : 'x-form-arrow-trigger',
41291     /**
41292      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
41293      */
41294     shadow:'sides',
41295     /**
41296      * @cfg {String} listAlign A valid anchor position value. See {@link Roo.Element#alignTo} for details on supported
41297      * anchor positions (defaults to 'tl-bl')
41298      */
41299     listAlign: 'tl-bl?',
41300     /**
41301      * @cfg {Number} maxHeight The maximum height in pixels of the dropdown list before scrollbars are shown (defaults to 300)
41302      */
41303     maxHeight: 300,
41304     /**
41305      * @cfg {String} triggerAction The action to execute when the trigger field is activated.  Use 'all' to run the
41306      * query specified by the allQuery config option (defaults to 'query')
41307      */
41308     triggerAction: 'query',
41309     /**
41310      * @cfg {Number} minChars The minimum number of characters the user must type before autocomplete and typeahead activate
41311      * (defaults to 4, does not apply if editable = false)
41312      */
41313     minChars : 4,
41314     /**
41315      * @cfg {Boolean} typeAhead True to populate and autoselect the remainder of the text being typed after a configurable
41316      * delay (typeAheadDelay) if it matches a known value (defaults to false)
41317      */
41318     typeAhead: false,
41319     /**
41320      * @cfg {Number} queryDelay The length of time in milliseconds to delay between the start of typing and sending the
41321      * query to filter the dropdown list (defaults to 500 if mode = 'remote' or 10 if mode = 'local')
41322      */
41323     queryDelay: 500,
41324     /**
41325      * @cfg {Number} pageSize If greater than 0, a paging toolbar is displayed in the footer of the dropdown list and the
41326      * filter queries will execute with page start and limit parameters.  Only applies when mode = 'remote' (defaults to 0)
41327      */
41328     pageSize: 0,
41329     /**
41330      * @cfg {Boolean} selectOnFocus True to select any existing text in the field immediately on focus.  Only applies
41331      * when editable = true (defaults to false)
41332      */
41333     selectOnFocus:false,
41334     /**
41335      * @cfg {String} queryParam Name of the query as it will be passed on the querystring (defaults to 'query')
41336      */
41337     queryParam: 'query',
41338     /**
41339      * @cfg {String} loadingText The text to display in the dropdown list while data is loading.  Only applies
41340      * when mode = 'remote' (defaults to 'Loading...')
41341      */
41342     loadingText: 'Loading...',
41343     /**
41344      * @cfg {Boolean} resizable True to add a resize handle to the bottom of the dropdown list (defaults to false)
41345      */
41346     resizable: false,
41347     /**
41348      * @cfg {Number} handleHeight The height in pixels of the dropdown list resize handle if resizable = true (defaults to 8)
41349      */
41350     handleHeight : 8,
41351     /**
41352      * @cfg {Boolean} editable False to prevent the user from typing text directly into the field, just like a
41353      * traditional select (defaults to true)
41354      */
41355     editable: true,
41356     /**
41357      * @cfg {String} allQuery The text query to send to the server to return all records for the list with no filtering (defaults to '')
41358      */
41359     allQuery: '',
41360     /**
41361      * @cfg {String} mode Set to 'local' if the ComboBox loads local data (defaults to 'remote' which loads from the server)
41362      */
41363     mode: 'remote',
41364     /**
41365      * @cfg {Number} minListWidth The minimum width of the dropdown list in pixels (defaults to 70, will be ignored if
41366      * listWidth has a higher value)
41367      */
41368     minListWidth : 70,
41369     /**
41370      * @cfg {Boolean} forceSelection True to restrict the selected value to one of the values in the list, false to
41371      * allow the user to set arbitrary text into the field (defaults to false)
41372      */
41373     forceSelection:false,
41374     /**
41375      * @cfg {Number} typeAheadDelay The length of time in milliseconds to wait until the typeahead text is displayed
41376      * if typeAhead = true (defaults to 250)
41377      */
41378     typeAheadDelay : 250,
41379     /**
41380      * @cfg {String} valueNotFoundText When using a name/value combo, if the value passed to setValue is not found in
41381      * the store, valueNotFoundText will be displayed as the field text if defined (defaults to undefined)
41382      */
41383     valueNotFoundText : undefined,
41384     /**
41385      * @cfg {Boolean} blockFocus Prevents all focus calls, so it can work with things like HTML edtor bar
41386      */
41387     blockFocus : false,
41388     
41389     /**
41390      * @cfg {Boolean} disableClear Disable showing of clear button.
41391      */
41392     disableClear : false,
41393     /**
41394      * @cfg {Boolean} alwaysQuery  Disable caching of results, and always send query
41395      */
41396     alwaysQuery : false,
41397     
41398     //private
41399     addicon : false,
41400     editicon: false,
41401     
41402     // element that contains real text value.. (when hidden is used..)
41403      
41404     // private
41405     onRender : function(ct, position)
41406     {
41407         Roo.form.ComboBox.superclass.onRender.call(this, ct, position);
41408         
41409         if(this.hiddenName){
41410             this.hiddenField = this.el.insertSibling({tag:'input', type:'hidden', name: this.hiddenName, id:  (this.hiddenId||this.hiddenName)},
41411                     'before', true);
41412             this.hiddenField.value =
41413                 this.hiddenValue !== undefined ? this.hiddenValue :
41414                 this.value !== undefined ? this.value : '';
41415
41416             // prevent input submission
41417             this.el.dom.removeAttribute('name');
41418              
41419              
41420         }
41421         
41422         if(Roo.isGecko){
41423             this.el.dom.setAttribute('autocomplete', 'off');
41424         }
41425
41426         var cls = 'x-combo-list';
41427
41428         this.list = new Roo.Layer({
41429             shadow: this.shadow, cls: [cls, this.listClass].join(' '), constrain:false
41430         });
41431
41432         var lw = this.listWidth || Math.max(this.wrap.getWidth(), this.minListWidth);
41433         this.list.setWidth(lw);
41434         this.list.swallowEvent('mousewheel');
41435         this.assetHeight = 0;
41436
41437         if(this.title){
41438             this.header = this.list.createChild({cls:cls+'-hd', html: this.title});
41439             this.assetHeight += this.header.getHeight();
41440         }
41441
41442         this.innerList = this.list.createChild({cls:cls+'-inner'});
41443         this.innerList.on('mouseover', this.onViewOver, this);
41444         this.innerList.on('mousemove', this.onViewMove, this);
41445         this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
41446         
41447         if(this.allowBlank && !this.pageSize && !this.disableClear){
41448             this.footer = this.list.createChild({cls:cls+'-ft'});
41449             this.pageTb = new Roo.Toolbar(this.footer);
41450            
41451         }
41452         if(this.pageSize){
41453             this.footer = this.list.createChild({cls:cls+'-ft'});
41454             this.pageTb = new Roo.PagingToolbar(this.footer, this.store,
41455                     {pageSize: this.pageSize});
41456             
41457         }
41458         
41459         if (this.pageTb && this.allowBlank && !this.disableClear) {
41460             var _this = this;
41461             this.pageTb.add(new Roo.Toolbar.Fill(), {
41462                 cls: 'x-btn-icon x-btn-clear',
41463                 text: '&#160;',
41464                 handler: function()
41465                 {
41466                     _this.collapse();
41467                     _this.clearValue();
41468                     _this.onSelect(false, -1);
41469                 }
41470             });
41471         }
41472         if (this.footer) {
41473             this.assetHeight += this.footer.getHeight();
41474         }
41475         
41476
41477         if(!this.tpl){
41478             this.tpl = '<div class="'+cls+'-item">{' + this.displayField + '}</div>';
41479         }
41480
41481         this.view = new Roo.View(this.innerList, this.tpl, {
41482             singleSelect:true,
41483             store: this.store,
41484             selectedClass: this.selectedClass
41485         });
41486
41487         this.view.on('click', this.onViewClick, this);
41488
41489         this.store.on('beforeload', this.onBeforeLoad, this);
41490         this.store.on('load', this.onLoad, this);
41491         this.store.on('loadexception', this.onLoadException, this);
41492
41493         if(this.resizable){
41494             this.resizer = new Roo.Resizable(this.list,  {
41495                pinned:true, handles:'se'
41496             });
41497             this.resizer.on('resize', function(r, w, h){
41498                 this.maxHeight = h-this.handleHeight-this.list.getFrameWidth('tb')-this.assetHeight;
41499                 this.listWidth = w;
41500                 this.innerList.setWidth(w - this.list.getFrameWidth('lr'));
41501                 this.restrictHeight();
41502             }, this);
41503             this[this.pageSize?'footer':'innerList'].setStyle('margin-bottom', this.handleHeight+'px');
41504         }
41505         if(!this.editable){
41506             this.editable = true;
41507             this.setEditable(false);
41508         }  
41509         
41510         
41511         if (typeof(this.events.add.listeners) != 'undefined') {
41512             
41513             this.addicon = this.wrap.createChild(
41514                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-combo-add' });  
41515        
41516             this.addicon.on('click', function(e) {
41517                 this.fireEvent('add', this);
41518             }, this);
41519         }
41520         if (typeof(this.events.edit.listeners) != 'undefined') {
41521             
41522             this.editicon = this.wrap.createChild(
41523                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-combo-edit' });  
41524             if (this.addicon) {
41525                 this.editicon.setStyle('margin-left', '40px');
41526             }
41527             this.editicon.on('click', function(e) {
41528                 
41529                 // we fire even  if inothing is selected..
41530                 this.fireEvent('edit', this, this.lastData );
41531                 
41532             }, this);
41533         }
41534         
41535         
41536         
41537     },
41538
41539     // private
41540     initEvents : function(){
41541         Roo.form.ComboBox.superclass.initEvents.call(this);
41542
41543         this.keyNav = new Roo.KeyNav(this.el, {
41544             "up" : function(e){
41545                 this.inKeyMode = true;
41546                 this.selectPrev();
41547             },
41548
41549             "down" : function(e){
41550                 if(!this.isExpanded()){
41551                     this.onTriggerClick();
41552                 }else{
41553                     this.inKeyMode = true;
41554                     this.selectNext();
41555                 }
41556             },
41557
41558             "enter" : function(e){
41559                 this.onViewClick();
41560                 //return true;
41561             },
41562
41563             "esc" : function(e){
41564                 this.collapse();
41565             },
41566
41567             "tab" : function(e){
41568                 this.onViewClick(false);
41569                 this.fireEvent("specialkey", this, e);
41570                 return true;
41571             },
41572
41573             scope : this,
41574
41575             doRelay : function(foo, bar, hname){
41576                 if(hname == 'down' || this.scope.isExpanded()){
41577                    return Roo.KeyNav.prototype.doRelay.apply(this, arguments);
41578                 }
41579                 return true;
41580             },
41581
41582             forceKeyDown: true
41583         });
41584         this.queryDelay = Math.max(this.queryDelay || 10,
41585                 this.mode == 'local' ? 10 : 250);
41586         this.dqTask = new Roo.util.DelayedTask(this.initQuery, this);
41587         if(this.typeAhead){
41588             this.taTask = new Roo.util.DelayedTask(this.onTypeAhead, this);
41589         }
41590         if(this.editable !== false){
41591             this.el.on("keyup", this.onKeyUp, this);
41592         }
41593         if(this.forceSelection){
41594             this.on('blur', this.doForce, this);
41595         }
41596     },
41597
41598     onDestroy : function(){
41599         if(this.view){
41600             this.view.setStore(null);
41601             this.view.el.removeAllListeners();
41602             this.view.el.remove();
41603             this.view.purgeListeners();
41604         }
41605         if(this.list){
41606             this.list.destroy();
41607         }
41608         if(this.store){
41609             this.store.un('beforeload', this.onBeforeLoad, this);
41610             this.store.un('load', this.onLoad, this);
41611             this.store.un('loadexception', this.onLoadException, this);
41612         }
41613         Roo.form.ComboBox.superclass.onDestroy.call(this);
41614     },
41615
41616     // private
41617     fireKey : function(e){
41618         if(e.isNavKeyPress() && !this.list.isVisible()){
41619             this.fireEvent("specialkey", this, e);
41620         }
41621     },
41622
41623     // private
41624     onResize: function(w, h){
41625         Roo.form.ComboBox.superclass.onResize.apply(this, arguments);
41626         
41627         if(typeof w != 'number'){
41628             // we do not handle it!?!?
41629             return;
41630         }
41631         var tw = this.trigger.getWidth();
41632         tw += this.addicon ? this.addicon.getWidth() : 0;
41633         tw += this.editicon ? this.editicon.getWidth() : 0;
41634         var x = w - tw;
41635         this.el.setWidth( this.adjustWidth('input', x));
41636             
41637         this.trigger.setStyle('left', x+'px');
41638         
41639         if(this.list && this.listWidth === undefined){
41640             var lw = Math.max(x + this.trigger.getWidth(), this.minListWidth);
41641             this.list.setWidth(lw);
41642             this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
41643         }
41644         
41645     
41646         
41647     },
41648
41649     /**
41650      * Allow or prevent the user from directly editing the field text.  If false is passed,
41651      * the user will only be able to select from the items defined in the dropdown list.  This method
41652      * is the runtime equivalent of setting the 'editable' config option at config time.
41653      * @param {Boolean} value True to allow the user to directly edit the field text
41654      */
41655     setEditable : function(value){
41656         if(value == this.editable){
41657             return;
41658         }
41659         this.editable = value;
41660         if(!value){
41661             this.el.dom.setAttribute('readOnly', true);
41662             this.el.on('mousedown', this.onTriggerClick,  this);
41663             this.el.addClass('x-combo-noedit');
41664         }else{
41665             this.el.dom.setAttribute('readOnly', false);
41666             this.el.un('mousedown', this.onTriggerClick,  this);
41667             this.el.removeClass('x-combo-noedit');
41668         }
41669     },
41670
41671     // private
41672     onBeforeLoad : function(){
41673         if(!this.hasFocus){
41674             return;
41675         }
41676         this.innerList.update(this.loadingText ?
41677                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
41678         this.restrictHeight();
41679         this.selectedIndex = -1;
41680     },
41681
41682     // private
41683     onLoad : function(){
41684         if(!this.hasFocus){
41685             return;
41686         }
41687         if(this.store.getCount() > 0){
41688             this.expand();
41689             this.restrictHeight();
41690             if(this.lastQuery == this.allQuery){
41691                 if(this.editable){
41692                     this.el.dom.select();
41693                 }
41694                 if(!this.selectByValue(this.value, true)){
41695                     this.select(0, true);
41696                 }
41697             }else{
41698                 this.selectNext();
41699                 if(this.typeAhead && this.lastKey != Roo.EventObject.BACKSPACE && this.lastKey != Roo.EventObject.DELETE){
41700                     this.taTask.delay(this.typeAheadDelay);
41701                 }
41702             }
41703         }else{
41704             this.onEmptyResults();
41705         }
41706         //this.el.focus();
41707     },
41708     // private
41709     onLoadException : function()
41710     {
41711         this.collapse();
41712         Roo.log(this.store.reader.jsonData);
41713         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
41714             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
41715         }
41716         
41717         
41718     },
41719     // private
41720     onTypeAhead : function(){
41721         if(this.store.getCount() > 0){
41722             var r = this.store.getAt(0);
41723             var newValue = r.data[this.displayField];
41724             var len = newValue.length;
41725             var selStart = this.getRawValue().length;
41726             if(selStart != len){
41727                 this.setRawValue(newValue);
41728                 this.selectText(selStart, newValue.length);
41729             }
41730         }
41731     },
41732
41733     // private
41734     onSelect : function(record, index){
41735         if(this.fireEvent('beforeselect', this, record, index) !== false){
41736             this.setFromData(index > -1 ? record.data : false);
41737             this.collapse();
41738             this.fireEvent('select', this, record, index);
41739         }
41740     },
41741
41742     /**
41743      * Returns the currently selected field value or empty string if no value is set.
41744      * @return {String} value The selected value
41745      */
41746     getValue : function(){
41747         if(this.valueField){
41748             return typeof this.value != 'undefined' ? this.value : '';
41749         }
41750         return Roo.form.ComboBox.superclass.getValue.call(this);
41751     },
41752
41753     /**
41754      * Clears any text/value currently set in the field
41755      */
41756     clearValue : function(){
41757         if(this.hiddenField){
41758             this.hiddenField.value = '';
41759         }
41760         this.value = '';
41761         this.setRawValue('');
41762         this.lastSelectionText = '';
41763         
41764     },
41765
41766     /**
41767      * Sets the specified value into the field.  If the value finds a match, the corresponding record text
41768      * will be displayed in the field.  If the value does not match the data value of an existing item,
41769      * and the valueNotFoundText config option is defined, it will be displayed as the default field text.
41770      * Otherwise the field will be blank (although the value will still be set).
41771      * @param {String} value The value to match
41772      */
41773     setValue : function(v){
41774         var text = v;
41775         if(this.valueField){
41776             var r = this.findRecord(this.valueField, v);
41777             if(r){
41778                 text = r.data[this.displayField];
41779             }else if(this.valueNotFoundText !== undefined){
41780                 text = this.valueNotFoundText;
41781             }
41782         }
41783         this.lastSelectionText = text;
41784         if(this.hiddenField){
41785             this.hiddenField.value = v;
41786         }
41787         Roo.form.ComboBox.superclass.setValue.call(this, text);
41788         this.value = v;
41789     },
41790     /**
41791      * @property {Object} the last set data for the element
41792      */
41793     
41794     lastData : false,
41795     /**
41796      * Sets the value of the field based on a object which is related to the record format for the store.
41797      * @param {Object} value the value to set as. or false on reset?
41798      */
41799     setFromData : function(o){
41800         var dv = ''; // display value
41801         var vv = ''; // value value..
41802         this.lastData = o;
41803         if (this.displayField) {
41804             dv = !o || typeof(o[this.displayField]) == 'undefined' ? '' : o[this.displayField];
41805         } else {
41806             // this is an error condition!!!
41807             Roo.log('no  displayField value set for '+ (this.name ? this.name : this.id));
41808         }
41809         
41810         if(this.valueField){
41811             vv = !o || typeof(o[this.valueField]) == 'undefined' ? dv : o[this.valueField];
41812         }
41813         if(this.hiddenField){
41814             this.hiddenField.value = vv;
41815             
41816             this.lastSelectionText = dv;
41817             Roo.form.ComboBox.superclass.setValue.call(this, dv);
41818             this.value = vv;
41819             return;
41820         }
41821         // no hidden field.. - we store the value in 'value', but still display
41822         // display field!!!!
41823         this.lastSelectionText = dv;
41824         Roo.form.ComboBox.superclass.setValue.call(this, dv);
41825         this.value = vv;
41826         
41827         
41828     },
41829     // private
41830     reset : function(){
41831         // overridden so that last data is reset..
41832         this.setValue(this.resetValue);
41833         this.originalValue = this.getValue();
41834         this.clearInvalid();
41835         this.lastData = false;
41836         if (this.view) {
41837             this.view.clearSelections();
41838         }
41839     },
41840     // private
41841     findRecord : function(prop, value){
41842         var record;
41843         if(this.store.getCount() > 0){
41844             this.store.each(function(r){
41845                 if(r.data[prop] == value){
41846                     record = r;
41847                     return false;
41848                 }
41849                 return true;
41850             });
41851         }
41852         return record;
41853     },
41854     
41855     getName: function()
41856     {
41857         // returns hidden if it's set..
41858         if (!this.rendered) {return ''};
41859         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
41860         
41861     },
41862     // private
41863     onViewMove : function(e, t){
41864         this.inKeyMode = false;
41865     },
41866
41867     // private
41868     onViewOver : function(e, t){
41869         if(this.inKeyMode){ // prevent key nav and mouse over conflicts
41870             return;
41871         }
41872         var item = this.view.findItemFromChild(t);
41873         if(item){
41874             var index = this.view.indexOf(item);
41875             this.select(index, false);
41876         }
41877     },
41878
41879     // private
41880     onViewClick : function(doFocus)
41881     {
41882         var index = this.view.getSelectedIndexes()[0];
41883         var r = this.store.getAt(index);
41884         if(r){
41885             this.onSelect(r, index);
41886         }
41887         if(doFocus !== false && !this.blockFocus){
41888             this.el.focus();
41889         }
41890     },
41891
41892     // private
41893     restrictHeight : function(){
41894         this.innerList.dom.style.height = '';
41895         var inner = this.innerList.dom;
41896         var h = Math.max(inner.clientHeight, inner.offsetHeight, inner.scrollHeight);
41897         this.innerList.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
41898         this.list.beginUpdate();
41899         this.list.setHeight(this.innerList.getHeight()+this.list.getFrameWidth('tb')+(this.resizable?this.handleHeight:0)+this.assetHeight);
41900         this.list.alignTo(this.el, this.listAlign);
41901         this.list.endUpdate();
41902     },
41903
41904     // private
41905     onEmptyResults : function(){
41906         this.collapse();
41907     },
41908
41909     /**
41910      * Returns true if the dropdown list is expanded, else false.
41911      */
41912     isExpanded : function(){
41913         return this.list.isVisible();
41914     },
41915
41916     /**
41917      * Select an item in the dropdown list by its data value. This function does NOT cause the select event to fire.
41918      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
41919      * @param {String} value The data value of the item to select
41920      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
41921      * selected item if it is not currently in view (defaults to true)
41922      * @return {Boolean} True if the value matched an item in the list, else false
41923      */
41924     selectByValue : function(v, scrollIntoView){
41925         if(v !== undefined && v !== null){
41926             var r = this.findRecord(this.valueField || this.displayField, v);
41927             if(r){
41928                 this.select(this.store.indexOf(r), scrollIntoView);
41929                 return true;
41930             }
41931         }
41932         return false;
41933     },
41934
41935     /**
41936      * Select an item in the dropdown list by its numeric index in the list. This function does NOT cause the select event to fire.
41937      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
41938      * @param {Number} index The zero-based index of the list item to select
41939      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
41940      * selected item if it is not currently in view (defaults to true)
41941      */
41942     select : function(index, scrollIntoView){
41943         this.selectedIndex = index;
41944         this.view.select(index);
41945         if(scrollIntoView !== false){
41946             var el = this.view.getNode(index);
41947             if(el){
41948                 this.innerList.scrollChildIntoView(el, false);
41949             }
41950         }
41951     },
41952
41953     // private
41954     selectNext : function(){
41955         var ct = this.store.getCount();
41956         if(ct > 0){
41957             if(this.selectedIndex == -1){
41958                 this.select(0);
41959             }else if(this.selectedIndex < ct-1){
41960                 this.select(this.selectedIndex+1);
41961             }
41962         }
41963     },
41964
41965     // private
41966     selectPrev : function(){
41967         var ct = this.store.getCount();
41968         if(ct > 0){
41969             if(this.selectedIndex == -1){
41970                 this.select(0);
41971             }else if(this.selectedIndex != 0){
41972                 this.select(this.selectedIndex-1);
41973             }
41974         }
41975     },
41976
41977     // private
41978     onKeyUp : function(e){
41979         if(this.editable !== false && !e.isSpecialKey()){
41980             this.lastKey = e.getKey();
41981             this.dqTask.delay(this.queryDelay);
41982         }
41983     },
41984
41985     // private
41986     validateBlur : function(){
41987         return !this.list || !this.list.isVisible();   
41988     },
41989
41990     // private
41991     initQuery : function(){
41992         this.doQuery(this.getRawValue());
41993     },
41994
41995     // private
41996     doForce : function(){
41997         if(this.el.dom.value.length > 0){
41998             this.el.dom.value =
41999                 this.lastSelectionText === undefined ? '' : this.lastSelectionText;
42000              
42001         }
42002     },
42003
42004     /**
42005      * Execute a query to filter the dropdown list.  Fires the beforequery event prior to performing the
42006      * query allowing the query action to be canceled if needed.
42007      * @param {String} query The SQL query to execute
42008      * @param {Boolean} forceAll True to force the query to execute even if there are currently fewer characters
42009      * in the field than the minimum specified by the minChars config option.  It also clears any filter previously
42010      * saved in the current store (defaults to false)
42011      */
42012     doQuery : function(q, forceAll){
42013         if(q === undefined || q === null){
42014             q = '';
42015         }
42016         var qe = {
42017             query: q,
42018             forceAll: forceAll,
42019             combo: this,
42020             cancel:false
42021         };
42022         if(this.fireEvent('beforequery', qe)===false || qe.cancel){
42023             return false;
42024         }
42025         q = qe.query;
42026         forceAll = qe.forceAll;
42027         if(forceAll === true || (q.length >= this.minChars)){
42028             if(this.lastQuery != q || this.alwaysQuery){
42029                 this.lastQuery = q;
42030                 if(this.mode == 'local'){
42031                     this.selectedIndex = -1;
42032                     if(forceAll){
42033                         this.store.clearFilter();
42034                     }else{
42035                         this.store.filter(this.displayField, q);
42036                     }
42037                     this.onLoad();
42038                 }else{
42039                     this.store.baseParams[this.queryParam] = q;
42040                     this.store.load({
42041                         params: this.getParams(q)
42042                     });
42043                     this.expand();
42044                 }
42045             }else{
42046                 this.selectedIndex = -1;
42047                 this.onLoad();   
42048             }
42049         }
42050     },
42051
42052     // private
42053     getParams : function(q){
42054         var p = {};
42055         //p[this.queryParam] = q;
42056         if(this.pageSize){
42057             p.start = 0;
42058             p.limit = this.pageSize;
42059         }
42060         return p;
42061     },
42062
42063     /**
42064      * Hides the dropdown list if it is currently expanded. Fires the 'collapse' event on completion.
42065      */
42066     collapse : function(){
42067         if(!this.isExpanded()){
42068             return;
42069         }
42070         this.list.hide();
42071         Roo.get(document).un('mousedown', this.collapseIf, this);
42072         Roo.get(document).un('mousewheel', this.collapseIf, this);
42073         if (!this.editable) {
42074             Roo.get(document).un('keydown', this.listKeyPress, this);
42075         }
42076         this.fireEvent('collapse', this);
42077     },
42078
42079     // private
42080     collapseIf : function(e){
42081         if(!e.within(this.wrap) && !e.within(this.list)){
42082             this.collapse();
42083         }
42084     },
42085
42086     /**
42087      * Expands the dropdown list if it is currently hidden. Fires the 'expand' event on completion.
42088      */
42089     expand : function(){
42090         if(this.isExpanded() || !this.hasFocus){
42091             return;
42092         }
42093         this.list.alignTo(this.el, this.listAlign);
42094         this.list.show();
42095         Roo.get(document).on('mousedown', this.collapseIf, this);
42096         Roo.get(document).on('mousewheel', this.collapseIf, this);
42097         if (!this.editable) {
42098             Roo.get(document).on('keydown', this.listKeyPress, this);
42099         }
42100         
42101         this.fireEvent('expand', this);
42102     },
42103
42104     // private
42105     // Implements the default empty TriggerField.onTriggerClick function
42106     onTriggerClick : function(){
42107         if(this.disabled){
42108             return;
42109         }
42110         if(this.isExpanded()){
42111             this.collapse();
42112             if (!this.blockFocus) {
42113                 this.el.focus();
42114             }
42115             
42116         }else {
42117             this.hasFocus = true;
42118             if(this.triggerAction == 'all') {
42119                 this.doQuery(this.allQuery, true);
42120             } else {
42121                 this.doQuery(this.getRawValue());
42122             }
42123             if (!this.blockFocus) {
42124                 this.el.focus();
42125             }
42126         }
42127     },
42128     listKeyPress : function(e)
42129     {
42130         //Roo.log('listkeypress');
42131         // scroll to first matching element based on key pres..
42132         if (e.isSpecialKey()) {
42133             return false;
42134         }
42135         var k = String.fromCharCode(e.getKey()).toUpperCase();
42136         //Roo.log(k);
42137         var match  = false;
42138         var csel = this.view.getSelectedNodes();
42139         var cselitem = false;
42140         if (csel.length) {
42141             var ix = this.view.indexOf(csel[0]);
42142             cselitem  = this.store.getAt(ix);
42143             if (!cselitem.get(this.displayField) || cselitem.get(this.displayField).substring(0,1).toUpperCase() != k) {
42144                 cselitem = false;
42145             }
42146             
42147         }
42148         
42149         this.store.each(function(v) { 
42150             if (cselitem) {
42151                 // start at existing selection.
42152                 if (cselitem.id == v.id) {
42153                     cselitem = false;
42154                 }
42155                 return;
42156             }
42157                 
42158             if (v.get(this.displayField) && v.get(this.displayField).substring(0,1).toUpperCase() == k) {
42159                 match = this.store.indexOf(v);
42160                 return false;
42161             }
42162         }, this);
42163         
42164         if (match === false) {
42165             return true; // no more action?
42166         }
42167         // scroll to?
42168         this.view.select(match);
42169         var sn = Roo.get(this.view.getSelectedNodes()[0]);
42170         sn.scrollIntoView(sn.dom.parentNode, false);
42171     } 
42172
42173     /** 
42174     * @cfg {Boolean} grow 
42175     * @hide 
42176     */
42177     /** 
42178     * @cfg {Number} growMin 
42179     * @hide 
42180     */
42181     /** 
42182     * @cfg {Number} growMax 
42183     * @hide 
42184     */
42185     /**
42186      * @hide
42187      * @method autoSize
42188      */
42189 });/*
42190  * Copyright(c) 2010-2012, Roo J Solutions Limited
42191  *
42192  * Licence LGPL
42193  *
42194  */
42195
42196 /**
42197  * @class Roo.form.ComboBoxArray
42198  * @extends Roo.form.TextField
42199  * A facebook style adder... for lists of email / people / countries  etc...
42200  * pick multiple items from a combo box, and shows each one.
42201  *
42202  *  Fred [x]  Brian [x]  [Pick another |v]
42203  *
42204  *
42205  *  For this to work: it needs various extra information
42206  *    - normal combo problay has
42207  *      name, hiddenName
42208  *    + displayField, valueField
42209  *
42210  *    For our purpose...
42211  *
42212  *
42213  *   If we change from 'extends' to wrapping...
42214  *   
42215  *  
42216  *
42217  
42218  
42219  * @constructor
42220  * Create a new ComboBoxArray.
42221  * @param {Object} config Configuration options
42222  */
42223  
42224
42225 Roo.form.ComboBoxArray = function(config)
42226 {
42227     this.addEvents({
42228         /**
42229          * @event beforeremove
42230          * Fires before remove the value from the list
42231              * @param {Roo.form.ComboBoxArray} _self This combo box array
42232              * @param {Roo.form.ComboBoxArray.Item} item removed item
42233              */
42234         'beforeremove' : true,
42235         /**
42236          * @event remove
42237          * Fires when remove the value from the list
42238              * @param {Roo.form.ComboBoxArray} _self This combo box array
42239              * @param {Roo.form.ComboBoxArray.Item} item removed item
42240              */
42241         'remove' : true
42242         
42243         
42244     });
42245     
42246     Roo.form.ComboBoxArray.superclass.constructor.call(this, config);
42247     
42248     this.items = new Roo.util.MixedCollection(false);
42249     
42250     // construct the child combo...
42251     
42252     
42253     
42254     
42255    
42256     
42257 }
42258
42259  
42260 Roo.extend(Roo.form.ComboBoxArray, Roo.form.TextField,
42261
42262     /**
42263      * @cfg {Roo.form.Combo} combo The combo box that is wrapped
42264      */
42265     
42266     lastData : false,
42267     
42268     // behavies liek a hiddne field
42269     inputType:      'hidden',
42270     /**
42271      * @cfg {Number} width The width of the box that displays the selected element
42272      */ 
42273     width:          300,
42274
42275     
42276     
42277     /**
42278      * @cfg {String} name    The name of the visable items on this form (eg. titles not ids)
42279      */
42280     name : false,
42281     /**
42282      * @cfg {String} hiddenName    The hidden name of the field, often contains an comma seperated list of names
42283      */
42284     hiddenName : false,
42285     
42286     
42287     // private the array of items that are displayed..
42288     items  : false,
42289     // private - the hidden field el.
42290     hiddenEl : false,
42291     // private - the filed el..
42292     el : false,
42293     
42294     //validateValue : function() { return true; }, // all values are ok!
42295     //onAddClick: function() { },
42296     
42297     onRender : function(ct, position) 
42298     {
42299         
42300         // create the standard hidden element
42301         //Roo.form.ComboBoxArray.superclass.onRender.call(this, ct, position);
42302         
42303         
42304         // give fake names to child combo;
42305         this.combo.hiddenName = this.hiddenName ? (this.hiddenName+'-subcombo') : this.hiddenName;
42306         this.combo.name = this.name ? (this.name+'-subcombo') : this.name;
42307         
42308         this.combo = Roo.factory(this.combo, Roo.form);
42309         this.combo.onRender(ct, position);
42310         if (typeof(this.combo.width) != 'undefined') {
42311             this.combo.onResize(this.combo.width,0);
42312         }
42313         
42314         this.combo.initEvents();
42315         
42316         // assigned so form know we need to do this..
42317         this.store          = this.combo.store;
42318         this.valueField     = this.combo.valueField;
42319         this.displayField   = this.combo.displayField ;
42320         
42321         
42322         this.combo.wrap.addClass('x-cbarray-grp');
42323         
42324         var cbwrap = this.combo.wrap.createChild(
42325             {tag: 'div', cls: 'x-cbarray-cb'},
42326             this.combo.el.dom
42327         );
42328         
42329              
42330         this.hiddenEl = this.combo.wrap.createChild({
42331             tag: 'input',  type:'hidden' , name: this.hiddenName, value : ''
42332         });
42333         this.el = this.combo.wrap.createChild({
42334             tag: 'input',  type:'hidden' , name: this.name, value : ''
42335         });
42336          //   this.el.dom.removeAttribute("name");
42337         
42338         
42339         this.outerWrap = this.combo.wrap;
42340         this.wrap = cbwrap;
42341         
42342         this.outerWrap.setWidth(this.width);
42343         this.outerWrap.dom.removeChild(this.el.dom);
42344         
42345         this.wrap.dom.appendChild(this.el.dom);
42346         this.outerWrap.dom.removeChild(this.combo.trigger.dom);
42347         this.combo.wrap.dom.appendChild(this.combo.trigger.dom);
42348         
42349         this.combo.trigger.setStyle('position','relative');
42350         this.combo.trigger.setStyle('left', '0px');
42351         this.combo.trigger.setStyle('top', '2px');
42352         
42353         this.combo.el.setStyle('vertical-align', 'text-bottom');
42354         
42355         //this.trigger.setStyle('vertical-align', 'top');
42356         
42357         // this should use the code from combo really... on('add' ....)
42358         if (this.adder) {
42359             
42360         
42361             this.adder = this.outerWrap.createChild(
42362                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-adder', style: 'margin-left:2px'});  
42363             var _t = this;
42364             this.adder.on('click', function(e) {
42365                 _t.fireEvent('adderclick', this, e);
42366             }, _t);
42367         }
42368         //var _t = this;
42369         //this.adder.on('click', this.onAddClick, _t);
42370         
42371         
42372         this.combo.on('select', function(cb, rec, ix) {
42373             this.addItem(rec.data);
42374             
42375             cb.setValue('');
42376             cb.el.dom.value = '';
42377             //cb.lastData = rec.data;
42378             // add to list
42379             
42380         }, this);
42381         
42382         
42383     },
42384     
42385     
42386     getName: function()
42387     {
42388         // returns hidden if it's set..
42389         if (!this.rendered) {return ''};
42390         return  this.hiddenName ? this.hiddenName : this.name;
42391         
42392     },
42393     
42394     
42395     onResize: function(w, h){
42396         
42397         return;
42398         // not sure if this is needed..
42399         //this.combo.onResize(w,h);
42400         
42401         if(typeof w != 'number'){
42402             // we do not handle it!?!?
42403             return;
42404         }
42405         var tw = this.combo.trigger.getWidth();
42406         tw += this.addicon ? this.addicon.getWidth() : 0;
42407         tw += this.editicon ? this.editicon.getWidth() : 0;
42408         var x = w - tw;
42409         this.combo.el.setWidth( this.combo.adjustWidth('input', x));
42410             
42411         this.combo.trigger.setStyle('left', '0px');
42412         
42413         if(this.list && this.listWidth === undefined){
42414             var lw = Math.max(x + this.combo.trigger.getWidth(), this.combo.minListWidth);
42415             this.list.setWidth(lw);
42416             this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
42417         }
42418         
42419     
42420         
42421     },
42422     
42423     addItem: function(rec)
42424     {
42425         var valueField = this.combo.valueField;
42426         var displayField = this.combo.displayField;
42427         
42428         if (this.items.indexOfKey(rec[valueField]) > -1) {
42429             //console.log("GOT " + rec.data.id);
42430             return;
42431         }
42432         
42433         var x = new Roo.form.ComboBoxArray.Item({
42434             //id : rec[this.idField],
42435             data : rec,
42436             displayField : displayField ,
42437             tipField : displayField ,
42438             cb : this
42439         });
42440         // use the 
42441         this.items.add(rec[valueField],x);
42442         // add it before the element..
42443         this.updateHiddenEl();
42444         x.render(this.outerWrap, this.wrap.dom);
42445         // add the image handler..
42446     },
42447     
42448     updateHiddenEl : function()
42449     {
42450         this.validate();
42451         if (!this.hiddenEl) {
42452             return;
42453         }
42454         var ar = [];
42455         var idField = this.combo.valueField;
42456         
42457         this.items.each(function(f) {
42458             ar.push(f.data[idField]);
42459         });
42460         this.hiddenEl.dom.value = ar.join(',');
42461         this.validate();
42462     },
42463     
42464     reset : function()
42465     {
42466         this.items.clear();
42467         
42468         Roo.each(this.outerWrap.select('.x-cbarray-item', true).elements, function(el){
42469            el.remove();
42470         });
42471         
42472         this.el.dom.value = '';
42473         if (this.hiddenEl) {
42474             this.hiddenEl.dom.value = '';
42475         }
42476         
42477     },
42478     getValue: function()
42479     {
42480         return this.hiddenEl ? this.hiddenEl.dom.value : '';
42481     },
42482     setValue: function(v) // not a valid action - must use addItems..
42483     {
42484         
42485         this.reset();
42486          
42487         if (this.store.isLocal && (typeof(v) == 'string')) {
42488             // then we can use the store to find the values..
42489             // comma seperated at present.. this needs to allow JSON based encoding..
42490             this.hiddenEl.value  = v;
42491             var v_ar = [];
42492             Roo.each(v.split(','), function(k) {
42493                 Roo.log("CHECK " + this.valueField + ',' + k);
42494                 var li = this.store.query(this.valueField, k);
42495                 if (!li.length) {
42496                     return;
42497                 }
42498                 var add = {};
42499                 add[this.valueField] = k;
42500                 add[this.displayField] = li.item(0).data[this.displayField];
42501                 
42502                 this.addItem(add);
42503             }, this) 
42504              
42505         }
42506         if (typeof(v) == 'object' ) {
42507             // then let's assume it's an array of objects..
42508             Roo.each(v, function(l) {
42509                 this.addItem(l);
42510             }, this);
42511              
42512         }
42513         
42514         
42515     },
42516     setFromData: function(v)
42517     {
42518         // this recieves an object, if setValues is called.
42519         this.reset();
42520         this.el.dom.value = v[this.displayField];
42521         this.hiddenEl.dom.value = v[this.valueField];
42522         if (typeof(v[this.valueField]) != 'string' || !v[this.valueField].length) {
42523             return;
42524         }
42525         var kv = v[this.valueField];
42526         var dv = v[this.displayField];
42527         kv = typeof(kv) != 'string' ? '' : kv;
42528         dv = typeof(dv) != 'string' ? '' : dv;
42529         
42530         
42531         var keys = kv.split(',');
42532         var display = dv.split(',');
42533         for (var i = 0 ; i < keys.length; i++) {
42534             
42535             add = {};
42536             add[this.valueField] = keys[i];
42537             add[this.displayField] = display[i];
42538             this.addItem(add);
42539         }
42540       
42541         
42542     },
42543     
42544     /**
42545      * Validates the combox array value
42546      * @return {Boolean} True if the value is valid, else false
42547      */
42548     validate : function(){
42549         if(this.disabled || this.validateValue(this.processValue(this.getValue()))){
42550             this.clearInvalid();
42551             return true;
42552         }
42553         return false;
42554     },
42555     
42556     validateValue : function(value){
42557         return Roo.form.ComboBoxArray.superclass.validateValue.call(this, this.getValue());
42558         
42559     },
42560     
42561     /*@
42562      * overide
42563      * 
42564      */
42565     isDirty : function() {
42566         if(this.disabled) {
42567             return false;
42568         }
42569         
42570         try {
42571             var d = Roo.decode(String(this.originalValue));
42572         } catch (e) {
42573             return String(this.getValue()) !== String(this.originalValue);
42574         }
42575         
42576         var originalValue = [];
42577         
42578         for (var i = 0; i < d.length; i++){
42579             originalValue.push(d[i][this.valueField]);
42580         }
42581         
42582         return String(this.getValue()) !== String(originalValue.join(','));
42583         
42584     }
42585     
42586 });
42587
42588
42589
42590 /**
42591  * @class Roo.form.ComboBoxArray.Item
42592  * @extends Roo.BoxComponent
42593  * A selected item in the list
42594  *  Fred [x]  Brian [x]  [Pick another |v]
42595  * 
42596  * @constructor
42597  * Create a new item.
42598  * @param {Object} config Configuration options
42599  */
42600  
42601 Roo.form.ComboBoxArray.Item = function(config) {
42602     config.id = Roo.id();
42603     Roo.form.ComboBoxArray.Item.superclass.constructor.call(this, config);
42604 }
42605
42606 Roo.extend(Roo.form.ComboBoxArray.Item, Roo.BoxComponent, {
42607     data : {},
42608     cb: false,
42609     displayField : false,
42610     tipField : false,
42611     
42612     
42613     defaultAutoCreate : {
42614         tag: 'div',
42615         cls: 'x-cbarray-item',
42616         cn : [ 
42617             { tag: 'div' },
42618             {
42619                 tag: 'img',
42620                 width:16,
42621                 height : 16,
42622                 src : Roo.BLANK_IMAGE_URL ,
42623                 align: 'center'
42624             }
42625         ]
42626         
42627     },
42628     
42629  
42630     onRender : function(ct, position)
42631     {
42632         Roo.form.Field.superclass.onRender.call(this, ct, position);
42633         
42634         if(!this.el){
42635             var cfg = this.getAutoCreate();
42636             this.el = ct.createChild(cfg, position);
42637         }
42638         
42639         this.el.child('img').dom.setAttribute('src', Roo.BLANK_IMAGE_URL);
42640         
42641         this.el.child('div').dom.innerHTML = this.cb.renderer ? 
42642             this.cb.renderer(this.data) :
42643             String.format('{0}',this.data[this.displayField]);
42644         
42645             
42646         this.el.child('div').dom.setAttribute('qtip',
42647                         String.format('{0}',this.data[this.tipField])
42648         );
42649         
42650         this.el.child('img').on('click', this.remove, this);
42651         
42652     },
42653    
42654     remove : function()
42655     {
42656         if(this.cb.disabled){
42657             return;
42658         }
42659         
42660         if(false !== this.cb.fireEvent('beforeremove', this.cb, this)){
42661             this.cb.items.remove(this);
42662             this.el.child('img').un('click', this.remove, this);
42663             this.el.remove();
42664             this.cb.updateHiddenEl();
42665
42666             this.cb.fireEvent('remove', this.cb, this);
42667         }
42668         
42669     }
42670 });/*
42671  * RooJS Library 1.1.1
42672  * Copyright(c) 2008-2011  Alan Knowles
42673  *
42674  * License - LGPL
42675  */
42676  
42677
42678 /**
42679  * @class Roo.form.ComboNested
42680  * @extends Roo.form.ComboBox
42681  * A combobox for that allows selection of nested items in a list,
42682  * eg.
42683  *
42684  *  Book
42685  *    -> red
42686  *    -> green
42687  *  Table
42688  *    -> square
42689  *      ->red
42690  *      ->green
42691  *    -> rectangle
42692  *      ->green
42693  *      
42694  * 
42695  * @constructor
42696  * Create a new ComboNested
42697  * @param {Object} config Configuration options
42698  */
42699 Roo.form.ComboNested = function(config){
42700     Roo.form.ComboCheck.superclass.constructor.call(this, config);
42701     // should verify some data...
42702     // like
42703     // hiddenName = required..
42704     // displayField = required
42705     // valudField == required
42706     var req= [ 'hiddenName', 'displayField', 'valueField' ];
42707     var _t = this;
42708     Roo.each(req, function(e) {
42709         if ((typeof(_t[e]) == 'undefined' ) || !_t[e].length) {
42710             throw "Roo.form.ComboNested : missing value for: " + e;
42711         }
42712     });
42713      
42714     
42715 };
42716
42717 Roo.extend(Roo.form.ComboNested, Roo.form.ComboBox, {
42718    
42719    
42720     list : null, // the outermost div..
42721     innerLists : null, // the
42722     views : null,
42723     stores : null,
42724     // private
42725     onRender : function(ct, position)
42726     {
42727         Roo.form.ComboBox.superclass.onRender.call(this, ct, position); // skip parent call - got to above..
42728         
42729         if(this.hiddenName){
42730             this.hiddenField = this.el.insertSibling({tag:'input', type:'hidden', name: this.hiddenName, id:  (this.hiddenId||this.hiddenName)},
42731                     'before', true);
42732             this.hiddenField.value =
42733                 this.hiddenValue !== undefined ? this.hiddenValue :
42734                 this.value !== undefined ? this.value : '';
42735
42736             // prevent input submission
42737             this.el.dom.removeAttribute('name');
42738              
42739              
42740         }
42741         
42742         if(Roo.isGecko){
42743             this.el.dom.setAttribute('autocomplete', 'off');
42744         }
42745
42746         var cls = 'x-combo-list';
42747
42748         this.list = new Roo.Layer({
42749             shadow: this.shadow, cls: [cls, this.listClass].join(' '), constrain:false
42750         });
42751
42752         var lw = this.listWidth || Math.max(this.wrap.getWidth(), this.minListWidth);
42753         this.list.setWidth(lw);
42754         this.list.swallowEvent('mousewheel');
42755         this.assetHeight = 0;
42756
42757         if(this.title){
42758             this.header = this.list.createChild({cls:cls+'-hd', html: this.title});
42759             this.assetHeight += this.header.getHeight();
42760         }
42761         this.innerLists = [];
42762         this.views = [];
42763         this.stores = [];
42764         for (var i =0 ; i < 3; i++) {
42765             this.onRenderList( cls, i);
42766         }
42767         
42768         // always needs footer, as we are going to have an 'OK' button.
42769         this.footer = this.list.createChild({cls:cls+'-ft'});
42770         this.pageTb = new Roo.Toolbar(this.footer);  
42771         var _this = this;
42772         this.pageTb.add(  {
42773             
42774             text: 'Done',
42775             handler: function()
42776             {
42777                 _this.collapse();
42778             }
42779         });
42780         
42781         if ( this.allowBlank && !this.disableClear) {
42782             
42783             this.pageTb.add(new Roo.Toolbar.Fill(), {
42784                 cls: 'x-btn-icon x-btn-clear',
42785                 text: '&#160;',
42786                 handler: function()
42787                 {
42788                     _this.collapse();
42789                     _this.clearValue();
42790                     _this.onSelect(false, -1);
42791                 }
42792             });
42793         }
42794         if (this.footer) {
42795             this.assetHeight += this.footer.getHeight();
42796         }
42797         
42798     },
42799     onRenderList : function (  cls, i)
42800     {
42801         
42802         var lw = Math.floor(
42803                 ((this.listWidth * 3 || Math.max(this.wrap.getWidth(), this.minListWidth)) - this.list.getFrameWidth('lr')) / 3
42804         );
42805         
42806         this.list.setWidth(lw); // default to '1'
42807
42808         var il = this.innerLists[i] = this.list.createChild({cls:cls+'-inner'});
42809         //il.on('mouseover', this.onViewOver, this, { list:  i });
42810         //il.on('mousemove', this.onViewMove, this, { list:  i });
42811         il.setWidth(lw);
42812         il.setStyle({ 'overflow-x' : 'hidden'});
42813
42814         if(!this.tpl){
42815             this.tpl = new Roo.Template({
42816                 html :  '<div class="'+cls+'-item '+cls+'-item-{cn:this.isEmpty}">{' + this.displayField + '}</div>',
42817                 isEmpty: function (value, allValues) {
42818                     return value && value.length ? 'has-children' : 'no-children'
42819                 }
42820             });
42821         }
42822         
42823         var store  = this.store;
42824         if (i > 0) {
42825             store  = new Roo.data.SimpleStore({
42826                 fields : this.store.reader.meta.fields,
42827                 data : [ ]
42828             });
42829         }
42830         this.stores[i]  = store;
42831                 
42832         
42833         
42834         var view = this.views[i] = new Roo.View(
42835             il,
42836             this.tpl,
42837             {
42838                 singleSelect:true,
42839                 store: store,
42840                 selectedClass: this.selectedClass
42841             }
42842         );
42843         view.getEl().setWidth(lw);
42844         view.getEl().setStyle({
42845             position: i < 1 ? 'relative' : 'absolute',
42846             top: 0,
42847             left: (i * lw ) + 'px',
42848             display : i > 0 ? 'none' : 'block'
42849         });
42850         view.on('selectionchange', this.onSelectChange, this, {list : i });
42851         view.on('dblclick', this.onDoubleClick, this, {list : i });
42852         //view.on('click', this.onViewClick, this, { list : i });
42853
42854         store.on('beforeload', this.onBeforeLoad, this);
42855         store.on('load',  this.onStoreLoad, this, { list  : i});
42856         store.on('loadexception', this.onLoadException, this);
42857
42858         // hide the other vies..
42859         
42860         
42861         
42862     },
42863     onResize : function()  {},
42864     
42865     restrictHeight : function()
42866     {
42867         var mh = 0;
42868         Roo.each(this.innerLists, function(il,i) {
42869             var el = this.views[i].getEl();
42870             el.dom.style.height = '';
42871             var inner = el.dom;
42872             var h = Math.max(inner.clientHeight, inner.offsetHeight, inner.scrollHeight);
42873             // only adjust heights on other ones..
42874             if (i < 1) {
42875                 
42876                 el.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
42877                 il.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
42878                 mh = Math.max(el.getHeight(), mh);
42879             }
42880             
42881             
42882         }, this);
42883         
42884         this.list.beginUpdate();
42885         this.list.setHeight(mh+this.list.getFrameWidth('tb')+this.assetHeight);
42886         this.list.alignTo(this.el, this.listAlign);
42887         this.list.endUpdate();
42888         
42889     },
42890      
42891     
42892     // -- store handlers..
42893     // private
42894     onBeforeLoad : function()
42895     {
42896         if(!this.hasFocus){
42897             return;
42898         }
42899         this.innerLists[0].update(this.loadingText ?
42900                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
42901         this.restrictHeight();
42902         this.selectedIndex = -1;
42903     },
42904     // private
42905     onLoad : function(a,b,c,d)
42906     {
42907         
42908         if(!this.hasFocus){
42909             return;
42910         }
42911         
42912         if(this.store.getCount() > 0) {
42913             this.expand();
42914             this.restrictHeight();   
42915         } else {
42916             this.onEmptyResults();
42917         }
42918         /*
42919         this.stores[1].loadData([]);
42920         this.stores[2].loadData([]);
42921         this.views
42922         */    
42923     
42924         //this.el.focus();
42925     },
42926     onStoreLoad : function ()
42927     {
42928         Roo.log(arguments);
42929     },
42930     
42931     // private
42932     onLoadException : function()
42933     {
42934         this.collapse();
42935         Roo.log(this.store.reader.jsonData);
42936         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
42937             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
42938         }
42939         
42940         
42941     } ,
42942      
42943      
42944
42945     onSelectChange : function (view, sels, opts )
42946     {
42947         var ix = view.getSelectedIndexes();
42948         
42949         
42950         if (opts.list > 1) {
42951              
42952             this.setFromData(ix.length ? view.store.getAt(ix[0]).data : {});
42953             return;
42954         }
42955         
42956         if (!ix.length) {
42957             this.setFromData({});
42958             this.stores[opts.list+1].loadData( [] );
42959             return;
42960         }
42961         
42962         var rec = view.store.getAt(ix[0]);
42963         this.setFromData(rec.data);
42964         
42965         var lw = Math.floor(
42966                 ((this.listWidth * 3 || Math.max(this.wrap.getWidth(), this.minListWidth)) - this.list.getFrameWidth('lr')) / 3
42967         );
42968         
42969         this.stores[opts.list+1].loadData( typeof(rec.data.cn) == 'undefined' ? [] : rec.data.cn);
42970         this.views[opts.list+1].getEl().setHeight( this.innerLists[0].getHeight());
42971         this.views[opts.list+1].getEl().setStyle({ display : rec.data.cn.length ? 'block' : 'none' });
42972         this.innerLists[opts.list+1].setHeight( this.innerLists[0].getHeight());
42973         this.list.setWidth(lw * (opts.list + (rec.data.cn.length ? 2 : 1))); 
42974     },
42975     onDoubleClick : function()
42976     {
42977         this.collapse(); //??
42978     },
42979     
42980      
42981     
42982     findRecord : function (prop,value)
42983     {
42984         return this.findRecordInStore(this.store, prop,value);
42985     },
42986     
42987      // private
42988     findRecordInStore : function(store, prop, value)
42989     {
42990         var cstore = new Roo.data.SimpleStore({
42991             reader : this.store.reader,  // we need array reader.. for 
42992             data : [ ]
42993         });
42994         var _this = this;
42995         var record  = false;
42996         if(store.getCount() > 0){
42997            store.each(function(r){
42998                 if(r.data[prop] == value){
42999                     record = r;
43000                     return false;
43001                 }
43002                 if (r.data.cn && r.data.cn.length) {
43003                     cstore.loadData( r.data.cn);
43004                     var cret = _this.findRecordInStore(cstore, prop, value);
43005                     if (cret !== false) {
43006                         record = cret;
43007                         return false;
43008                     }
43009                 }
43010                 
43011                 return true;
43012             });
43013         }
43014         return record;
43015     }
43016     
43017     
43018     
43019     
43020 });/*
43021  * Based on:
43022  * Ext JS Library 1.1.1
43023  * Copyright(c) 2006-2007, Ext JS, LLC.
43024  *
43025  * Originally Released Under LGPL - original licence link has changed is not relivant.
43026  *
43027  * Fork - LGPL
43028  * <script type="text/javascript">
43029  */
43030 /**
43031  * @class Roo.form.Checkbox
43032  * @extends Roo.form.Field
43033  * Single checkbox field.  Can be used as a direct replacement for traditional checkbox fields.
43034  * @constructor
43035  * Creates a new Checkbox
43036  * @param {Object} config Configuration options
43037  */
43038 Roo.form.Checkbox = function(config){
43039     Roo.form.Checkbox.superclass.constructor.call(this, config);
43040     this.addEvents({
43041         /**
43042          * @event check
43043          * Fires when the checkbox is checked or unchecked.
43044              * @param {Roo.form.Checkbox} this This checkbox
43045              * @param {Boolean} checked The new checked value
43046              */
43047         check : true
43048     });
43049 };
43050
43051 Roo.extend(Roo.form.Checkbox, Roo.form.Field,  {
43052     /**
43053      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
43054      */
43055     focusClass : undefined,
43056     /**
43057      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
43058      */
43059     fieldClass: "x-form-field",
43060     /**
43061      * @cfg {Boolean} checked True if the the checkbox should render already checked (defaults to false)
43062      */
43063     checked: false,
43064     /**
43065      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
43066      * {tag: "input", type: "checkbox", autocomplete: "off"})
43067      */
43068     defaultAutoCreate : { tag: "input", type: 'hidden', autocomplete: "off"},
43069     /**
43070      * @cfg {String} boxLabel The text that appears beside the checkbox
43071      */
43072     boxLabel : "",
43073     /**
43074      * @cfg {String} inputValue The value that should go into the generated input element's value attribute
43075      */  
43076     inputValue : '1',
43077     /**
43078      * @cfg {String} valueOff The value that should go into the generated input element's value when unchecked.
43079      */
43080      valueOff: '0', // value when not checked..
43081
43082     actionMode : 'viewEl', 
43083     //
43084     // private
43085     itemCls : 'x-menu-check-item x-form-item',
43086     groupClass : 'x-menu-group-item',
43087     inputType : 'hidden',
43088     
43089     
43090     inSetChecked: false, // check that we are not calling self...
43091     
43092     inputElement: false, // real input element?
43093     basedOn: false, // ????
43094     
43095     isFormField: true, // not sure where this is needed!!!!
43096
43097     onResize : function(){
43098         Roo.form.Checkbox.superclass.onResize.apply(this, arguments);
43099         if(!this.boxLabel){
43100             this.el.alignTo(this.wrap, 'c-c');
43101         }
43102     },
43103
43104     initEvents : function(){
43105         Roo.form.Checkbox.superclass.initEvents.call(this);
43106         this.el.on("click", this.onClick,  this);
43107         this.el.on("change", this.onClick,  this);
43108     },
43109
43110
43111     getResizeEl : function(){
43112         return this.wrap;
43113     },
43114
43115     getPositionEl : function(){
43116         return this.wrap;
43117     },
43118
43119     // private
43120     onRender : function(ct, position){
43121         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
43122         /*
43123         if(this.inputValue !== undefined){
43124             this.el.dom.value = this.inputValue;
43125         }
43126         */
43127         //this.wrap = this.el.wrap({cls: "x-form-check-wrap"});
43128         this.wrap = this.el.wrap({cls: 'x-menu-check-item '});
43129         var viewEl = this.wrap.createChild({ 
43130             tag: 'img', cls: 'x-menu-item-icon', style: 'margin: 0px;' ,src : Roo.BLANK_IMAGE_URL });
43131         this.viewEl = viewEl;   
43132         this.wrap.on('click', this.onClick,  this); 
43133         
43134         this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
43135         this.el.on('propertychange', this.setFromHidden,  this);  //ie
43136         
43137         
43138         
43139         if(this.boxLabel){
43140             this.wrap.createChild({tag: 'label', htmlFor: this.el.id, cls: 'x-form-cb-label', html: this.boxLabel});
43141         //    viewEl.on('click', this.onClick,  this); 
43142         }
43143         //if(this.checked){
43144             this.setChecked(this.checked);
43145         //}else{
43146             //this.checked = this.el.dom;
43147         //}
43148
43149     },
43150
43151     // private
43152     initValue : Roo.emptyFn,
43153
43154     /**
43155      * Returns the checked state of the checkbox.
43156      * @return {Boolean} True if checked, else false
43157      */
43158     getValue : function(){
43159         if(this.el){
43160             return String(this.el.dom.value) == String(this.inputValue ) ? this.inputValue : this.valueOff;
43161         }
43162         return this.valueOff;
43163         
43164     },
43165
43166         // private
43167     onClick : function(){ 
43168         if (this.disabled) {
43169             return;
43170         }
43171         this.setChecked(!this.checked);
43172
43173         //if(this.el.dom.checked != this.checked){
43174         //    this.setValue(this.el.dom.checked);
43175        // }
43176     },
43177
43178     /**
43179      * Sets the checked state of the checkbox.
43180      * On is always based on a string comparison between inputValue and the param.
43181      * @param {Boolean/String} value - the value to set 
43182      * @param {Boolean/String} suppressEvent - whether to suppress the checkchange event.
43183      */
43184     setValue : function(v,suppressEvent){
43185         
43186         
43187         //this.checked = (v === true || v === 'true' || v == '1' || String(v).toLowerCase() == 'on');
43188         //if(this.el && this.el.dom){
43189         //    this.el.dom.checked = this.checked;
43190         //    this.el.dom.defaultChecked = this.checked;
43191         //}
43192         this.setChecked(String(v) === String(this.inputValue), suppressEvent);
43193         //this.fireEvent("check", this, this.checked);
43194     },
43195     // private..
43196     setChecked : function(state,suppressEvent)
43197     {
43198         if (this.inSetChecked) {
43199             this.checked = state;
43200             return;
43201         }
43202         
43203     
43204         if(this.wrap){
43205             this.wrap[state ? 'addClass' : 'removeClass']('x-menu-item-checked');
43206         }
43207         this.checked = state;
43208         if(suppressEvent !== true){
43209             this.fireEvent('check', this, state);
43210         }
43211         this.inSetChecked = true;
43212         this.el.dom.value = state ? this.inputValue : this.valueOff;
43213         this.inSetChecked = false;
43214         
43215     },
43216     // handle setting of hidden value by some other method!!?!?
43217     setFromHidden: function()
43218     {
43219         if(!this.el){
43220             return;
43221         }
43222         //console.log("SET FROM HIDDEN");
43223         //alert('setFrom hidden');
43224         this.setValue(this.el.dom.value);
43225     },
43226     
43227     onDestroy : function()
43228     {
43229         if(this.viewEl){
43230             Roo.get(this.viewEl).remove();
43231         }
43232          
43233         Roo.form.Checkbox.superclass.onDestroy.call(this);
43234     },
43235     
43236     setBoxLabel : function(str)
43237     {
43238         this.wrap.select('.x-form-cb-label', true).first().dom.innerHTML = str;
43239     }
43240
43241 });/*
43242  * Based on:
43243  * Ext JS Library 1.1.1
43244  * Copyright(c) 2006-2007, Ext JS, LLC.
43245  *
43246  * Originally Released Under LGPL - original licence link has changed is not relivant.
43247  *
43248  * Fork - LGPL
43249  * <script type="text/javascript">
43250  */
43251  
43252 /**
43253  * @class Roo.form.Radio
43254  * @extends Roo.form.Checkbox
43255  * Single radio field.  Same as Checkbox, but provided as a convenience for automatically setting the input type.
43256  * Radio grouping is handled automatically by the browser if you give each radio in a group the same name.
43257  * @constructor
43258  * Creates a new Radio
43259  * @param {Object} config Configuration options
43260  */
43261 Roo.form.Radio = function(){
43262     Roo.form.Radio.superclass.constructor.apply(this, arguments);
43263 };
43264 Roo.extend(Roo.form.Radio, Roo.form.Checkbox, {
43265     inputType: 'radio',
43266
43267     /**
43268      * If this radio is part of a group, it will return the selected value
43269      * @return {String}
43270      */
43271     getGroupValue : function(){
43272         return this.el.up('form').child('input[name='+this.el.dom.name+']:checked', true).value;
43273     },
43274     
43275     
43276     onRender : function(ct, position){
43277         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
43278         
43279         if(this.inputValue !== undefined){
43280             this.el.dom.value = this.inputValue;
43281         }
43282          
43283         this.wrap = this.el.wrap({cls: "x-form-check-wrap"});
43284         //this.wrap = this.el.wrap({cls: 'x-menu-check-item '});
43285         //var viewEl = this.wrap.createChild({ 
43286         //    tag: 'img', cls: 'x-menu-item-icon', style: 'margin: 0px;' ,src : Roo.BLANK_IMAGE_URL });
43287         //this.viewEl = viewEl;   
43288         //this.wrap.on('click', this.onClick,  this); 
43289         
43290         //this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
43291         //this.el.on('propertychange', this.setFromHidden,  this);  //ie
43292         
43293         
43294         
43295         if(this.boxLabel){
43296             this.wrap.createChild({tag: 'label', htmlFor: this.el.id, cls: 'x-form-cb-label', html: this.boxLabel});
43297         //    viewEl.on('click', this.onClick,  this); 
43298         }
43299          if(this.checked){
43300             this.el.dom.checked =   'checked' ;
43301         }
43302          
43303     } 
43304     
43305     
43306 });//<script type="text/javascript">
43307
43308 /*
43309  * Based  Ext JS Library 1.1.1
43310  * Copyright(c) 2006-2007, Ext JS, LLC.
43311  * LGPL
43312  *
43313  */
43314  
43315 /**
43316  * @class Roo.HtmlEditorCore
43317  * @extends Roo.Component
43318  * Provides a the editing component for the HTML editors in Roo. (bootstrap and Roo.form)
43319  *
43320  * any element that has display set to 'none' can cause problems in Safari and Firefox.<br/><br/>
43321  */
43322
43323 Roo.HtmlEditorCore = function(config){
43324     
43325     
43326     Roo.HtmlEditorCore.superclass.constructor.call(this, config);
43327     
43328     
43329     this.addEvents({
43330         /**
43331          * @event initialize
43332          * Fires when the editor is fully initialized (including the iframe)
43333          * @param {Roo.HtmlEditorCore} this
43334          */
43335         initialize: true,
43336         /**
43337          * @event activate
43338          * Fires when the editor is first receives the focus. Any insertion must wait
43339          * until after this event.
43340          * @param {Roo.HtmlEditorCore} this
43341          */
43342         activate: true,
43343          /**
43344          * @event beforesync
43345          * Fires before the textarea is updated with content from the editor iframe. Return false
43346          * to cancel the sync.
43347          * @param {Roo.HtmlEditorCore} this
43348          * @param {String} html
43349          */
43350         beforesync: true,
43351          /**
43352          * @event beforepush
43353          * Fires before the iframe editor is updated with content from the textarea. Return false
43354          * to cancel the push.
43355          * @param {Roo.HtmlEditorCore} this
43356          * @param {String} html
43357          */
43358         beforepush: true,
43359          /**
43360          * @event sync
43361          * Fires when the textarea is updated with content from the editor iframe.
43362          * @param {Roo.HtmlEditorCore} this
43363          * @param {String} html
43364          */
43365         sync: true,
43366          /**
43367          * @event push
43368          * Fires when the iframe editor is updated with content from the textarea.
43369          * @param {Roo.HtmlEditorCore} this
43370          * @param {String} html
43371          */
43372         push: true,
43373         
43374         /**
43375          * @event editorevent
43376          * Fires when on any editor (mouse up/down cursor movement etc.) - used for toolbar hooks.
43377          * @param {Roo.HtmlEditorCore} this
43378          */
43379         editorevent: true
43380         
43381     });
43382     
43383     // at this point this.owner is set, so we can start working out the whitelisted / blacklisted elements
43384     
43385     // defaults : white / black...
43386     this.applyBlacklists();
43387     
43388     
43389     
43390 };
43391
43392
43393 Roo.extend(Roo.HtmlEditorCore, Roo.Component,  {
43394
43395
43396      /**
43397      * @cfg {Roo.form.HtmlEditor|Roo.bootstrap.HtmlEditor} the owner field 
43398      */
43399     
43400     owner : false,
43401     
43402      /**
43403      * @cfg {String} resizable  's' or 'se' or 'e' - wrapps the element in a
43404      *                        Roo.resizable.
43405      */
43406     resizable : false,
43407      /**
43408      * @cfg {Number} height (in pixels)
43409      */   
43410     height: 300,
43411    /**
43412      * @cfg {Number} width (in pixels)
43413      */   
43414     width: 500,
43415     
43416     /**
43417      * @cfg {Array} stylesheets url of stylesheets. set to [] to disable stylesheets.
43418      * 
43419      */
43420     stylesheets: false,
43421     
43422     // id of frame..
43423     frameId: false,
43424     
43425     // private properties
43426     validationEvent : false,
43427     deferHeight: true,
43428     initialized : false,
43429     activated : false,
43430     sourceEditMode : false,
43431     onFocus : Roo.emptyFn,
43432     iframePad:3,
43433     hideMode:'offsets',
43434     
43435     clearUp: true,
43436     
43437     // blacklist + whitelisted elements..
43438     black: false,
43439     white: false,
43440      
43441     bodyCls : '',
43442
43443     /**
43444      * Protected method that will not generally be called directly. It
43445      * is called when the editor initializes the iframe with HTML contents. Override this method if you
43446      * want to change the initialization markup of the iframe (e.g. to add stylesheets).
43447      */
43448     getDocMarkup : function(){
43449         // body styles..
43450         var st = '';
43451         
43452         // inherit styels from page...?? 
43453         if (this.stylesheets === false) {
43454             
43455             Roo.get(document.head).select('style').each(function(node) {
43456                 st += node.dom.outerHTML || new XMLSerializer().serializeToString(node.dom);
43457             });
43458             
43459             Roo.get(document.head).select('link').each(function(node) { 
43460                 st += node.dom.outerHTML || new XMLSerializer().serializeToString(node.dom);
43461             });
43462             
43463         } else if (!this.stylesheets.length) {
43464                 // simple..
43465                 st = '<style type="text/css">' +
43466                     'body{border:0;margin:0;padding:3px;height:98%;cursor:text;}' +
43467                    '</style>';
43468         } else { 
43469             st = '<style type="text/css">' +
43470                     this.stylesheets +
43471                 '</style>';
43472         }
43473         
43474         st +=  '<style type="text/css">' +
43475             'IMG { cursor: pointer } ' +
43476         '</style>';
43477
43478         var cls = 'roo-htmleditor-body';
43479         
43480         if(this.bodyCls.length){
43481             cls += ' ' + this.bodyCls;
43482         }
43483         
43484         return '<html><head>' + st  +
43485             //<style type="text/css">' +
43486             //'body{border:0;margin:0;padding:3px;height:98%;cursor:text;}' +
43487             //'</style>' +
43488             ' </head><body class="' +  cls + '"></body></html>';
43489     },
43490
43491     // private
43492     onRender : function(ct, position)
43493     {
43494         var _t = this;
43495         //Roo.HtmlEditorCore.superclass.onRender.call(this, ct, position);
43496         this.el = this.owner.inputEl ? this.owner.inputEl() : this.owner.el;
43497         
43498         
43499         this.el.dom.style.border = '0 none';
43500         this.el.dom.setAttribute('tabIndex', -1);
43501         this.el.addClass('x-hidden hide');
43502         
43503         
43504         
43505         if(Roo.isIE){ // fix IE 1px bogus margin
43506             this.el.applyStyles('margin-top:-1px;margin-bottom:-1px;')
43507         }
43508        
43509         
43510         this.frameId = Roo.id();
43511         
43512          
43513         
43514         var iframe = this.owner.wrap.createChild({
43515             tag: 'iframe',
43516             cls: 'form-control', // bootstrap..
43517             id: this.frameId,
43518             name: this.frameId,
43519             frameBorder : 'no',
43520             'src' : Roo.SSL_SECURE_URL ? Roo.SSL_SECURE_URL  :  "javascript:false"
43521         }, this.el
43522         );
43523         
43524         
43525         this.iframe = iframe.dom;
43526
43527          this.assignDocWin();
43528         
43529         this.doc.designMode = 'on';
43530        
43531         this.doc.open();
43532         this.doc.write(this.getDocMarkup());
43533         this.doc.close();
43534
43535         
43536         var task = { // must defer to wait for browser to be ready
43537             run : function(){
43538                 //console.log("run task?" + this.doc.readyState);
43539                 this.assignDocWin();
43540                 if(this.doc.body || this.doc.readyState == 'complete'){
43541                     try {
43542                         this.doc.designMode="on";
43543                     } catch (e) {
43544                         return;
43545                     }
43546                     Roo.TaskMgr.stop(task);
43547                     this.initEditor.defer(10, this);
43548                 }
43549             },
43550             interval : 10,
43551             duration: 10000,
43552             scope: this
43553         };
43554         Roo.TaskMgr.start(task);
43555
43556     },
43557
43558     // private
43559     onResize : function(w, h)
43560     {
43561          Roo.log('resize: ' +w + ',' + h );
43562         //Roo.HtmlEditorCore.superclass.onResize.apply(this, arguments);
43563         if(!this.iframe){
43564             return;
43565         }
43566         if(typeof w == 'number'){
43567             
43568             this.iframe.style.width = w + 'px';
43569         }
43570         if(typeof h == 'number'){
43571             
43572             this.iframe.style.height = h + 'px';
43573             if(this.doc){
43574                 (this.doc.body || this.doc.documentElement).style.height = (h - (this.iframePad*2)) + 'px';
43575             }
43576         }
43577         
43578     },
43579
43580     /**
43581      * Toggles the editor between standard and source edit mode.
43582      * @param {Boolean} sourceEdit (optional) True for source edit, false for standard
43583      */
43584     toggleSourceEdit : function(sourceEditMode){
43585         
43586         this.sourceEditMode = sourceEditMode === true;
43587         
43588         if(this.sourceEditMode){
43589  
43590             Roo.get(this.iframe).addClass(['x-hidden','hide']);     //FIXME - what's the BS styles for these
43591             
43592         }else{
43593             Roo.get(this.iframe).removeClass(['x-hidden','hide']);
43594             //this.iframe.className = '';
43595             this.deferFocus();
43596         }
43597         //this.setSize(this.owner.wrap.getSize());
43598         //this.fireEvent('editmodechange', this, this.sourceEditMode);
43599     },
43600
43601     
43602   
43603
43604     /**
43605      * Protected method that will not generally be called directly. If you need/want
43606      * custom HTML cleanup, this is the method you should override.
43607      * @param {String} html The HTML to be cleaned
43608      * return {String} The cleaned HTML
43609      */
43610     cleanHtml : function(html){
43611         html = String(html);
43612         if(html.length > 5){
43613             if(Roo.isSafari){ // strip safari nonsense
43614                 html = html.replace(/\sclass="(?:Apple-style-span|khtml-block-placeholder)"/gi, '');
43615             }
43616         }
43617         if(html == '&nbsp;'){
43618             html = '';
43619         }
43620         return html;
43621     },
43622
43623     /**
43624      * HTML Editor -> Textarea
43625      * Protected method that will not generally be called directly. Syncs the contents
43626      * of the editor iframe with the textarea.
43627      */
43628     syncValue : function(){
43629         if(this.initialized){
43630             var bd = (this.doc.body || this.doc.documentElement);
43631             //this.cleanUpPaste(); -- this is done else where and causes havoc..
43632             var html = bd.innerHTML;
43633             if(Roo.isSafari){
43634                 var bs = bd.getAttribute('style'); // Safari puts text-align styles on the body element!
43635                 var m = bs ? bs.match(/text-align:(.*?);/i) : false;
43636                 if(m && m[1]){
43637                     html = '<div style="'+m[0]+'">' + html + '</div>';
43638                 }
43639             }
43640             html = this.cleanHtml(html);
43641             // fix up the special chars.. normaly like back quotes in word...
43642             // however we do not want to do this with chinese..
43643             html = html.replace(/[\uD800-\uDBFF][\uDC00-\uDFFF]|[\u0080-\uFFFF]/g, function(match) {
43644                 
43645                 var cc = match.charCodeAt();
43646
43647                 // Get the character value, handling surrogate pairs
43648                 if (match.length == 2) {
43649                     // It's a surrogate pair, calculate the Unicode code point
43650                     var high = match.charCodeAt(0) - 0xD800;
43651                     var low  = match.charCodeAt(1) - 0xDC00;
43652                     cc = (high * 0x400) + low + 0x10000;
43653                 }  else if (
43654                     (cc >= 0x4E00 && cc < 0xA000 ) ||
43655                     (cc >= 0x3400 && cc < 0x4E00 ) ||
43656                     (cc >= 0xf900 && cc < 0xfb00 )
43657                 ) {
43658                         return match;
43659                 }  
43660          
43661                 // No, use a numeric entity. Here we brazenly (and possibly mistakenly)
43662                 return "&#" + cc + ";";
43663                 
43664                 
43665             });
43666             
43667             
43668              
43669             if(this.owner.fireEvent('beforesync', this, html) !== false){
43670                 this.el.dom.value = html;
43671                 this.owner.fireEvent('sync', this, html);
43672             }
43673         }
43674     },
43675
43676     /**
43677      * Protected method that will not generally be called directly. Pushes the value of the textarea
43678      * into the iframe editor.
43679      */
43680     pushValue : function(){
43681         if(this.initialized){
43682             var v = this.el.dom.value.trim();
43683             
43684 //            if(v.length < 1){
43685 //                v = '&#160;';
43686 //            }
43687             
43688             if(this.owner.fireEvent('beforepush', this, v) !== false){
43689                 var d = (this.doc.body || this.doc.documentElement);
43690                 d.innerHTML = v;
43691                 this.cleanUpPaste();
43692                 this.el.dom.value = d.innerHTML;
43693                 this.owner.fireEvent('push', this, v);
43694             }
43695         }
43696     },
43697
43698     // private
43699     deferFocus : function(){
43700         this.focus.defer(10, this);
43701     },
43702
43703     // doc'ed in Field
43704     focus : function(){
43705         if(this.win && !this.sourceEditMode){
43706             this.win.focus();
43707         }else{
43708             this.el.focus();
43709         }
43710     },
43711     
43712     assignDocWin: function()
43713     {
43714         var iframe = this.iframe;
43715         
43716          if(Roo.isIE){
43717             this.doc = iframe.contentWindow.document;
43718             this.win = iframe.contentWindow;
43719         } else {
43720 //            if (!Roo.get(this.frameId)) {
43721 //                return;
43722 //            }
43723 //            this.doc = (iframe.contentDocument || Roo.get(this.frameId).dom.document);
43724 //            this.win = Roo.get(this.frameId).dom.contentWindow;
43725             
43726             if (!Roo.get(this.frameId) && !iframe.contentDocument) {
43727                 return;
43728             }
43729             
43730             this.doc = (iframe.contentDocument || Roo.get(this.frameId).dom.document);
43731             this.win = (iframe.contentWindow || Roo.get(this.frameId).dom.contentWindow);
43732         }
43733     },
43734     
43735     // private
43736     initEditor : function(){
43737         //console.log("INIT EDITOR");
43738         this.assignDocWin();
43739         
43740         
43741         
43742         this.doc.designMode="on";
43743         this.doc.open();
43744         this.doc.write(this.getDocMarkup());
43745         this.doc.close();
43746         
43747         var dbody = (this.doc.body || this.doc.documentElement);
43748         //var ss = this.el.getStyles('font-size', 'font-family', 'background-image', 'background-repeat');
43749         // this copies styles from the containing element into thsi one..
43750         // not sure why we need all of this..
43751         //var ss = this.el.getStyles('font-size', 'background-image', 'background-repeat');
43752         
43753         //var ss = this.el.getStyles( 'background-image', 'background-repeat');
43754         //ss['background-attachment'] = 'fixed'; // w3c
43755         dbody.bgProperties = 'fixed'; // ie
43756         //Roo.DomHelper.applyStyles(dbody, ss);
43757         Roo.EventManager.on(this.doc, {
43758             //'mousedown': this.onEditorEvent,
43759             'mouseup': this.onEditorEvent,
43760             'dblclick': this.onEditorEvent,
43761             'click': this.onEditorEvent,
43762             'keyup': this.onEditorEvent,
43763             buffer:100,
43764             scope: this
43765         });
43766         if(Roo.isGecko){
43767             Roo.EventManager.on(this.doc, 'keypress', this.mozKeyPress, this);
43768         }
43769         if(Roo.isIE || Roo.isSafari || Roo.isOpera){
43770             Roo.EventManager.on(this.doc, 'keydown', this.fixKeys, this);
43771         }
43772         this.initialized = true;
43773
43774         this.owner.fireEvent('initialize', this);
43775         this.pushValue();
43776     },
43777
43778     // private
43779     onDestroy : function(){
43780         
43781         
43782         
43783         if(this.rendered){
43784             
43785             //for (var i =0; i < this.toolbars.length;i++) {
43786             //    // fixme - ask toolbars for heights?
43787             //    this.toolbars[i].onDestroy();
43788            // }
43789             
43790             //this.wrap.dom.innerHTML = '';
43791             //this.wrap.remove();
43792         }
43793     },
43794
43795     // private
43796     onFirstFocus : function(){
43797         
43798         this.assignDocWin();
43799         
43800         
43801         this.activated = true;
43802          
43803     
43804         if(Roo.isGecko){ // prevent silly gecko errors
43805             this.win.focus();
43806             var s = this.win.getSelection();
43807             if(!s.focusNode || s.focusNode.nodeType != 3){
43808                 var r = s.getRangeAt(0);
43809                 r.selectNodeContents((this.doc.body || this.doc.documentElement));
43810                 r.collapse(true);
43811                 this.deferFocus();
43812             }
43813             try{
43814                 this.execCmd('useCSS', true);
43815                 this.execCmd('styleWithCSS', false);
43816             }catch(e){}
43817         }
43818         this.owner.fireEvent('activate', this);
43819     },
43820
43821     // private
43822     adjustFont: function(btn){
43823         var adjust = btn.cmd == 'increasefontsize' ? 1 : -1;
43824         //if(Roo.isSafari){ // safari
43825         //    adjust *= 2;
43826        // }
43827         var v = parseInt(this.doc.queryCommandValue('FontSize')|| 3, 10);
43828         if(Roo.isSafari){ // safari
43829             var sm = { 10 : 1, 13: 2, 16:3, 18:4, 24: 5, 32:6, 48: 7 };
43830             v =  (v < 10) ? 10 : v;
43831             v =  (v > 48) ? 48 : v;
43832             v = typeof(sm[v]) == 'undefined' ? 1 : sm[v];
43833             
43834         }
43835         
43836         
43837         v = Math.max(1, v+adjust);
43838         
43839         this.execCmd('FontSize', v  );
43840     },
43841
43842     onEditorEvent : function(e)
43843     {
43844         this.owner.fireEvent('editorevent', this, e);
43845       //  this.updateToolbar();
43846         this.syncValue(); //we can not sync so often.. sync cleans, so this breaks stuff
43847     },
43848
43849     insertTag : function(tg)
43850     {
43851         // could be a bit smarter... -> wrap the current selected tRoo..
43852         if (tg.toLowerCase() == 'span' ||
43853             tg.toLowerCase() == 'code' ||
43854             tg.toLowerCase() == 'sup' ||
43855             tg.toLowerCase() == 'sub' 
43856             ) {
43857             
43858             range = this.createRange(this.getSelection());
43859             var wrappingNode = this.doc.createElement(tg.toLowerCase());
43860             wrappingNode.appendChild(range.extractContents());
43861             range.insertNode(wrappingNode);
43862
43863             return;
43864             
43865             
43866             
43867         }
43868         this.execCmd("formatblock",   tg);
43869         
43870     },
43871     
43872     insertText : function(txt)
43873     {
43874         
43875         
43876         var range = this.createRange();
43877         range.deleteContents();
43878                //alert(Sender.getAttribute('label'));
43879                
43880         range.insertNode(this.doc.createTextNode(txt));
43881     } ,
43882     
43883      
43884
43885     /**
43886      * Executes a Midas editor command on the editor document and performs necessary focus and
43887      * toolbar updates. <b>This should only be called after the editor is initialized.</b>
43888      * @param {String} cmd The Midas command
43889      * @param {String/Boolean} value (optional) The value to pass to the command (defaults to null)
43890      */
43891     relayCmd : function(cmd, value){
43892         this.win.focus();
43893         this.execCmd(cmd, value);
43894         this.owner.fireEvent('editorevent', this);
43895         //this.updateToolbar();
43896         this.owner.deferFocus();
43897     },
43898
43899     /**
43900      * Executes a Midas editor command directly on the editor document.
43901      * For visual commands, you should use {@link #relayCmd} instead.
43902      * <b>This should only be called after the editor is initialized.</b>
43903      * @param {String} cmd The Midas command
43904      * @param {String/Boolean} value (optional) The value to pass to the command (defaults to null)
43905      */
43906     execCmd : function(cmd, value){
43907         this.doc.execCommand(cmd, false, value === undefined ? null : value);
43908         this.syncValue();
43909     },
43910  
43911  
43912    
43913     /**
43914      * Inserts the passed text at the current cursor position. Note: the editor must be initialized and activated
43915      * to insert tRoo.
43916      * @param {String} text | dom node.. 
43917      */
43918     insertAtCursor : function(text)
43919     {
43920         
43921         if(!this.activated){
43922             return;
43923         }
43924         /*
43925         if(Roo.isIE){
43926             this.win.focus();
43927             var r = this.doc.selection.createRange();
43928             if(r){
43929                 r.collapse(true);
43930                 r.pasteHTML(text);
43931                 this.syncValue();
43932                 this.deferFocus();
43933             
43934             }
43935             return;
43936         }
43937         */
43938         if(Roo.isGecko || Roo.isOpera || Roo.isSafari){
43939             this.win.focus();
43940             
43941             
43942             // from jquery ui (MIT licenced)
43943             var range, node;
43944             var win = this.win;
43945             
43946             if (win.getSelection && win.getSelection().getRangeAt) {
43947                 range = win.getSelection().getRangeAt(0);
43948                 node = typeof(text) == 'string' ? range.createContextualFragment(text) : text;
43949                 range.insertNode(node);
43950             } else if (win.document.selection && win.document.selection.createRange) {
43951                 // no firefox support
43952                 var txt = typeof(text) == 'string' ? text : text.outerHTML;
43953                 win.document.selection.createRange().pasteHTML(txt);
43954             } else {
43955                 // no firefox support
43956                 var txt = typeof(text) == 'string' ? text : text.outerHTML;
43957                 this.execCmd('InsertHTML', txt);
43958             } 
43959             
43960             this.syncValue();
43961             
43962             this.deferFocus();
43963         }
43964     },
43965  // private
43966     mozKeyPress : function(e){
43967         if(e.ctrlKey){
43968             var c = e.getCharCode(), cmd;
43969           
43970             if(c > 0){
43971                 c = String.fromCharCode(c).toLowerCase();
43972                 switch(c){
43973                     case 'b':
43974                         cmd = 'bold';
43975                         break;
43976                     case 'i':
43977                         cmd = 'italic';
43978                         break;
43979                     
43980                     case 'u':
43981                         cmd = 'underline';
43982                         break;
43983                     
43984                     case 'v':
43985                         this.cleanUpPaste.defer(100, this);
43986                         return;
43987                         
43988                 }
43989                 if(cmd){
43990                     this.win.focus();
43991                     this.execCmd(cmd);
43992                     this.deferFocus();
43993                     e.preventDefault();
43994                 }
43995                 
43996             }
43997         }
43998     },
43999
44000     // private
44001     fixKeys : function(){ // load time branching for fastest keydown performance
44002         if(Roo.isIE){
44003             return function(e){
44004                 var k = e.getKey(), r;
44005                 if(k == e.TAB){
44006                     e.stopEvent();
44007                     r = this.doc.selection.createRange();
44008                     if(r){
44009                         r.collapse(true);
44010                         r.pasteHTML('&#160;&#160;&#160;&#160;');
44011                         this.deferFocus();
44012                     }
44013                     return;
44014                 }
44015                 
44016                 if(k == e.ENTER){
44017                     r = this.doc.selection.createRange();
44018                     if(r){
44019                         var target = r.parentElement();
44020                         if(!target || target.tagName.toLowerCase() != 'li'){
44021                             e.stopEvent();
44022                             r.pasteHTML('<br />');
44023                             r.collapse(false);
44024                             r.select();
44025                         }
44026                     }
44027                 }
44028                 if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44029                     this.cleanUpPaste.defer(100, this);
44030                     return;
44031                 }
44032                 
44033                 
44034             };
44035         }else if(Roo.isOpera){
44036             return function(e){
44037                 var k = e.getKey();
44038                 if(k == e.TAB){
44039                     e.stopEvent();
44040                     this.win.focus();
44041                     this.execCmd('InsertHTML','&#160;&#160;&#160;&#160;');
44042                     this.deferFocus();
44043                 }
44044                 if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44045                     this.cleanUpPaste.defer(100, this);
44046                     return;
44047                 }
44048                 
44049             };
44050         }else if(Roo.isSafari){
44051             return function(e){
44052                 var k = e.getKey();
44053                 
44054                 if(k == e.TAB){
44055                     e.stopEvent();
44056                     this.execCmd('InsertText','\t');
44057                     this.deferFocus();
44058                     return;
44059                 }
44060                if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44061                     this.cleanUpPaste.defer(100, this);
44062                     return;
44063                 }
44064                 
44065              };
44066         }
44067     }(),
44068     
44069     getAllAncestors: function()
44070     {
44071         var p = this.getSelectedNode();
44072         var a = [];
44073         if (!p) {
44074             a.push(p); // push blank onto stack..
44075             p = this.getParentElement();
44076         }
44077         
44078         
44079         while (p && (p.nodeType == 1) && (p.tagName.toLowerCase() != 'body')) {
44080             a.push(p);
44081             p = p.parentNode;
44082         }
44083         a.push(this.doc.body);
44084         return a;
44085     },
44086     lastSel : false,
44087     lastSelNode : false,
44088     
44089     
44090     getSelection : function() 
44091     {
44092         this.assignDocWin();
44093         return Roo.isIE ? this.doc.selection : this.win.getSelection();
44094     },
44095     
44096     getSelectedNode: function() 
44097     {
44098         // this may only work on Gecko!!!
44099         
44100         // should we cache this!!!!
44101         
44102         
44103         
44104          
44105         var range = this.createRange(this.getSelection()).cloneRange();
44106         
44107         if (Roo.isIE) {
44108             var parent = range.parentElement();
44109             while (true) {
44110                 var testRange = range.duplicate();
44111                 testRange.moveToElementText(parent);
44112                 if (testRange.inRange(range)) {
44113                     break;
44114                 }
44115                 if ((parent.nodeType != 1) || (parent.tagName.toLowerCase() == 'body')) {
44116                     break;
44117                 }
44118                 parent = parent.parentElement;
44119             }
44120             return parent;
44121         }
44122         
44123         // is ancestor a text element.
44124         var ac =  range.commonAncestorContainer;
44125         if (ac.nodeType == 3) {
44126             ac = ac.parentNode;
44127         }
44128         
44129         var ar = ac.childNodes;
44130          
44131         var nodes = [];
44132         var other_nodes = [];
44133         var has_other_nodes = false;
44134         for (var i=0;i<ar.length;i++) {
44135             if ((ar[i].nodeType == 3) && (!ar[i].data.length)) { // empty text ? 
44136                 continue;
44137             }
44138             // fullly contained node.
44139             
44140             if (this.rangeIntersectsNode(range,ar[i]) && this.rangeCompareNode(range,ar[i]) == 3) {
44141                 nodes.push(ar[i]);
44142                 continue;
44143             }
44144             
44145             // probably selected..
44146             if ((ar[i].nodeType == 1) && this.rangeIntersectsNode(range,ar[i]) && (this.rangeCompareNode(range,ar[i]) > 0)) {
44147                 other_nodes.push(ar[i]);
44148                 continue;
44149             }
44150             // outer..
44151             if (!this.rangeIntersectsNode(range,ar[i])|| (this.rangeCompareNode(range,ar[i]) == 0))  {
44152                 continue;
44153             }
44154             
44155             
44156             has_other_nodes = true;
44157         }
44158         if (!nodes.length && other_nodes.length) {
44159             nodes= other_nodes;
44160         }
44161         if (has_other_nodes || !nodes.length || (nodes.length > 1)) {
44162             return false;
44163         }
44164         
44165         return nodes[0];
44166     },
44167     createRange: function(sel)
44168     {
44169         // this has strange effects when using with 
44170         // top toolbar - not sure if it's a great idea.
44171         //this.editor.contentWindow.focus();
44172         if (typeof sel != "undefined") {
44173             try {
44174                 return sel.getRangeAt ? sel.getRangeAt(0) : sel.createRange();
44175             } catch(e) {
44176                 return this.doc.createRange();
44177             }
44178         } else {
44179             return this.doc.createRange();
44180         }
44181     },
44182     getParentElement: function()
44183     {
44184         
44185         this.assignDocWin();
44186         var sel = Roo.isIE ? this.doc.selection : this.win.getSelection();
44187         
44188         var range = this.createRange(sel);
44189          
44190         try {
44191             var p = range.commonAncestorContainer;
44192             while (p.nodeType == 3) { // text node
44193                 p = p.parentNode;
44194             }
44195             return p;
44196         } catch (e) {
44197             return null;
44198         }
44199     
44200     },
44201     /***
44202      *
44203      * Range intersection.. the hard stuff...
44204      *  '-1' = before
44205      *  '0' = hits..
44206      *  '1' = after.
44207      *         [ -- selected range --- ]
44208      *   [fail]                        [fail]
44209      *
44210      *    basically..
44211      *      if end is before start or  hits it. fail.
44212      *      if start is after end or hits it fail.
44213      *
44214      *   if either hits (but other is outside. - then it's not 
44215      *   
44216      *    
44217      **/
44218     
44219     
44220     // @see http://www.thismuchiknow.co.uk/?p=64.
44221     rangeIntersectsNode : function(range, node)
44222     {
44223         var nodeRange = node.ownerDocument.createRange();
44224         try {
44225             nodeRange.selectNode(node);
44226         } catch (e) {
44227             nodeRange.selectNodeContents(node);
44228         }
44229     
44230         var rangeStartRange = range.cloneRange();
44231         rangeStartRange.collapse(true);
44232     
44233         var rangeEndRange = range.cloneRange();
44234         rangeEndRange.collapse(false);
44235     
44236         var nodeStartRange = nodeRange.cloneRange();
44237         nodeStartRange.collapse(true);
44238     
44239         var nodeEndRange = nodeRange.cloneRange();
44240         nodeEndRange.collapse(false);
44241     
44242         return rangeStartRange.compareBoundaryPoints(
44243                  Range.START_TO_START, nodeEndRange) == -1 &&
44244                rangeEndRange.compareBoundaryPoints(
44245                  Range.START_TO_START, nodeStartRange) == 1;
44246         
44247          
44248     },
44249     rangeCompareNode : function(range, node)
44250     {
44251         var nodeRange = node.ownerDocument.createRange();
44252         try {
44253             nodeRange.selectNode(node);
44254         } catch (e) {
44255             nodeRange.selectNodeContents(node);
44256         }
44257         
44258         
44259         range.collapse(true);
44260     
44261         nodeRange.collapse(true);
44262      
44263         var ss = range.compareBoundaryPoints( Range.START_TO_START, nodeRange);
44264         var ee = range.compareBoundaryPoints(  Range.END_TO_END, nodeRange);
44265          
44266         //Roo.log(node.tagName + ': ss='+ss +', ee='+ee)
44267         
44268         var nodeIsBefore   =  ss == 1;
44269         var nodeIsAfter    = ee == -1;
44270         
44271         if (nodeIsBefore && nodeIsAfter) {
44272             return 0; // outer
44273         }
44274         if (!nodeIsBefore && nodeIsAfter) {
44275             return 1; //right trailed.
44276         }
44277         
44278         if (nodeIsBefore && !nodeIsAfter) {
44279             return 2;  // left trailed.
44280         }
44281         // fully contined.
44282         return 3;
44283     },
44284
44285     // private? - in a new class?
44286     cleanUpPaste :  function()
44287     {
44288         // cleans up the whole document..
44289         Roo.log('cleanuppaste');
44290         
44291         this.cleanUpChildren(this.doc.body);
44292         var clean = this.cleanWordChars(this.doc.body.innerHTML);
44293         if (clean != this.doc.body.innerHTML) {
44294             this.doc.body.innerHTML = clean;
44295         }
44296         
44297     },
44298     
44299     cleanWordChars : function(input) {// change the chars to hex code
44300         var he = Roo.HtmlEditorCore;
44301         
44302         var output = input;
44303         Roo.each(he.swapCodes, function(sw) { 
44304             var swapper = new RegExp("\\u" + sw[0].toString(16), "g"); // hex codes
44305             
44306             output = output.replace(swapper, sw[1]);
44307         });
44308         
44309         return output;
44310     },
44311     
44312     
44313     cleanUpChildren : function (n)
44314     {
44315         if (!n.childNodes.length) {
44316             return;
44317         }
44318         for (var i = n.childNodes.length-1; i > -1 ; i--) {
44319            this.cleanUpChild(n.childNodes[i]);
44320         }
44321     },
44322     
44323     
44324         
44325     
44326     cleanUpChild : function (node)
44327     {
44328         var ed = this;
44329         //console.log(node);
44330         if (node.nodeName == "#text") {
44331             // clean up silly Windows -- stuff?
44332             return; 
44333         }
44334         if (node.nodeName == "#comment") {
44335             node.parentNode.removeChild(node);
44336             // clean up silly Windows -- stuff?
44337             return; 
44338         }
44339         var lcname = node.tagName.toLowerCase();
44340         // we ignore whitelists... ?? = not really the way to go, but we probably have not got a full
44341         // whitelist of tags..
44342         
44343         if (this.black.indexOf(lcname) > -1 && this.clearUp ) {
44344             // remove node.
44345             node.parentNode.removeChild(node);
44346             return;
44347             
44348         }
44349         
44350         var remove_keep_children= Roo.HtmlEditorCore.remove.indexOf(node.tagName.toLowerCase()) > -1;
44351         
44352         // spans with no attributes - just remove them..
44353         if ((!node.attributes || !node.attributes.length) && lcname == 'span') { 
44354             remove_keep_children = true;
44355         }
44356         
44357         // remove <a name=....> as rendering on yahoo mailer is borked with this.
44358         // this will have to be flaged elsewhere - perhaps ablack=name... on the mailer..
44359         
44360         //if (node.tagName.toLowerCase() == 'a' && !node.hasAttribute('href')) {
44361         //    remove_keep_children = true;
44362         //}
44363         
44364         if (remove_keep_children) {
44365             this.cleanUpChildren(node);
44366             // inserts everything just before this node...
44367             while (node.childNodes.length) {
44368                 var cn = node.childNodes[0];
44369                 node.removeChild(cn);
44370                 node.parentNode.insertBefore(cn, node);
44371             }
44372             node.parentNode.removeChild(node);
44373             return;
44374         }
44375         
44376         if (!node.attributes || !node.attributes.length) {
44377             
44378           
44379             
44380             
44381             this.cleanUpChildren(node);
44382             return;
44383         }
44384         
44385         function cleanAttr(n,v)
44386         {
44387             
44388             if (v.match(/^\./) || v.match(/^\//)) {
44389                 return;
44390             }
44391             if (v.match(/^(http|https):\/\//) || v.match(/^mailto:/) || v.match(/^ftp:/)) {
44392                 return;
44393             }
44394             if (v.match(/^#/)) {
44395                 return;
44396             }
44397 //            Roo.log("(REMOVE TAG)"+ node.tagName +'.' + n + '=' + v);
44398             node.removeAttribute(n);
44399             
44400         }
44401         
44402         var cwhite = this.cwhite;
44403         var cblack = this.cblack;
44404             
44405         function cleanStyle(n,v)
44406         {
44407             if (v.match(/expression/)) { //XSS?? should we even bother..
44408                 node.removeAttribute(n);
44409                 return;
44410             }
44411             
44412             var parts = v.split(/;/);
44413             var clean = [];
44414             
44415             Roo.each(parts, function(p) {
44416                 p = p.replace(/^\s+/g,'').replace(/\s+$/g,'');
44417                 if (!p.length) {
44418                     return true;
44419                 }
44420                 var l = p.split(':').shift().replace(/\s+/g,'');
44421                 l = l.replace(/^\s+/g,'').replace(/\s+$/g,'');
44422                 
44423                 if ( cwhite.length && cblack.indexOf(l) > -1) {
44424 //                    Roo.log('(REMOVE CSS)' + node.tagName +'.' + n + ':'+l + '=' + v);
44425                     //node.removeAttribute(n);
44426                     return true;
44427                 }
44428                 //Roo.log()
44429                 // only allow 'c whitelisted system attributes'
44430                 if ( cwhite.length &&  cwhite.indexOf(l) < 0) {
44431 //                    Roo.log('(REMOVE CSS)' + node.tagName +'.' + n + ':'+l + '=' + v);
44432                     //node.removeAttribute(n);
44433                     return true;
44434                 }
44435                 
44436                 
44437                  
44438                 
44439                 clean.push(p);
44440                 return true;
44441             });
44442             if (clean.length) { 
44443                 node.setAttribute(n, clean.join(';'));
44444             } else {
44445                 node.removeAttribute(n);
44446             }
44447             
44448         }
44449         
44450         
44451         for (var i = node.attributes.length-1; i > -1 ; i--) {
44452             var a = node.attributes[i];
44453             //console.log(a);
44454             
44455             if (a.name.toLowerCase().substr(0,2)=='on')  {
44456                 node.removeAttribute(a.name);
44457                 continue;
44458             }
44459             if (Roo.HtmlEditorCore.ablack.indexOf(a.name.toLowerCase()) > -1) {
44460                 node.removeAttribute(a.name);
44461                 continue;
44462             }
44463             if (Roo.HtmlEditorCore.aclean.indexOf(a.name.toLowerCase()) > -1) {
44464                 cleanAttr(a.name,a.value); // fixme..
44465                 continue;
44466             }
44467             if (a.name == 'style') {
44468                 cleanStyle(a.name,a.value);
44469                 continue;
44470             }
44471             /// clean up MS crap..
44472             // tecnically this should be a list of valid class'es..
44473             
44474             
44475             if (a.name == 'class') {
44476                 if (a.value.match(/^Mso/)) {
44477                     node.removeAttribute('class');
44478                 }
44479                 
44480                 if (a.value.match(/^body$/)) {
44481                     node.removeAttribute('class');
44482                 }
44483                 continue;
44484             }
44485             
44486             // style cleanup!?
44487             // class cleanup?
44488             
44489         }
44490         
44491         
44492         this.cleanUpChildren(node);
44493         
44494         
44495     },
44496     
44497     /**
44498      * Clean up MS wordisms...
44499      */
44500     cleanWord : function(node)
44501     {
44502         if (!node) {
44503             this.cleanWord(this.doc.body);
44504             return;
44505         }
44506         
44507         if(
44508                 node.nodeName == 'SPAN' &&
44509                 !node.hasAttributes() &&
44510                 node.childNodes.length == 1 &&
44511                 node.firstChild.nodeName == "#text"  
44512         ) {
44513             var textNode = node.firstChild;
44514             node.removeChild(textNode);
44515             if (node.getAttribute('lang') != 'zh-CN') {   // do not space pad on chinese characters..
44516                 node.parentNode.insertBefore(node.ownerDocument.createTextNode(" "), node);
44517             }
44518             node.parentNode.insertBefore(textNode, node);
44519             if (node.getAttribute('lang') != 'zh-CN') {   // do not space pad on chinese characters..
44520                 node.parentNode.insertBefore(node.ownerDocument.createTextNode(" ") , node);
44521             }
44522             node.parentNode.removeChild(node);
44523         }
44524         
44525         if (node.nodeName == "#text") {
44526             // clean up silly Windows -- stuff?
44527             return; 
44528         }
44529         if (node.nodeName == "#comment") {
44530             node.parentNode.removeChild(node);
44531             // clean up silly Windows -- stuff?
44532             return; 
44533         }
44534         
44535         if (node.tagName.toLowerCase().match(/^(style|script|applet|embed|noframes|noscript)$/)) {
44536             node.parentNode.removeChild(node);
44537             return;
44538         }
44539         //Roo.log(node.tagName);
44540         // remove - but keep children..
44541         if (node.tagName.toLowerCase().match(/^(meta|link|\\?xml:|st1:|o:|v:|font)/)) {
44542             //Roo.log('-- removed');
44543             while (node.childNodes.length) {
44544                 var cn = node.childNodes[0];
44545                 node.removeChild(cn);
44546                 node.parentNode.insertBefore(cn, node);
44547                 // move node to parent - and clean it..
44548                 this.cleanWord(cn);
44549             }
44550             node.parentNode.removeChild(node);
44551             /// no need to iterate chidlren = it's got none..
44552             //this.iterateChildren(node, this.cleanWord);
44553             return;
44554         }
44555         // clean styles
44556         if (node.className.length) {
44557             
44558             var cn = node.className.split(/\W+/);
44559             var cna = [];
44560             Roo.each(cn, function(cls) {
44561                 if (cls.match(/Mso[a-zA-Z]+/)) {
44562                     return;
44563                 }
44564                 cna.push(cls);
44565             });
44566             node.className = cna.length ? cna.join(' ') : '';
44567             if (!cna.length) {
44568                 node.removeAttribute("class");
44569             }
44570         }
44571         
44572         if (node.hasAttribute("lang")) {
44573             node.removeAttribute("lang");
44574         }
44575         
44576         if (node.hasAttribute("style")) {
44577             
44578             var styles = node.getAttribute("style").split(";");
44579             var nstyle = [];
44580             Roo.each(styles, function(s) {
44581                 if (!s.match(/:/)) {
44582                     return;
44583                 }
44584                 var kv = s.split(":");
44585                 if (kv[0].match(/^(mso-|line|font|background|margin|padding|color)/)) {
44586                     return;
44587                 }
44588                 // what ever is left... we allow.
44589                 nstyle.push(s);
44590             });
44591             node.setAttribute("style", nstyle.length ? nstyle.join(';') : '');
44592             if (!nstyle.length) {
44593                 node.removeAttribute('style');
44594             }
44595         }
44596         this.iterateChildren(node, this.cleanWord);
44597         
44598         
44599         
44600     },
44601     /**
44602      * iterateChildren of a Node, calling fn each time, using this as the scole..
44603      * @param {DomNode} node node to iterate children of.
44604      * @param {Function} fn method of this class to call on each item.
44605      */
44606     iterateChildren : function(node, fn)
44607     {
44608         if (!node.childNodes.length) {
44609                 return;
44610         }
44611         for (var i = node.childNodes.length-1; i > -1 ; i--) {
44612            fn.call(this, node.childNodes[i])
44613         }
44614     },
44615     
44616     
44617     /**
44618      * cleanTableWidths.
44619      *
44620      * Quite often pasting from word etc.. results in tables with column and widths.
44621      * This does not work well on fluid HTML layouts - like emails. - so this code should hunt an destroy them..
44622      *
44623      */
44624     cleanTableWidths : function(node)
44625     {
44626          
44627          
44628         if (!node) {
44629             this.cleanTableWidths(this.doc.body);
44630             return;
44631         }
44632         
44633         // ignore list...
44634         if (node.nodeName == "#text" || node.nodeName == "#comment") {
44635             return; 
44636         }
44637         Roo.log(node.tagName);
44638         if (!node.tagName.toLowerCase().match(/^(table|td|tr)$/)) {
44639             this.iterateChildren(node, this.cleanTableWidths);
44640             return;
44641         }
44642         if (node.hasAttribute('width')) {
44643             node.removeAttribute('width');
44644         }
44645         
44646          
44647         if (node.hasAttribute("style")) {
44648             // pretty basic...
44649             
44650             var styles = node.getAttribute("style").split(";");
44651             var nstyle = [];
44652             Roo.each(styles, function(s) {
44653                 if (!s.match(/:/)) {
44654                     return;
44655                 }
44656                 var kv = s.split(":");
44657                 if (kv[0].match(/^\s*(width|min-width)\s*$/)) {
44658                     return;
44659                 }
44660                 // what ever is left... we allow.
44661                 nstyle.push(s);
44662             });
44663             node.setAttribute("style", nstyle.length ? nstyle.join(';') : '');
44664             if (!nstyle.length) {
44665                 node.removeAttribute('style');
44666             }
44667         }
44668         
44669         this.iterateChildren(node, this.cleanTableWidths);
44670         
44671         
44672     },
44673     
44674     
44675     
44676     
44677     domToHTML : function(currentElement, depth, nopadtext) {
44678         
44679         depth = depth || 0;
44680         nopadtext = nopadtext || false;
44681     
44682         if (!currentElement) {
44683             return this.domToHTML(this.doc.body);
44684         }
44685         
44686         //Roo.log(currentElement);
44687         var j;
44688         var allText = false;
44689         var nodeName = currentElement.nodeName;
44690         var tagName = Roo.util.Format.htmlEncode(currentElement.tagName);
44691         
44692         if  (nodeName == '#text') {
44693             
44694             return nopadtext ? currentElement.nodeValue : currentElement.nodeValue.trim();
44695         }
44696         
44697         
44698         var ret = '';
44699         if (nodeName != 'BODY') {
44700              
44701             var i = 0;
44702             // Prints the node tagName, such as <A>, <IMG>, etc
44703             if (tagName) {
44704                 var attr = [];
44705                 for(i = 0; i < currentElement.attributes.length;i++) {
44706                     // quoting?
44707                     var aname = currentElement.attributes.item(i).name;
44708                     if (!currentElement.attributes.item(i).value.length) {
44709                         continue;
44710                     }
44711                     attr.push(aname + '="' + Roo.util.Format.htmlEncode(currentElement.attributes.item(i).value) + '"' );
44712                 }
44713                 
44714                 ret = "<"+currentElement.tagName+ ( attr.length ? (' ' + attr.join(' ') ) : '') + ">";
44715             } 
44716             else {
44717                 
44718                 // eack
44719             }
44720         } else {
44721             tagName = false;
44722         }
44723         if (['IMG', 'BR', 'HR', 'INPUT'].indexOf(tagName) > -1) {
44724             return ret;
44725         }
44726         if (['PRE', 'TEXTAREA', 'TD', 'A', 'SPAN'].indexOf(tagName) > -1) { // or code?
44727             nopadtext = true;
44728         }
44729         
44730         
44731         // Traverse the tree
44732         i = 0;
44733         var currentElementChild = currentElement.childNodes.item(i);
44734         var allText = true;
44735         var innerHTML  = '';
44736         lastnode = '';
44737         while (currentElementChild) {
44738             // Formatting code (indent the tree so it looks nice on the screen)
44739             var nopad = nopadtext;
44740             if (lastnode == 'SPAN') {
44741                 nopad  = true;
44742             }
44743             // text
44744             if  (currentElementChild.nodeName == '#text') {
44745                 var toadd = Roo.util.Format.htmlEncode(currentElementChild.nodeValue);
44746                 toadd = nopadtext ? toadd : toadd.trim();
44747                 if (!nopad && toadd.length > 80) {
44748                     innerHTML  += "\n" + (new Array( depth + 1 )).join( "  "  );
44749                 }
44750                 innerHTML  += toadd;
44751                 
44752                 i++;
44753                 currentElementChild = currentElement.childNodes.item(i);
44754                 lastNode = '';
44755                 continue;
44756             }
44757             allText = false;
44758             
44759             innerHTML  += nopad ? '' : "\n" + (new Array( depth + 1 )).join( "  "  );
44760                 
44761             // Recursively traverse the tree structure of the child node
44762             innerHTML   += this.domToHTML(currentElementChild, depth+1, nopadtext);
44763             lastnode = currentElementChild.nodeName;
44764             i++;
44765             currentElementChild=currentElement.childNodes.item(i);
44766         }
44767         
44768         ret += innerHTML;
44769         
44770         if (!allText) {
44771                 // The remaining code is mostly for formatting the tree
44772             ret+= nopadtext ? '' : "\n" + (new Array( depth  )).join( "  "  );
44773         }
44774         
44775         
44776         if (tagName) {
44777             ret+= "</"+tagName+">";
44778         }
44779         return ret;
44780         
44781     },
44782         
44783     applyBlacklists : function()
44784     {
44785         var w = typeof(this.owner.white) != 'undefined' && this.owner.white ? this.owner.white  : [];
44786         var b = typeof(this.owner.black) != 'undefined' && this.owner.black ? this.owner.black :  [];
44787         
44788         this.white = [];
44789         this.black = [];
44790         Roo.each(Roo.HtmlEditorCore.white, function(tag) {
44791             if (b.indexOf(tag) > -1) {
44792                 return;
44793             }
44794             this.white.push(tag);
44795             
44796         }, this);
44797         
44798         Roo.each(w, function(tag) {
44799             if (b.indexOf(tag) > -1) {
44800                 return;
44801             }
44802             if (this.white.indexOf(tag) > -1) {
44803                 return;
44804             }
44805             this.white.push(tag);
44806             
44807         }, this);
44808         
44809         
44810         Roo.each(Roo.HtmlEditorCore.black, function(tag) {
44811             if (w.indexOf(tag) > -1) {
44812                 return;
44813             }
44814             this.black.push(tag);
44815             
44816         }, this);
44817         
44818         Roo.each(b, function(tag) {
44819             if (w.indexOf(tag) > -1) {
44820                 return;
44821             }
44822             if (this.black.indexOf(tag) > -1) {
44823                 return;
44824             }
44825             this.black.push(tag);
44826             
44827         }, this);
44828         
44829         
44830         w = typeof(this.owner.cwhite) != 'undefined' && this.owner.cwhite ? this.owner.cwhite  : [];
44831         b = typeof(this.owner.cblack) != 'undefined' && this.owner.cblack ? this.owner.cblack :  [];
44832         
44833         this.cwhite = [];
44834         this.cblack = [];
44835         Roo.each(Roo.HtmlEditorCore.cwhite, function(tag) {
44836             if (b.indexOf(tag) > -1) {
44837                 return;
44838             }
44839             this.cwhite.push(tag);
44840             
44841         }, this);
44842         
44843         Roo.each(w, function(tag) {
44844             if (b.indexOf(tag) > -1) {
44845                 return;
44846             }
44847             if (this.cwhite.indexOf(tag) > -1) {
44848                 return;
44849             }
44850             this.cwhite.push(tag);
44851             
44852         }, this);
44853         
44854         
44855         Roo.each(Roo.HtmlEditorCore.cblack, function(tag) {
44856             if (w.indexOf(tag) > -1) {
44857                 return;
44858             }
44859             this.cblack.push(tag);
44860             
44861         }, this);
44862         
44863         Roo.each(b, function(tag) {
44864             if (w.indexOf(tag) > -1) {
44865                 return;
44866             }
44867             if (this.cblack.indexOf(tag) > -1) {
44868                 return;
44869             }
44870             this.cblack.push(tag);
44871             
44872         }, this);
44873     },
44874     
44875     setStylesheets : function(stylesheets)
44876     {
44877         if(typeof(stylesheets) == 'string'){
44878             Roo.get(this.iframe.contentDocument.head).createChild({
44879                 tag : 'link',
44880                 rel : 'stylesheet',
44881                 type : 'text/css',
44882                 href : stylesheets
44883             });
44884             
44885             return;
44886         }
44887         var _this = this;
44888      
44889         Roo.each(stylesheets, function(s) {
44890             if(!s.length){
44891                 return;
44892             }
44893             
44894             Roo.get(_this.iframe.contentDocument.head).createChild({
44895                 tag : 'link',
44896                 rel : 'stylesheet',
44897                 type : 'text/css',
44898                 href : s
44899             });
44900         });
44901
44902         
44903     },
44904     
44905     removeStylesheets : function()
44906     {
44907         var _this = this;
44908         
44909         Roo.each(Roo.get(_this.iframe.contentDocument.head).select('link[rel=stylesheet]', true).elements, function(s){
44910             s.remove();
44911         });
44912     },
44913     
44914     setStyle : function(style)
44915     {
44916         Roo.get(this.iframe.contentDocument.head).createChild({
44917             tag : 'style',
44918             type : 'text/css',
44919             html : style
44920         });
44921
44922         return;
44923     }
44924     
44925     // hide stuff that is not compatible
44926     /**
44927      * @event blur
44928      * @hide
44929      */
44930     /**
44931      * @event change
44932      * @hide
44933      */
44934     /**
44935      * @event focus
44936      * @hide
44937      */
44938     /**
44939      * @event specialkey
44940      * @hide
44941      */
44942     /**
44943      * @cfg {String} fieldClass @hide
44944      */
44945     /**
44946      * @cfg {String} focusClass @hide
44947      */
44948     /**
44949      * @cfg {String} autoCreate @hide
44950      */
44951     /**
44952      * @cfg {String} inputType @hide
44953      */
44954     /**
44955      * @cfg {String} invalidClass @hide
44956      */
44957     /**
44958      * @cfg {String} invalidText @hide
44959      */
44960     /**
44961      * @cfg {String} msgFx @hide
44962      */
44963     /**
44964      * @cfg {String} validateOnBlur @hide
44965      */
44966 });
44967
44968 Roo.HtmlEditorCore.white = [
44969         'area', 'br', 'img', 'input', 'hr', 'wbr',
44970         
44971        'address', 'blockquote', 'center', 'dd',      'dir',       'div', 
44972        'dl',      'dt',         'h1',     'h2',      'h3',        'h4', 
44973        'h5',      'h6',         'hr',     'isindex', 'listing',   'marquee', 
44974        'menu',    'multicol',   'ol',     'p',       'plaintext', 'pre', 
44975        'table',   'ul',         'xmp', 
44976        
44977        'caption', 'col', 'colgroup', 'tbody', 'td', 'tfoot', 'th', 
44978       'thead',   'tr', 
44979      
44980       'dir', 'menu', 'ol', 'ul', 'dl',
44981        
44982       'embed',  'object'
44983 ];
44984
44985
44986 Roo.HtmlEditorCore.black = [
44987     //    'embed',  'object', // enable - backend responsiblity to clean thiese
44988         'applet', // 
44989         'base',   'basefont', 'bgsound', 'blink',  'body', 
44990         'frame',  'frameset', 'head',    'html',   'ilayer', 
44991         'iframe', 'layer',  'link',     'meta',    'object',   
44992         'script', 'style' ,'title',  'xml' // clean later..
44993 ];
44994 Roo.HtmlEditorCore.clean = [
44995     'script', 'style', 'title', 'xml'
44996 ];
44997 Roo.HtmlEditorCore.remove = [
44998     'font'
44999 ];
45000 // attributes..
45001
45002 Roo.HtmlEditorCore.ablack = [
45003     'on'
45004 ];
45005     
45006 Roo.HtmlEditorCore.aclean = [ 
45007     'action', 'background', 'codebase', 'dynsrc', 'href', 'lowsrc' 
45008 ];
45009
45010 // protocols..
45011 Roo.HtmlEditorCore.pwhite= [
45012         'http',  'https',  'mailto'
45013 ];
45014
45015 // white listed style attributes.
45016 Roo.HtmlEditorCore.cwhite= [
45017       //  'text-align', /// default is to allow most things..
45018       
45019          
45020 //        'font-size'//??
45021 ];
45022
45023 // black listed style attributes.
45024 Roo.HtmlEditorCore.cblack= [
45025       //  'font-size' -- this can be set by the project 
45026 ];
45027
45028
45029 Roo.HtmlEditorCore.swapCodes   =[ 
45030     [    8211, "--" ], 
45031     [    8212, "--" ], 
45032     [    8216,  "'" ],  
45033     [    8217, "'" ],  
45034     [    8220, '"' ],  
45035     [    8221, '"' ],  
45036     [    8226, "*" ],  
45037     [    8230, "..." ]
45038 ]; 
45039
45040     //<script type="text/javascript">
45041
45042 /*
45043  * Ext JS Library 1.1.1
45044  * Copyright(c) 2006-2007, Ext JS, LLC.
45045  * Licence LGPL
45046  * 
45047  */
45048  
45049  
45050 Roo.form.HtmlEditor = function(config){
45051     
45052     
45053     
45054     Roo.form.HtmlEditor.superclass.constructor.call(this, config);
45055     
45056     if (!this.toolbars) {
45057         this.toolbars = [];
45058     }
45059     this.editorcore = new Roo.HtmlEditorCore(Roo.apply({ owner : this} , config));
45060     
45061     
45062 };
45063
45064 /**
45065  * @class Roo.form.HtmlEditor
45066  * @extends Roo.form.Field
45067  * Provides a lightweight HTML Editor component.
45068  *
45069  * This has been tested on Fireforx / Chrome.. IE may not be so great..
45070  * 
45071  * <br><br><b>Note: The focus/blur and validation marking functionality inherited from Ext.form.Field is NOT
45072  * supported by this editor.</b><br/><br/>
45073  * An Editor is a sensitive component that can't be used in all spots standard fields can be used. Putting an Editor within
45074  * any element that has display set to 'none' can cause problems in Safari and Firefox.<br/><br/>
45075  */
45076 Roo.extend(Roo.form.HtmlEditor, Roo.form.Field, {
45077     /**
45078      * @cfg {Boolean} clearUp
45079      */
45080     clearUp : true,
45081       /**
45082      * @cfg {Array} toolbars Array of toolbars. - defaults to just the Standard one
45083      */
45084     toolbars : false,
45085    
45086      /**
45087      * @cfg {String} resizable  's' or 'se' or 'e' - wrapps the element in a
45088      *                        Roo.resizable.
45089      */
45090     resizable : false,
45091      /**
45092      * @cfg {Number} height (in pixels)
45093      */   
45094     height: 300,
45095    /**
45096      * @cfg {Number} width (in pixels)
45097      */   
45098     width: 500,
45099     
45100     /**
45101      * @cfg {Array} stylesheets url of stylesheets. set to [] to disable stylesheets.
45102      * 
45103      */
45104     stylesheets: false,
45105     
45106     
45107      /**
45108      * @cfg {Array} blacklist of css styles style attributes (blacklist overrides whitelist)
45109      * 
45110      */
45111     cblack: false,
45112     /**
45113      * @cfg {Array} whitelist of css styles style attributes (blacklist overrides whitelist)
45114      * 
45115      */
45116     cwhite: false,
45117     
45118      /**
45119      * @cfg {Array} blacklist of html tags - in addition to standard blacklist.
45120      * 
45121      */
45122     black: false,
45123     /**
45124      * @cfg {Array} whitelist of html tags - in addition to statndard whitelist
45125      * 
45126      */
45127     white: false,
45128     
45129     // id of frame..
45130     frameId: false,
45131     
45132     // private properties
45133     validationEvent : false,
45134     deferHeight: true,
45135     initialized : false,
45136     activated : false,
45137     
45138     onFocus : Roo.emptyFn,
45139     iframePad:3,
45140     hideMode:'offsets',
45141     
45142     actionMode : 'container', // defaults to hiding it...
45143     
45144     defaultAutoCreate : { // modified by initCompnoent..
45145         tag: "textarea",
45146         style:"width:500px;height:300px;",
45147         autocomplete: "new-password"
45148     },
45149
45150     // private
45151     initComponent : function(){
45152         this.addEvents({
45153             /**
45154              * @event initialize
45155              * Fires when the editor is fully initialized (including the iframe)
45156              * @param {HtmlEditor} this
45157              */
45158             initialize: true,
45159             /**
45160              * @event activate
45161              * Fires when the editor is first receives the focus. Any insertion must wait
45162              * until after this event.
45163              * @param {HtmlEditor} this
45164              */
45165             activate: true,
45166              /**
45167              * @event beforesync
45168              * Fires before the textarea is updated with content from the editor iframe. Return false
45169              * to cancel the sync.
45170              * @param {HtmlEditor} this
45171              * @param {String} html
45172              */
45173             beforesync: true,
45174              /**
45175              * @event beforepush
45176              * Fires before the iframe editor is updated with content from the textarea. Return false
45177              * to cancel the push.
45178              * @param {HtmlEditor} this
45179              * @param {String} html
45180              */
45181             beforepush: true,
45182              /**
45183              * @event sync
45184              * Fires when the textarea is updated with content from the editor iframe.
45185              * @param {HtmlEditor} this
45186              * @param {String} html
45187              */
45188             sync: true,
45189              /**
45190              * @event push
45191              * Fires when the iframe editor is updated with content from the textarea.
45192              * @param {HtmlEditor} this
45193              * @param {String} html
45194              */
45195             push: true,
45196              /**
45197              * @event editmodechange
45198              * Fires when the editor switches edit modes
45199              * @param {HtmlEditor} this
45200              * @param {Boolean} sourceEdit True if source edit, false if standard editing.
45201              */
45202             editmodechange: true,
45203             /**
45204              * @event editorevent
45205              * Fires when on any editor (mouse up/down cursor movement etc.) - used for toolbar hooks.
45206              * @param {HtmlEditor} this
45207              */
45208             editorevent: true,
45209             /**
45210              * @event firstfocus
45211              * Fires when on first focus - needed by toolbars..
45212              * @param {HtmlEditor} this
45213              */
45214             firstfocus: true,
45215             /**
45216              * @event autosave
45217              * Auto save the htmlEditor value as a file into Events
45218              * @param {HtmlEditor} this
45219              */
45220             autosave: true,
45221             /**
45222              * @event savedpreview
45223              * preview the saved version of htmlEditor
45224              * @param {HtmlEditor} this
45225              */
45226             savedpreview: true,
45227             
45228             /**
45229             * @event stylesheetsclick
45230             * Fires when press the Sytlesheets button
45231             * @param {Roo.HtmlEditorCore} this
45232             */
45233             stylesheetsclick: true
45234         });
45235         this.defaultAutoCreate =  {
45236             tag: "textarea",
45237             style:'width: ' + this.width + 'px;height: ' + this.height + 'px;',
45238             autocomplete: "new-password"
45239         };
45240     },
45241
45242     /**
45243      * Protected method that will not generally be called directly. It
45244      * is called when the editor creates its toolbar. Override this method if you need to
45245      * add custom toolbar buttons.
45246      * @param {HtmlEditor} editor
45247      */
45248     createToolbar : function(editor){
45249         Roo.log("create toolbars");
45250         if (!editor.toolbars || !editor.toolbars.length) {
45251             editor.toolbars = [ new Roo.form.HtmlEditor.ToolbarStandard() ]; // can be empty?
45252         }
45253         
45254         for (var i =0 ; i < editor.toolbars.length;i++) {
45255             editor.toolbars[i] = Roo.factory(
45256                     typeof(editor.toolbars[i]) == 'string' ?
45257                         { xtype: editor.toolbars[i]} : editor.toolbars[i],
45258                 Roo.form.HtmlEditor);
45259             editor.toolbars[i].init(editor);
45260         }
45261          
45262         
45263     },
45264
45265      
45266     // private
45267     onRender : function(ct, position)
45268     {
45269         var _t = this;
45270         Roo.form.HtmlEditor.superclass.onRender.call(this, ct, position);
45271         
45272         this.wrap = this.el.wrap({
45273             cls:'x-html-editor-wrap', cn:{cls:'x-html-editor-tb'}
45274         });
45275         
45276         this.editorcore.onRender(ct, position);
45277          
45278         if (this.resizable) {
45279             this.resizeEl = new Roo.Resizable(this.wrap, {
45280                 pinned : true,
45281                 wrap: true,
45282                 dynamic : true,
45283                 minHeight : this.height,
45284                 height: this.height,
45285                 handles : this.resizable,
45286                 width: this.width,
45287                 listeners : {
45288                     resize : function(r, w, h) {
45289                         _t.onResize(w,h); // -something
45290                     }
45291                 }
45292             });
45293             
45294         }
45295         this.createToolbar(this);
45296        
45297         
45298         if(!this.width){
45299             this.setSize(this.wrap.getSize());
45300         }
45301         if (this.resizeEl) {
45302             this.resizeEl.resizeTo.defer(100, this.resizeEl,[ this.width,this.height ] );
45303             // should trigger onReize..
45304         }
45305         
45306         this.keyNav = new Roo.KeyNav(this.el, {
45307             
45308             "tab" : function(e){
45309                 e.preventDefault();
45310                 
45311                 var value = this.getValue();
45312                 
45313                 var start = this.el.dom.selectionStart;
45314                 var end = this.el.dom.selectionEnd;
45315                 
45316                 if(!e.shiftKey){
45317                     
45318                     this.setValue(value.substring(0, start) + "\t" + value.substring(end));
45319                     this.el.dom.setSelectionRange(end + 1, end + 1);
45320                     return;
45321                 }
45322                 
45323                 var f = value.substring(0, start).split("\t");
45324                 
45325                 if(f.pop().length != 0){
45326                     return;
45327                 }
45328                 
45329                 this.setValue(f.join("\t") + value.substring(end));
45330                 this.el.dom.setSelectionRange(start - 1, start - 1);
45331                 
45332             },
45333             
45334             "home" : function(e){
45335                 e.preventDefault();
45336                 
45337                 var curr = this.el.dom.selectionStart;
45338                 var lines = this.getValue().split("\n");
45339                 
45340                 if(!lines.length){
45341                     return;
45342                 }
45343                 
45344                 if(e.ctrlKey){
45345                     this.el.dom.setSelectionRange(0, 0);
45346                     return;
45347                 }
45348                 
45349                 var pos = 0;
45350                 
45351                 for (var i = 0; i < lines.length;i++) {
45352                     pos += lines[i].length;
45353                     
45354                     if(i != 0){
45355                         pos += 1;
45356                     }
45357                     
45358                     if(pos < curr){
45359                         continue;
45360                     }
45361                     
45362                     pos -= lines[i].length;
45363                     
45364                     break;
45365                 }
45366                 
45367                 if(!e.shiftKey){
45368                     this.el.dom.setSelectionRange(pos, pos);
45369                     return;
45370                 }
45371                 
45372                 this.el.dom.selectionStart = pos;
45373                 this.el.dom.selectionEnd = curr;
45374             },
45375             
45376             "end" : function(e){
45377                 e.preventDefault();
45378                 
45379                 var curr = this.el.dom.selectionStart;
45380                 var lines = this.getValue().split("\n");
45381                 
45382                 if(!lines.length){
45383                     return;
45384                 }
45385                 
45386                 if(e.ctrlKey){
45387                     this.el.dom.setSelectionRange(this.getValue().length, this.getValue().length);
45388                     return;
45389                 }
45390                 
45391                 var pos = 0;
45392                 
45393                 for (var i = 0; i < lines.length;i++) {
45394                     
45395                     pos += lines[i].length;
45396                     
45397                     if(i != 0){
45398                         pos += 1;
45399                     }
45400                     
45401                     if(pos < curr){
45402                         continue;
45403                     }
45404                     
45405                     break;
45406                 }
45407                 
45408                 if(!e.shiftKey){
45409                     this.el.dom.setSelectionRange(pos, pos);
45410                     return;
45411                 }
45412                 
45413                 this.el.dom.selectionStart = curr;
45414                 this.el.dom.selectionEnd = pos;
45415             },
45416
45417             scope : this,
45418
45419             doRelay : function(foo, bar, hname){
45420                 return Roo.KeyNav.prototype.doRelay.apply(this, arguments);
45421             },
45422
45423             forceKeyDown: true
45424         });
45425         
45426 //        if(this.autosave && this.w){
45427 //            this.autoSaveFn = setInterval(this.autosave, 1000);
45428 //        }
45429     },
45430
45431     // private
45432     onResize : function(w, h)
45433     {
45434         Roo.form.HtmlEditor.superclass.onResize.apply(this, arguments);
45435         var ew = false;
45436         var eh = false;
45437         
45438         if(this.el ){
45439             if(typeof w == 'number'){
45440                 var aw = w - this.wrap.getFrameWidth('lr');
45441                 this.el.setWidth(this.adjustWidth('textarea', aw));
45442                 ew = aw;
45443             }
45444             if(typeof h == 'number'){
45445                 var tbh = 0;
45446                 for (var i =0; i < this.toolbars.length;i++) {
45447                     // fixme - ask toolbars for heights?
45448                     tbh += this.toolbars[i].tb.el.getHeight();
45449                     if (this.toolbars[i].footer) {
45450                         tbh += this.toolbars[i].footer.el.getHeight();
45451                     }
45452                 }
45453                 
45454                 
45455                 
45456                 
45457                 var ah = h - this.wrap.getFrameWidth('tb') - tbh;// this.tb.el.getHeight();
45458                 ah -= 5; // knock a few pixes off for look..
45459 //                Roo.log(ah);
45460                 this.el.setHeight(this.adjustWidth('textarea', ah));
45461                 var eh = ah;
45462             }
45463         }
45464         Roo.log('onResize:' + [w,h,ew,eh].join(',') );
45465         this.editorcore.onResize(ew,eh);
45466         
45467     },
45468
45469     /**
45470      * Toggles the editor between standard and source edit mode.
45471      * @param {Boolean} sourceEdit (optional) True for source edit, false for standard
45472      */
45473     toggleSourceEdit : function(sourceEditMode)
45474     {
45475         this.editorcore.toggleSourceEdit(sourceEditMode);
45476         
45477         if(this.editorcore.sourceEditMode){
45478             Roo.log('editor - showing textarea');
45479             
45480 //            Roo.log('in');
45481 //            Roo.log(this.syncValue());
45482             this.editorcore.syncValue();
45483             this.el.removeClass('x-hidden');
45484             this.el.dom.removeAttribute('tabIndex');
45485             this.el.focus();
45486             
45487             for (var i = 0; i < this.toolbars.length; i++) {
45488                 if(this.toolbars[i] instanceof Roo.form.HtmlEditor.ToolbarContext){
45489                     this.toolbars[i].tb.hide();
45490                     this.toolbars[i].footer.hide();
45491                 }
45492             }
45493             
45494         }else{
45495             Roo.log('editor - hiding textarea');
45496 //            Roo.log('out')
45497 //            Roo.log(this.pushValue()); 
45498             this.editorcore.pushValue();
45499             
45500             this.el.addClass('x-hidden');
45501             this.el.dom.setAttribute('tabIndex', -1);
45502             
45503             for (var i = 0; i < this.toolbars.length; i++) {
45504                 if(this.toolbars[i] instanceof Roo.form.HtmlEditor.ToolbarContext){
45505                     this.toolbars[i].tb.show();
45506                     this.toolbars[i].footer.show();
45507                 }
45508             }
45509             
45510             //this.deferFocus();
45511         }
45512         
45513         this.setSize(this.wrap.getSize());
45514         this.onResize(this.wrap.getSize().width, this.wrap.getSize().height);
45515         
45516         this.fireEvent('editmodechange', this, this.editorcore.sourceEditMode);
45517     },
45518  
45519     // private (for BoxComponent)
45520     adjustSize : Roo.BoxComponent.prototype.adjustSize,
45521
45522     // private (for BoxComponent)
45523     getResizeEl : function(){
45524         return this.wrap;
45525     },
45526
45527     // private (for BoxComponent)
45528     getPositionEl : function(){
45529         return this.wrap;
45530     },
45531
45532     // private
45533     initEvents : function(){
45534         this.originalValue = this.getValue();
45535     },
45536
45537     /**
45538      * Overridden and disabled. The editor element does not support standard valid/invalid marking. @hide
45539      * @method
45540      */
45541     markInvalid : Roo.emptyFn,
45542     /**
45543      * Overridden and disabled. The editor element does not support standard valid/invalid marking. @hide
45544      * @method
45545      */
45546     clearInvalid : Roo.emptyFn,
45547
45548     setValue : function(v){
45549         Roo.form.HtmlEditor.superclass.setValue.call(this, v);
45550         this.editorcore.pushValue();
45551     },
45552
45553      
45554     // private
45555     deferFocus : function(){
45556         this.focus.defer(10, this);
45557     },
45558
45559     // doc'ed in Field
45560     focus : function(){
45561         this.editorcore.focus();
45562         
45563     },
45564       
45565
45566     // private
45567     onDestroy : function(){
45568         
45569         
45570         
45571         if(this.rendered){
45572             
45573             for (var i =0; i < this.toolbars.length;i++) {
45574                 // fixme - ask toolbars for heights?
45575                 this.toolbars[i].onDestroy();
45576             }
45577             
45578             this.wrap.dom.innerHTML = '';
45579             this.wrap.remove();
45580         }
45581     },
45582
45583     // private
45584     onFirstFocus : function(){
45585         //Roo.log("onFirstFocus");
45586         this.editorcore.onFirstFocus();
45587          for (var i =0; i < this.toolbars.length;i++) {
45588             this.toolbars[i].onFirstFocus();
45589         }
45590         
45591     },
45592     
45593     // private
45594     syncValue : function()
45595     {
45596         this.editorcore.syncValue();
45597     },
45598     
45599     pushValue : function()
45600     {
45601         this.editorcore.pushValue();
45602     },
45603     
45604     setStylesheets : function(stylesheets)
45605     {
45606         this.editorcore.setStylesheets(stylesheets);
45607     },
45608     
45609     removeStylesheets : function()
45610     {
45611         this.editorcore.removeStylesheets();
45612     }
45613      
45614     
45615     // hide stuff that is not compatible
45616     /**
45617      * @event blur
45618      * @hide
45619      */
45620     /**
45621      * @event change
45622      * @hide
45623      */
45624     /**
45625      * @event focus
45626      * @hide
45627      */
45628     /**
45629      * @event specialkey
45630      * @hide
45631      */
45632     /**
45633      * @cfg {String} fieldClass @hide
45634      */
45635     /**
45636      * @cfg {String} focusClass @hide
45637      */
45638     /**
45639      * @cfg {String} autoCreate @hide
45640      */
45641     /**
45642      * @cfg {String} inputType @hide
45643      */
45644     /**
45645      * @cfg {String} invalidClass @hide
45646      */
45647     /**
45648      * @cfg {String} invalidText @hide
45649      */
45650     /**
45651      * @cfg {String} msgFx @hide
45652      */
45653     /**
45654      * @cfg {String} validateOnBlur @hide
45655      */
45656 });
45657  
45658     // <script type="text/javascript">
45659 /*
45660  * Based on
45661  * Ext JS Library 1.1.1
45662  * Copyright(c) 2006-2007, Ext JS, LLC.
45663  *  
45664  
45665  */
45666
45667 /**
45668  * @class Roo.form.HtmlEditorToolbar1
45669  * Basic Toolbar
45670  * 
45671  * Usage:
45672  *
45673  new Roo.form.HtmlEditor({
45674     ....
45675     toolbars : [
45676         new Roo.form.HtmlEditorToolbar1({
45677             disable : { fonts: 1 , format: 1, ..., ... , ...],
45678             btns : [ .... ]
45679         })
45680     }
45681      
45682  * 
45683  * @cfg {Object} disable List of elements to disable..
45684  * @cfg {Array} btns List of additional buttons.
45685  * 
45686  * 
45687  * NEEDS Extra CSS? 
45688  * .x-html-editor-tb .x-edit-none .x-btn-text { background: none; }
45689  */
45690  
45691 Roo.form.HtmlEditor.ToolbarStandard = function(config)
45692 {
45693     
45694     Roo.apply(this, config);
45695     
45696     // default disabled, based on 'good practice'..
45697     this.disable = this.disable || {};
45698     Roo.applyIf(this.disable, {
45699         fontSize : true,
45700         colors : true,
45701         specialElements : true
45702     });
45703     
45704     
45705     //Roo.form.HtmlEditorToolbar1.superclass.constructor.call(this, editor.wrap.dom.firstChild, [], config);
45706     // dont call parent... till later.
45707 }
45708
45709 Roo.apply(Roo.form.HtmlEditor.ToolbarStandard.prototype,  {
45710     
45711     tb: false,
45712     
45713     rendered: false,
45714     
45715     editor : false,
45716     editorcore : false,
45717     /**
45718      * @cfg {Object} disable  List of toolbar elements to disable
45719          
45720      */
45721     disable : false,
45722     
45723     
45724      /**
45725      * @cfg {String} createLinkText The default text for the create link prompt
45726      */
45727     createLinkText : 'Please enter the URL for the link:',
45728     /**
45729      * @cfg {String} defaultLinkValue The default value for the create link prompt (defaults to http:/ /)
45730      */
45731     defaultLinkValue : 'http:/'+'/',
45732    
45733     
45734       /**
45735      * @cfg {Array} fontFamilies An array of available font families
45736      */
45737     fontFamilies : [
45738         'Arial',
45739         'Courier New',
45740         'Tahoma',
45741         'Times New Roman',
45742         'Verdana'
45743     ],
45744     
45745     specialChars : [
45746            "&#169;",
45747           "&#174;",     
45748           "&#8482;",    
45749           "&#163;" ,    
45750          // "&#8212;",    
45751           "&#8230;",    
45752           "&#247;" ,    
45753         //  "&#225;" ,     ?? a acute?
45754            "&#8364;"    , //Euro
45755        //   "&#8220;"    ,
45756         //  "&#8221;"    ,
45757         //  "&#8226;"    ,
45758           "&#176;"  //   , // degrees
45759
45760          // "&#233;"     , // e ecute
45761          // "&#250;"     , // u ecute?
45762     ],
45763     
45764     specialElements : [
45765         {
45766             text: "Insert Table",
45767             xtype: 'MenuItem',
45768             xns : Roo.Menu,
45769             ihtml :  '<table><tr><td>Cell</td></tr></table>' 
45770                 
45771         },
45772         {    
45773             text: "Insert Image",
45774             xtype: 'MenuItem',
45775             xns : Roo.Menu,
45776             ihtml : '<img src="about:blank"/>'
45777             
45778         }
45779         
45780          
45781     ],
45782     
45783     
45784     inputElements : [ 
45785             "form", "input:text", "input:hidden", "input:checkbox", "input:radio", "input:password", 
45786             "input:submit", "input:button", "select", "textarea", "label" ],
45787     formats : [
45788         ["p"] ,  
45789         ["h1"],["h2"],["h3"],["h4"],["h5"],["h6"], 
45790         ["pre"],[ "code"], 
45791         ["abbr"],[ "acronym"],[ "address"],[ "cite"],[ "samp"],[ "var"],
45792         ['div'],['span'],
45793         ['sup'],['sub']
45794     ],
45795     
45796     cleanStyles : [
45797         "font-size"
45798     ],
45799      /**
45800      * @cfg {String} defaultFont default font to use.
45801      */
45802     defaultFont: 'tahoma',
45803    
45804     fontSelect : false,
45805     
45806     
45807     formatCombo : false,
45808     
45809     init : function(editor)
45810     {
45811         this.editor = editor;
45812         this.editorcore = editor.editorcore ? editor.editorcore : editor;
45813         var editorcore = this.editorcore;
45814         
45815         var _t = this;
45816         
45817         var fid = editorcore.frameId;
45818         var etb = this;
45819         function btn(id, toggle, handler){
45820             var xid = fid + '-'+ id ;
45821             return {
45822                 id : xid,
45823                 cmd : id,
45824                 cls : 'x-btn-icon x-edit-'+id,
45825                 enableToggle:toggle !== false,
45826                 scope: _t, // was editor...
45827                 handler:handler||_t.relayBtnCmd,
45828                 clickEvent:'mousedown',
45829                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
45830                 tabIndex:-1
45831             };
45832         }
45833         
45834         
45835         
45836         var tb = new Roo.Toolbar(editor.wrap.dom.firstChild);
45837         this.tb = tb;
45838          // stop form submits
45839         tb.el.on('click', function(e){
45840             e.preventDefault(); // what does this do?
45841         });
45842
45843         if(!this.disable.font) { // && !Roo.isSafari){
45844             /* why no safari for fonts 
45845             editor.fontSelect = tb.el.createChild({
45846                 tag:'select',
45847                 tabIndex: -1,
45848                 cls:'x-font-select',
45849                 html: this.createFontOptions()
45850             });
45851             
45852             editor.fontSelect.on('change', function(){
45853                 var font = editor.fontSelect.dom.value;
45854                 editor.relayCmd('fontname', font);
45855                 editor.deferFocus();
45856             }, editor);
45857             
45858             tb.add(
45859                 editor.fontSelect.dom,
45860                 '-'
45861             );
45862             */
45863             
45864         };
45865         if(!this.disable.formats){
45866             this.formatCombo = new Roo.form.ComboBox({
45867                 store: new Roo.data.SimpleStore({
45868                     id : 'tag',
45869                     fields: ['tag'],
45870                     data : this.formats // from states.js
45871                 }),
45872                 blockFocus : true,
45873                 name : '',
45874                 //autoCreate : {tag: "div",  size: "20"},
45875                 displayField:'tag',
45876                 typeAhead: false,
45877                 mode: 'local',
45878                 editable : false,
45879                 triggerAction: 'all',
45880                 emptyText:'Add tag',
45881                 selectOnFocus:true,
45882                 width:135,
45883                 listeners : {
45884                     'select': function(c, r, i) {
45885                         editorcore.insertTag(r.get('tag'));
45886                         editor.focus();
45887                     }
45888                 }
45889
45890             });
45891             tb.addField(this.formatCombo);
45892             
45893         }
45894         
45895         if(!this.disable.format){
45896             tb.add(
45897                 btn('bold'),
45898                 btn('italic'),
45899                 btn('underline'),
45900                 btn('strikethrough')
45901             );
45902         };
45903         if(!this.disable.fontSize){
45904             tb.add(
45905                 '-',
45906                 
45907                 
45908                 btn('increasefontsize', false, editorcore.adjustFont),
45909                 btn('decreasefontsize', false, editorcore.adjustFont)
45910             );
45911         };
45912         
45913         
45914         if(!this.disable.colors){
45915             tb.add(
45916                 '-', {
45917                     id:editorcore.frameId +'-forecolor',
45918                     cls:'x-btn-icon x-edit-forecolor',
45919                     clickEvent:'mousedown',
45920                     tooltip: this.buttonTips['forecolor'] || undefined,
45921                     tabIndex:-1,
45922                     menu : new Roo.menu.ColorMenu({
45923                         allowReselect: true,
45924                         focus: Roo.emptyFn,
45925                         value:'000000',
45926                         plain:true,
45927                         selectHandler: function(cp, color){
45928                             editorcore.execCmd('forecolor', Roo.isSafari || Roo.isIE ? '#'+color : color);
45929                             editor.deferFocus();
45930                         },
45931                         scope: editorcore,
45932                         clickEvent:'mousedown'
45933                     })
45934                 }, {
45935                     id:editorcore.frameId +'backcolor',
45936                     cls:'x-btn-icon x-edit-backcolor',
45937                     clickEvent:'mousedown',
45938                     tooltip: this.buttonTips['backcolor'] || undefined,
45939                     tabIndex:-1,
45940                     menu : new Roo.menu.ColorMenu({
45941                         focus: Roo.emptyFn,
45942                         value:'FFFFFF',
45943                         plain:true,
45944                         allowReselect: true,
45945                         selectHandler: function(cp, color){
45946                             if(Roo.isGecko){
45947                                 editorcore.execCmd('useCSS', false);
45948                                 editorcore.execCmd('hilitecolor', color);
45949                                 editorcore.execCmd('useCSS', true);
45950                                 editor.deferFocus();
45951                             }else{
45952                                 editorcore.execCmd(Roo.isOpera ? 'hilitecolor' : 'backcolor', 
45953                                     Roo.isSafari || Roo.isIE ? '#'+color : color);
45954                                 editor.deferFocus();
45955                             }
45956                         },
45957                         scope:editorcore,
45958                         clickEvent:'mousedown'
45959                     })
45960                 }
45961             );
45962         };
45963         // now add all the items...
45964         
45965
45966         if(!this.disable.alignments){
45967             tb.add(
45968                 '-',
45969                 btn('justifyleft'),
45970                 btn('justifycenter'),
45971                 btn('justifyright')
45972             );
45973         };
45974
45975         //if(!Roo.isSafari){
45976             if(!this.disable.links){
45977                 tb.add(
45978                     '-',
45979                     btn('createlink', false, this.createLink)    /// MOVE TO HERE?!!?!?!?!
45980                 );
45981             };
45982
45983             if(!this.disable.lists){
45984                 tb.add(
45985                     '-',
45986                     btn('insertorderedlist'),
45987                     btn('insertunorderedlist')
45988                 );
45989             }
45990             if(!this.disable.sourceEdit){
45991                 tb.add(
45992                     '-',
45993                     btn('sourceedit', true, function(btn){
45994                         this.toggleSourceEdit(btn.pressed);
45995                     })
45996                 );
45997             }
45998         //}
45999         
46000         var smenu = { };
46001         // special menu.. - needs to be tidied up..
46002         if (!this.disable.special) {
46003             smenu = {
46004                 text: "&#169;",
46005                 cls: 'x-edit-none',
46006                 
46007                 menu : {
46008                     items : []
46009                 }
46010             };
46011             for (var i =0; i < this.specialChars.length; i++) {
46012                 smenu.menu.items.push({
46013                     
46014                     html: this.specialChars[i],
46015                     handler: function(a,b) {
46016                         editorcore.insertAtCursor(String.fromCharCode(a.html.replace('&#','').replace(';', '')));
46017                         //editor.insertAtCursor(a.html);
46018                         
46019                     },
46020                     tabIndex:-1
46021                 });
46022             }
46023             
46024             
46025             tb.add(smenu);
46026             
46027             
46028         }
46029         
46030         var cmenu = { };
46031         if (!this.disable.cleanStyles) {
46032             cmenu = {
46033                 cls: 'x-btn-icon x-btn-clear',
46034                 
46035                 menu : {
46036                     items : []
46037                 }
46038             };
46039             for (var i =0; i < this.cleanStyles.length; i++) {
46040                 cmenu.menu.items.push({
46041                     actiontype : this.cleanStyles[i],
46042                     html: 'Remove ' + this.cleanStyles[i],
46043                     handler: function(a,b) {
46044 //                        Roo.log(a);
46045 //                        Roo.log(b);
46046                         var c = Roo.get(editorcore.doc.body);
46047                         c.select('[style]').each(function(s) {
46048                             s.dom.style.removeProperty(a.actiontype);
46049                         });
46050                         editorcore.syncValue();
46051                     },
46052                     tabIndex:-1
46053                 });
46054             }
46055              cmenu.menu.items.push({
46056                 actiontype : 'tablewidths',
46057                 html: 'Remove Table Widths',
46058                 handler: function(a,b) {
46059                     editorcore.cleanTableWidths();
46060                     editorcore.syncValue();
46061                 },
46062                 tabIndex:-1
46063             });
46064             cmenu.menu.items.push({
46065                 actiontype : 'word',
46066                 html: 'Remove MS Word Formating',
46067                 handler: function(a,b) {
46068                     editorcore.cleanWord();
46069                     editorcore.syncValue();
46070                 },
46071                 tabIndex:-1
46072             });
46073             
46074             cmenu.menu.items.push({
46075                 actiontype : 'all',
46076                 html: 'Remove All Styles',
46077                 handler: function(a,b) {
46078                     
46079                     var c = Roo.get(editorcore.doc.body);
46080                     c.select('[style]').each(function(s) {
46081                         s.dom.removeAttribute('style');
46082                     });
46083                     editorcore.syncValue();
46084                 },
46085                 tabIndex:-1
46086             });
46087             
46088             cmenu.menu.items.push({
46089                 actiontype : 'all',
46090                 html: 'Remove All CSS Classes',
46091                 handler: function(a,b) {
46092                     
46093                     var c = Roo.get(editorcore.doc.body);
46094                     c.select('[class]').each(function(s) {
46095                         s.dom.removeAttribute('class');
46096                     });
46097                     editorcore.cleanWord();
46098                     editorcore.syncValue();
46099                 },
46100                 tabIndex:-1
46101             });
46102             
46103              cmenu.menu.items.push({
46104                 actiontype : 'tidy',
46105                 html: 'Tidy HTML Source',
46106                 handler: function(a,b) {
46107                     editorcore.doc.body.innerHTML = editorcore.domToHTML();
46108                     editorcore.syncValue();
46109                 },
46110                 tabIndex:-1
46111             });
46112             
46113             
46114             tb.add(cmenu);
46115         }
46116          
46117         if (!this.disable.specialElements) {
46118             var semenu = {
46119                 text: "Other;",
46120                 cls: 'x-edit-none',
46121                 menu : {
46122                     items : []
46123                 }
46124             };
46125             for (var i =0; i < this.specialElements.length; i++) {
46126                 semenu.menu.items.push(
46127                     Roo.apply({ 
46128                         handler: function(a,b) {
46129                             editor.insertAtCursor(this.ihtml);
46130                         }
46131                     }, this.specialElements[i])
46132                 );
46133                     
46134             }
46135             
46136             tb.add(semenu);
46137             
46138             
46139         }
46140          
46141         
46142         if (this.btns) {
46143             for(var i =0; i< this.btns.length;i++) {
46144                 var b = Roo.factory(this.btns[i],Roo.form);
46145                 b.cls =  'x-edit-none';
46146                 
46147                 if(typeof(this.btns[i].cls) != 'undefined' && this.btns[i].cls.indexOf('x-init-enable') !== -1){
46148                     b.cls += ' x-init-enable';
46149                 }
46150                 
46151                 b.scope = editorcore;
46152                 tb.add(b);
46153             }
46154         
46155         }
46156         
46157         
46158         
46159         // disable everything...
46160         
46161         this.tb.items.each(function(item){
46162             
46163            if(
46164                 item.id != editorcore.frameId+ '-sourceedit' && 
46165                 (typeof(item.cls) != 'undefined' && item.cls.indexOf('x-init-enable') === -1)
46166             ){
46167                 
46168                 item.disable();
46169             }
46170         });
46171         this.rendered = true;
46172         
46173         // the all the btns;
46174         editor.on('editorevent', this.updateToolbar, this);
46175         // other toolbars need to implement this..
46176         //editor.on('editmodechange', this.updateToolbar, this);
46177     },
46178     
46179     
46180     relayBtnCmd : function(btn) {
46181         this.editorcore.relayCmd(btn.cmd);
46182     },
46183     // private used internally
46184     createLink : function(){
46185         Roo.log("create link?");
46186         var url = prompt(this.createLinkText, this.defaultLinkValue);
46187         if(url && url != 'http:/'+'/'){
46188             this.editorcore.relayCmd('createlink', url);
46189         }
46190     },
46191
46192     
46193     /**
46194      * Protected method that will not generally be called directly. It triggers
46195      * a toolbar update by reading the markup state of the current selection in the editor.
46196      */
46197     updateToolbar: function(){
46198
46199         if(!this.editorcore.activated){
46200             this.editor.onFirstFocus();
46201             return;
46202         }
46203
46204         var btns = this.tb.items.map, 
46205             doc = this.editorcore.doc,
46206             frameId = this.editorcore.frameId;
46207
46208         if(!this.disable.font && !Roo.isSafari){
46209             /*
46210             var name = (doc.queryCommandValue('FontName')||this.editor.defaultFont).toLowerCase();
46211             if(name != this.fontSelect.dom.value){
46212                 this.fontSelect.dom.value = name;
46213             }
46214             */
46215         }
46216         if(!this.disable.format){
46217             btns[frameId + '-bold'].toggle(doc.queryCommandState('bold'));
46218             btns[frameId + '-italic'].toggle(doc.queryCommandState('italic'));
46219             btns[frameId + '-underline'].toggle(doc.queryCommandState('underline'));
46220             btns[frameId + '-strikethrough'].toggle(doc.queryCommandState('strikethrough'));
46221         }
46222         if(!this.disable.alignments){
46223             btns[frameId + '-justifyleft'].toggle(doc.queryCommandState('justifyleft'));
46224             btns[frameId + '-justifycenter'].toggle(doc.queryCommandState('justifycenter'));
46225             btns[frameId + '-justifyright'].toggle(doc.queryCommandState('justifyright'));
46226         }
46227         if(!Roo.isSafari && !this.disable.lists){
46228             btns[frameId + '-insertorderedlist'].toggle(doc.queryCommandState('insertorderedlist'));
46229             btns[frameId + '-insertunorderedlist'].toggle(doc.queryCommandState('insertunorderedlist'));
46230         }
46231         
46232         var ans = this.editorcore.getAllAncestors();
46233         if (this.formatCombo) {
46234             
46235             
46236             var store = this.formatCombo.store;
46237             this.formatCombo.setValue("");
46238             for (var i =0; i < ans.length;i++) {
46239                 if (ans[i] && store.query('tag',ans[i].tagName.toLowerCase(), false).length) {
46240                     // select it..
46241                     this.formatCombo.setValue(ans[i].tagName.toLowerCase());
46242                     break;
46243                 }
46244             }
46245         }
46246         
46247         
46248         
46249         // hides menus... - so this cant be on a menu...
46250         Roo.menu.MenuMgr.hideAll();
46251
46252         //this.editorsyncValue();
46253     },
46254    
46255     
46256     createFontOptions : function(){
46257         var buf = [], fs = this.fontFamilies, ff, lc;
46258         
46259         
46260         
46261         for(var i = 0, len = fs.length; i< len; i++){
46262             ff = fs[i];
46263             lc = ff.toLowerCase();
46264             buf.push(
46265                 '<option value="',lc,'" style="font-family:',ff,';"',
46266                     (this.defaultFont == lc ? ' selected="true">' : '>'),
46267                     ff,
46268                 '</option>'
46269             );
46270         }
46271         return buf.join('');
46272     },
46273     
46274     toggleSourceEdit : function(sourceEditMode){
46275         
46276         Roo.log("toolbar toogle");
46277         if(sourceEditMode === undefined){
46278             sourceEditMode = !this.sourceEditMode;
46279         }
46280         this.sourceEditMode = sourceEditMode === true;
46281         var btn = this.tb.items.get(this.editorcore.frameId +'-sourceedit');
46282         // just toggle the button?
46283         if(btn.pressed !== this.sourceEditMode){
46284             btn.toggle(this.sourceEditMode);
46285             return;
46286         }
46287         
46288         if(sourceEditMode){
46289             Roo.log("disabling buttons");
46290             this.tb.items.each(function(item){
46291                 if(item.cmd != 'sourceedit' && (typeof(item.cls) != 'undefined' && item.cls.indexOf('x-init-enable') === -1)){
46292                     item.disable();
46293                 }
46294             });
46295           
46296         }else{
46297             Roo.log("enabling buttons");
46298             if(this.editorcore.initialized){
46299                 this.tb.items.each(function(item){
46300                     item.enable();
46301                 });
46302             }
46303             
46304         }
46305         Roo.log("calling toggole on editor");
46306         // tell the editor that it's been pressed..
46307         this.editor.toggleSourceEdit(sourceEditMode);
46308        
46309     },
46310      /**
46311      * Object collection of toolbar tooltips for the buttons in the editor. The key
46312      * is the command id associated with that button and the value is a valid QuickTips object.
46313      * For example:
46314 <pre><code>
46315 {
46316     bold : {
46317         title: 'Bold (Ctrl+B)',
46318         text: 'Make the selected text bold.',
46319         cls: 'x-html-editor-tip'
46320     },
46321     italic : {
46322         title: 'Italic (Ctrl+I)',
46323         text: 'Make the selected text italic.',
46324         cls: 'x-html-editor-tip'
46325     },
46326     ...
46327 </code></pre>
46328     * @type Object
46329      */
46330     buttonTips : {
46331         bold : {
46332             title: 'Bold (Ctrl+B)',
46333             text: 'Make the selected text bold.',
46334             cls: 'x-html-editor-tip'
46335         },
46336         italic : {
46337             title: 'Italic (Ctrl+I)',
46338             text: 'Make the selected text italic.',
46339             cls: 'x-html-editor-tip'
46340         },
46341         underline : {
46342             title: 'Underline (Ctrl+U)',
46343             text: 'Underline the selected text.',
46344             cls: 'x-html-editor-tip'
46345         },
46346         strikethrough : {
46347             title: 'Strikethrough',
46348             text: 'Strikethrough the selected text.',
46349             cls: 'x-html-editor-tip'
46350         },
46351         increasefontsize : {
46352             title: 'Grow Text',
46353             text: 'Increase the font size.',
46354             cls: 'x-html-editor-tip'
46355         },
46356         decreasefontsize : {
46357             title: 'Shrink Text',
46358             text: 'Decrease the font size.',
46359             cls: 'x-html-editor-tip'
46360         },
46361         backcolor : {
46362             title: 'Text Highlight Color',
46363             text: 'Change the background color of the selected text.',
46364             cls: 'x-html-editor-tip'
46365         },
46366         forecolor : {
46367             title: 'Font Color',
46368             text: 'Change the color of the selected text.',
46369             cls: 'x-html-editor-tip'
46370         },
46371         justifyleft : {
46372             title: 'Align Text Left',
46373             text: 'Align text to the left.',
46374             cls: 'x-html-editor-tip'
46375         },
46376         justifycenter : {
46377             title: 'Center Text',
46378             text: 'Center text in the editor.',
46379             cls: 'x-html-editor-tip'
46380         },
46381         justifyright : {
46382             title: 'Align Text Right',
46383             text: 'Align text to the right.',
46384             cls: 'x-html-editor-tip'
46385         },
46386         insertunorderedlist : {
46387             title: 'Bullet List',
46388             text: 'Start a bulleted list.',
46389             cls: 'x-html-editor-tip'
46390         },
46391         insertorderedlist : {
46392             title: 'Numbered List',
46393             text: 'Start a numbered list.',
46394             cls: 'x-html-editor-tip'
46395         },
46396         createlink : {
46397             title: 'Hyperlink',
46398             text: 'Make the selected text a hyperlink.',
46399             cls: 'x-html-editor-tip'
46400         },
46401         sourceedit : {
46402             title: 'Source Edit',
46403             text: 'Switch to source editing mode.',
46404             cls: 'x-html-editor-tip'
46405         }
46406     },
46407     // private
46408     onDestroy : function(){
46409         if(this.rendered){
46410             
46411             this.tb.items.each(function(item){
46412                 if(item.menu){
46413                     item.menu.removeAll();
46414                     if(item.menu.el){
46415                         item.menu.el.destroy();
46416                     }
46417                 }
46418                 item.destroy();
46419             });
46420              
46421         }
46422     },
46423     onFirstFocus: function() {
46424         this.tb.items.each(function(item){
46425            item.enable();
46426         });
46427     }
46428 });
46429
46430
46431
46432
46433 // <script type="text/javascript">
46434 /*
46435  * Based on
46436  * Ext JS Library 1.1.1
46437  * Copyright(c) 2006-2007, Ext JS, LLC.
46438  *  
46439  
46440  */
46441
46442  
46443 /**
46444  * @class Roo.form.HtmlEditor.ToolbarContext
46445  * Context Toolbar
46446  * 
46447  * Usage:
46448  *
46449  new Roo.form.HtmlEditor({
46450     ....
46451     toolbars : [
46452         { xtype: 'ToolbarStandard', styles : {} }
46453         { xtype: 'ToolbarContext', disable : {} }
46454     ]
46455 })
46456
46457      
46458  * 
46459  * @config : {Object} disable List of elements to disable.. (not done yet.)
46460  * @config : {Object} styles  Map of styles available.
46461  * 
46462  */
46463
46464 Roo.form.HtmlEditor.ToolbarContext = function(config)
46465 {
46466     
46467     Roo.apply(this, config);
46468     //Roo.form.HtmlEditorToolbar1.superclass.constructor.call(this, editor.wrap.dom.firstChild, [], config);
46469     // dont call parent... till later.
46470     this.styles = this.styles || {};
46471 }
46472
46473  
46474
46475 Roo.form.HtmlEditor.ToolbarContext.types = {
46476     'IMG' : {
46477         width : {
46478             title: "Width",
46479             width: 40
46480         },
46481         height:  {
46482             title: "Height",
46483             width: 40
46484         },
46485         align: {
46486             title: "Align",
46487             opts : [ [""],[ "left"],[ "right"],[ "center"],[ "top"]],
46488             width : 80
46489             
46490         },
46491         border: {
46492             title: "Border",
46493             width: 40
46494         },
46495         alt: {
46496             title: "Alt",
46497             width: 120
46498         },
46499         src : {
46500             title: "Src",
46501             width: 220
46502         }
46503         
46504     },
46505     'A' : {
46506         name : {
46507             title: "Name",
46508             width: 50
46509         },
46510         target:  {
46511             title: "Target",
46512             width: 120
46513         },
46514         href:  {
46515             title: "Href",
46516             width: 220
46517         } // border?
46518         
46519     },
46520     'TABLE' : {
46521         rows : {
46522             title: "Rows",
46523             width: 20
46524         },
46525         cols : {
46526             title: "Cols",
46527             width: 20
46528         },
46529         width : {
46530             title: "Width",
46531             width: 40
46532         },
46533         height : {
46534             title: "Height",
46535             width: 40
46536         },
46537         border : {
46538             title: "Border",
46539             width: 20
46540         }
46541     },
46542     'TD' : {
46543         width : {
46544             title: "Width",
46545             width: 40
46546         },
46547         height : {
46548             title: "Height",
46549             width: 40
46550         },   
46551         align: {
46552             title: "Align",
46553             opts : [[""],[ "left"],[ "center"],[ "right"],[ "justify"],[ "char"]],
46554             width: 80
46555         },
46556         valign: {
46557             title: "Valign",
46558             opts : [[""],[ "top"],[ "middle"],[ "bottom"],[ "baseline"]],
46559             width: 80
46560         },
46561         colspan: {
46562             title: "Colspan",
46563             width: 20
46564             
46565         },
46566          'font-family'  : {
46567             title : "Font",
46568             style : 'fontFamily',
46569             displayField: 'display',
46570             optname : 'font-family',
46571             width: 140
46572         }
46573     },
46574     'INPUT' : {
46575         name : {
46576             title: "name",
46577             width: 120
46578         },
46579         value : {
46580             title: "Value",
46581             width: 120
46582         },
46583         width : {
46584             title: "Width",
46585             width: 40
46586         }
46587     },
46588     'LABEL' : {
46589         'for' : {
46590             title: "For",
46591             width: 120
46592         }
46593     },
46594     'TEXTAREA' : {
46595           name : {
46596             title: "name",
46597             width: 120
46598         },
46599         rows : {
46600             title: "Rows",
46601             width: 20
46602         },
46603         cols : {
46604             title: "Cols",
46605             width: 20
46606         }
46607     },
46608     'SELECT' : {
46609         name : {
46610             title: "name",
46611             width: 120
46612         },
46613         selectoptions : {
46614             title: "Options",
46615             width: 200
46616         }
46617     },
46618     
46619     // should we really allow this??
46620     // should this just be 
46621     'BODY' : {
46622         title : {
46623             title: "Title",
46624             width: 200,
46625             disabled : true
46626         }
46627     },
46628     'SPAN' : {
46629         'font-family'  : {
46630             title : "Font",
46631             style : 'fontFamily',
46632             displayField: 'display',
46633             optname : 'font-family',
46634             width: 140
46635         }
46636     },
46637     'DIV' : {
46638         'font-family'  : {
46639             title : "Font",
46640             style : 'fontFamily',
46641             displayField: 'display',
46642             optname : 'font-family',
46643             width: 140
46644         }
46645     },
46646      'P' : {
46647         'font-family'  : {
46648             title : "Font",
46649             style : 'fontFamily',
46650             displayField: 'display',
46651             optname : 'font-family',
46652             width: 140
46653         }
46654     },
46655     
46656     '*' : {
46657         // empty..
46658     }
46659
46660 };
46661
46662 // this should be configurable.. - you can either set it up using stores, or modify options somehwere..
46663 Roo.form.HtmlEditor.ToolbarContext.stores = false;
46664
46665 Roo.form.HtmlEditor.ToolbarContext.options = {
46666         'font-family'  : [ 
46667                 [ 'Helvetica,Arial,sans-serif', 'Helvetica'],
46668                 [ 'Courier New', 'Courier New'],
46669                 [ 'Tahoma', 'Tahoma'],
46670                 [ 'Times New Roman,serif', 'Times'],
46671                 [ 'Verdana','Verdana' ]
46672         ]
46673 };
46674
46675 // fixme - these need to be configurable..
46676  
46677
46678 //Roo.form.HtmlEditor.ToolbarContext.types
46679
46680
46681 Roo.apply(Roo.form.HtmlEditor.ToolbarContext.prototype,  {
46682     
46683     tb: false,
46684     
46685     rendered: false,
46686     
46687     editor : false,
46688     editorcore : false,
46689     /**
46690      * @cfg {Object} disable  List of toolbar elements to disable
46691          
46692      */
46693     disable : false,
46694     /**
46695      * @cfg {Object} styles List of styles 
46696      *    eg. { '*' : [ 'headline' ] , 'TD' : [ 'underline', 'double-underline' ] } 
46697      *
46698      * These must be defined in the page, so they get rendered correctly..
46699      * .headline { }
46700      * TD.underline { }
46701      * 
46702      */
46703     styles : false,
46704     
46705     options: false,
46706     
46707     toolbars : false,
46708     
46709     init : function(editor)
46710     {
46711         this.editor = editor;
46712         this.editorcore = editor.editorcore ? editor.editorcore : editor;
46713         var editorcore = this.editorcore;
46714         
46715         var fid = editorcore.frameId;
46716         var etb = this;
46717         function btn(id, toggle, handler){
46718             var xid = fid + '-'+ id ;
46719             return {
46720                 id : xid,
46721                 cmd : id,
46722                 cls : 'x-btn-icon x-edit-'+id,
46723                 enableToggle:toggle !== false,
46724                 scope: editorcore, // was editor...
46725                 handler:handler||editorcore.relayBtnCmd,
46726                 clickEvent:'mousedown',
46727                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
46728                 tabIndex:-1
46729             };
46730         }
46731         // create a new element.
46732         var wdiv = editor.wrap.createChild({
46733                 tag: 'div'
46734             }, editor.wrap.dom.firstChild.nextSibling, true);
46735         
46736         // can we do this more than once??
46737         
46738          // stop form submits
46739       
46740  
46741         // disable everything...
46742         var ty= Roo.form.HtmlEditor.ToolbarContext.types;
46743         this.toolbars = {};
46744            
46745         for (var i in  ty) {
46746           
46747             this.toolbars[i] = this.buildToolbar(ty[i],i);
46748         }
46749         this.tb = this.toolbars.BODY;
46750         this.tb.el.show();
46751         this.buildFooter();
46752         this.footer.show();
46753         editor.on('hide', function( ) { this.footer.hide() }, this);
46754         editor.on('show', function( ) { this.footer.show() }, this);
46755         
46756          
46757         this.rendered = true;
46758         
46759         // the all the btns;
46760         editor.on('editorevent', this.updateToolbar, this);
46761         // other toolbars need to implement this..
46762         //editor.on('editmodechange', this.updateToolbar, this);
46763     },
46764     
46765     
46766     
46767     /**
46768      * Protected method that will not generally be called directly. It triggers
46769      * a toolbar update by reading the markup state of the current selection in the editor.
46770      *
46771      * Note you can force an update by calling on('editorevent', scope, false)
46772      */
46773     updateToolbar: function(editor,ev,sel){
46774
46775         //Roo.log(ev);
46776         // capture mouse up - this is handy for selecting images..
46777         // perhaps should go somewhere else...
46778         if(!this.editorcore.activated){
46779              this.editor.onFirstFocus();
46780             return;
46781         }
46782         
46783         
46784         
46785         // http://developer.yahoo.com/yui/docs/simple-editor.js.html
46786         // selectNode - might want to handle IE?
46787         if (ev &&
46788             (ev.type == 'mouseup' || ev.type == 'click' ) &&
46789             ev.target && ev.target.tagName == 'IMG') {
46790             // they have click on an image...
46791             // let's see if we can change the selection...
46792             sel = ev.target;
46793          
46794               var nodeRange = sel.ownerDocument.createRange();
46795             try {
46796                 nodeRange.selectNode(sel);
46797             } catch (e) {
46798                 nodeRange.selectNodeContents(sel);
46799             }
46800             //nodeRange.collapse(true);
46801             var s = this.editorcore.win.getSelection();
46802             s.removeAllRanges();
46803             s.addRange(nodeRange);
46804         }  
46805         
46806       
46807         var updateFooter = sel ? false : true;
46808         
46809         
46810         var ans = this.editorcore.getAllAncestors();
46811         
46812         // pick
46813         var ty= Roo.form.HtmlEditor.ToolbarContext.types;
46814         
46815         if (!sel) { 
46816             sel = ans.length ? (ans[0] ?  ans[0]  : ans[1]) : this.editorcore.doc.body;
46817             sel = sel ? sel : this.editorcore.doc.body;
46818             sel = sel.tagName.length ? sel : this.editorcore.doc.body;
46819             
46820         }
46821         // pick a menu that exists..
46822         var tn = sel.tagName.toUpperCase();
46823         //sel = typeof(ty[tn]) != 'undefined' ? sel : this.editor.doc.body;
46824         
46825         tn = sel.tagName.toUpperCase();
46826         
46827         var lastSel = this.tb.selectedNode;
46828         
46829         this.tb.selectedNode = sel;
46830         
46831         // if current menu does not match..
46832         
46833         if ((this.tb.name != tn) || (lastSel != this.tb.selectedNode) || ev === false) {
46834                 
46835             this.tb.el.hide();
46836             ///console.log("show: " + tn);
46837             this.tb =  typeof(ty[tn]) != 'undefined' ? this.toolbars[tn] : this.toolbars['*'];
46838             this.tb.el.show();
46839             // update name
46840             this.tb.items.first().el.innerHTML = tn + ':&nbsp;';
46841             
46842             
46843             // update attributes
46844             if (this.tb.fields) {
46845                 this.tb.fields.each(function(e) {
46846                     if (e.stylename) {
46847                         e.setValue(sel.style[e.stylename]);
46848                         return;
46849                     } 
46850                    e.setValue(sel.getAttribute(e.attrname));
46851                 });
46852             }
46853             
46854             var hasStyles = false;
46855             for(var i in this.styles) {
46856                 hasStyles = true;
46857                 break;
46858             }
46859             
46860             // update styles
46861             if (hasStyles) { 
46862                 var st = this.tb.fields.item(0);
46863                 
46864                 st.store.removeAll();
46865                
46866                 
46867                 var cn = sel.className.split(/\s+/);
46868                 
46869                 var avs = [];
46870                 if (this.styles['*']) {
46871                     
46872                     Roo.each(this.styles['*'], function(v) {
46873                         avs.push( [ v , cn.indexOf(v) > -1 ? 1 : 0 ] );         
46874                     });
46875                 }
46876                 if (this.styles[tn]) { 
46877                     Roo.each(this.styles[tn], function(v) {
46878                         avs.push( [ v , cn.indexOf(v) > -1 ? 1 : 0 ] );         
46879                     });
46880                 }
46881                 
46882                 st.store.loadData(avs);
46883                 st.collapse();
46884                 st.setValue(cn);
46885             }
46886             // flag our selected Node.
46887             this.tb.selectedNode = sel;
46888            
46889            
46890             Roo.menu.MenuMgr.hideAll();
46891
46892         }
46893         
46894         if (!updateFooter) {
46895             //this.footDisp.dom.innerHTML = ''; 
46896             return;
46897         }
46898         // update the footer
46899         //
46900         var html = '';
46901         
46902         this.footerEls = ans.reverse();
46903         Roo.each(this.footerEls, function(a,i) {
46904             if (!a) { return; }
46905             html += html.length ? ' &gt; '  :  '';
46906             
46907             html += '<span class="x-ed-loc-' + i + '">' + a.tagName + '</span>';
46908             
46909         });
46910        
46911         // 
46912         var sz = this.footDisp.up('td').getSize();
46913         this.footDisp.dom.style.width = (sz.width -10) + 'px';
46914         this.footDisp.dom.style.marginLeft = '5px';
46915         
46916         this.footDisp.dom.style.overflow = 'hidden';
46917         
46918         this.footDisp.dom.innerHTML = html;
46919             
46920         //this.editorsyncValue();
46921     },
46922      
46923     
46924    
46925        
46926     // private
46927     onDestroy : function(){
46928         if(this.rendered){
46929             
46930             this.tb.items.each(function(item){
46931                 if(item.menu){
46932                     item.menu.removeAll();
46933                     if(item.menu.el){
46934                         item.menu.el.destroy();
46935                     }
46936                 }
46937                 item.destroy();
46938             });
46939              
46940         }
46941     },
46942     onFirstFocus: function() {
46943         // need to do this for all the toolbars..
46944         this.tb.items.each(function(item){
46945            item.enable();
46946         });
46947     },
46948     buildToolbar: function(tlist, nm)
46949     {
46950         var editor = this.editor;
46951         var editorcore = this.editorcore;
46952          // create a new element.
46953         var wdiv = editor.wrap.createChild({
46954                 tag: 'div'
46955             }, editor.wrap.dom.firstChild.nextSibling, true);
46956         
46957        
46958         var tb = new Roo.Toolbar(wdiv);
46959         // add the name..
46960         
46961         tb.add(nm+ ":&nbsp;");
46962         
46963         var styles = [];
46964         for(var i in this.styles) {
46965             styles.push(i);
46966         }
46967         
46968         // styles...
46969         if (styles && styles.length) {
46970             
46971             // this needs a multi-select checkbox...
46972             tb.addField( new Roo.form.ComboBox({
46973                 store: new Roo.data.SimpleStore({
46974                     id : 'val',
46975                     fields: ['val', 'selected'],
46976                     data : [] 
46977                 }),
46978                 name : '-roo-edit-className',
46979                 attrname : 'className',
46980                 displayField: 'val',
46981                 typeAhead: false,
46982                 mode: 'local',
46983                 editable : false,
46984                 triggerAction: 'all',
46985                 emptyText:'Select Style',
46986                 selectOnFocus:true,
46987                 width: 130,
46988                 listeners : {
46989                     'select': function(c, r, i) {
46990                         // initial support only for on class per el..
46991                         tb.selectedNode.className =  r ? r.get('val') : '';
46992                         editorcore.syncValue();
46993                     }
46994                 }
46995     
46996             }));
46997         }
46998         
46999         var tbc = Roo.form.HtmlEditor.ToolbarContext;
47000         var tbops = tbc.options;
47001         
47002         for (var i in tlist) {
47003             
47004             var item = tlist[i];
47005             tb.add(item.title + ":&nbsp;");
47006             
47007             
47008             //optname == used so you can configure the options available..
47009             var opts = item.opts ? item.opts : false;
47010             if (item.optname) {
47011                 opts = tbops[item.optname];
47012            
47013             }
47014             
47015             if (opts) {
47016                 // opts == pulldown..
47017                 tb.addField( new Roo.form.ComboBox({
47018                     store:   typeof(tbc.stores[i]) != 'undefined' ?  Roo.factory(tbc.stores[i],Roo.data) : new Roo.data.SimpleStore({
47019                         id : 'val',
47020                         fields: ['val', 'display'],
47021                         data : opts  
47022                     }),
47023                     name : '-roo-edit-' + i,
47024                     attrname : i,
47025                     stylename : item.style ? item.style : false,
47026                     displayField: item.displayField ? item.displayField : 'val',
47027                     valueField :  'val',
47028                     typeAhead: false,
47029                     mode: typeof(tbc.stores[i]) != 'undefined'  ? 'remote' : 'local',
47030                     editable : false,
47031                     triggerAction: 'all',
47032                     emptyText:'Select',
47033                     selectOnFocus:true,
47034                     width: item.width ? item.width  : 130,
47035                     listeners : {
47036                         'select': function(c, r, i) {
47037                             if (c.stylename) {
47038                                 tb.selectedNode.style[c.stylename] =  r.get('val');
47039                                 return;
47040                             }
47041                             tb.selectedNode.setAttribute(c.attrname, r.get('val'));
47042                         }
47043                     }
47044
47045                 }));
47046                 continue;
47047                     
47048                  
47049                 
47050                 tb.addField( new Roo.form.TextField({
47051                     name: i,
47052                     width: 100,
47053                     //allowBlank:false,
47054                     value: ''
47055                 }));
47056                 continue;
47057             }
47058             tb.addField( new Roo.form.TextField({
47059                 name: '-roo-edit-' + i,
47060                 attrname : i,
47061                 
47062                 width: item.width,
47063                 //allowBlank:true,
47064                 value: '',
47065                 listeners: {
47066                     'change' : function(f, nv, ov) {
47067                         tb.selectedNode.setAttribute(f.attrname, nv);
47068                         editorcore.syncValue();
47069                     }
47070                 }
47071             }));
47072              
47073         }
47074         
47075         var _this = this;
47076         
47077         if(nm == 'BODY'){
47078             tb.addSeparator();
47079         
47080             tb.addButton( {
47081                 text: 'Stylesheets',
47082
47083                 listeners : {
47084                     click : function ()
47085                     {
47086                         _this.editor.fireEvent('stylesheetsclick', _this.editor);
47087                     }
47088                 }
47089             });
47090         }
47091         
47092         tb.addFill();
47093         tb.addButton( {
47094             text: 'Remove Tag',
47095     
47096             listeners : {
47097                 click : function ()
47098                 {
47099                     // remove
47100                     // undo does not work.
47101                      
47102                     var sn = tb.selectedNode;
47103                     
47104                     var pn = sn.parentNode;
47105                     
47106                     var stn =  sn.childNodes[0];
47107                     var en = sn.childNodes[sn.childNodes.length - 1 ];
47108                     while (sn.childNodes.length) {
47109                         var node = sn.childNodes[0];
47110                         sn.removeChild(node);
47111                         //Roo.log(node);
47112                         pn.insertBefore(node, sn);
47113                         
47114                     }
47115                     pn.removeChild(sn);
47116                     var range = editorcore.createRange();
47117         
47118                     range.setStart(stn,0);
47119                     range.setEnd(en,0); //????
47120                     //range.selectNode(sel);
47121                     
47122                     
47123                     var selection = editorcore.getSelection();
47124                     selection.removeAllRanges();
47125                     selection.addRange(range);
47126                     
47127                     
47128                     
47129                     //_this.updateToolbar(null, null, pn);
47130                     _this.updateToolbar(null, null, null);
47131                     _this.footDisp.dom.innerHTML = ''; 
47132                 }
47133             }
47134             
47135                     
47136                 
47137             
47138         });
47139         
47140         
47141         tb.el.on('click', function(e){
47142             e.preventDefault(); // what does this do?
47143         });
47144         tb.el.setVisibilityMode( Roo.Element.DISPLAY);
47145         tb.el.hide();
47146         tb.name = nm;
47147         // dont need to disable them... as they will get hidden
47148         return tb;
47149          
47150         
47151     },
47152     buildFooter : function()
47153     {
47154         
47155         var fel = this.editor.wrap.createChild();
47156         this.footer = new Roo.Toolbar(fel);
47157         // toolbar has scrolly on left / right?
47158         var footDisp= new Roo.Toolbar.Fill();
47159         var _t = this;
47160         this.footer.add(
47161             {
47162                 text : '&lt;',
47163                 xtype: 'Button',
47164                 handler : function() {
47165                     _t.footDisp.scrollTo('left',0,true)
47166                 }
47167             }
47168         );
47169         this.footer.add( footDisp );
47170         this.footer.add( 
47171             {
47172                 text : '&gt;',
47173                 xtype: 'Button',
47174                 handler : function() {
47175                     // no animation..
47176                     _t.footDisp.select('span').last().scrollIntoView(_t.footDisp,true);
47177                 }
47178             }
47179         );
47180         var fel = Roo.get(footDisp.el);
47181         fel.addClass('x-editor-context');
47182         this.footDispWrap = fel; 
47183         this.footDispWrap.overflow  = 'hidden';
47184         
47185         this.footDisp = fel.createChild();
47186         this.footDispWrap.on('click', this.onContextClick, this)
47187         
47188         
47189     },
47190     onContextClick : function (ev,dom)
47191     {
47192         ev.preventDefault();
47193         var  cn = dom.className;
47194         //Roo.log(cn);
47195         if (!cn.match(/x-ed-loc-/)) {
47196             return;
47197         }
47198         var n = cn.split('-').pop();
47199         var ans = this.footerEls;
47200         var sel = ans[n];
47201         
47202          // pick
47203         var range = this.editorcore.createRange();
47204         
47205         range.selectNodeContents(sel);
47206         //range.selectNode(sel);
47207         
47208         
47209         var selection = this.editorcore.getSelection();
47210         selection.removeAllRanges();
47211         selection.addRange(range);
47212         
47213         
47214         
47215         this.updateToolbar(null, null, sel);
47216         
47217         
47218     }
47219     
47220     
47221     
47222     
47223     
47224 });
47225
47226
47227
47228
47229
47230 /*
47231  * Based on:
47232  * Ext JS Library 1.1.1
47233  * Copyright(c) 2006-2007, Ext JS, LLC.
47234  *
47235  * Originally Released Under LGPL - original licence link has changed is not relivant.
47236  *
47237  * Fork - LGPL
47238  * <script type="text/javascript">
47239  */
47240  
47241 /**
47242  * @class Roo.form.BasicForm
47243  * @extends Roo.util.Observable
47244  * Supplies the functionality to do "actions" on forms and initialize Roo.form.Field types on existing markup.
47245  * @constructor
47246  * @param {String/HTMLElement/Roo.Element} el The form element or its id
47247  * @param {Object} config Configuration options
47248  */
47249 Roo.form.BasicForm = function(el, config){
47250     this.allItems = [];
47251     this.childForms = [];
47252     Roo.apply(this, config);
47253     /*
47254      * The Roo.form.Field items in this form.
47255      * @type MixedCollection
47256      */
47257      
47258      
47259     this.items = new Roo.util.MixedCollection(false, function(o){
47260         return o.id || (o.id = Roo.id());
47261     });
47262     this.addEvents({
47263         /**
47264          * @event beforeaction
47265          * Fires before any action is performed. Return false to cancel the action.
47266          * @param {Form} this
47267          * @param {Action} action The action to be performed
47268          */
47269         beforeaction: true,
47270         /**
47271          * @event actionfailed
47272          * Fires when an action fails.
47273          * @param {Form} this
47274          * @param {Action} action The action that failed
47275          */
47276         actionfailed : true,
47277         /**
47278          * @event actioncomplete
47279          * Fires when an action is completed.
47280          * @param {Form} this
47281          * @param {Action} action The action that completed
47282          */
47283         actioncomplete : true
47284     });
47285     if(el){
47286         this.initEl(el);
47287     }
47288     Roo.form.BasicForm.superclass.constructor.call(this);
47289     
47290     Roo.form.BasicForm.popover.apply();
47291 };
47292
47293 Roo.extend(Roo.form.BasicForm, Roo.util.Observable, {
47294     /**
47295      * @cfg {String} method
47296      * The request method to use (GET or POST) for form actions if one isn't supplied in the action options.
47297      */
47298     /**
47299      * @cfg {DataReader} reader
47300      * An Roo.data.DataReader (e.g. {@link Roo.data.XmlReader}) to be used to read data when executing "load" actions.
47301      * This is optional as there is built-in support for processing JSON.
47302      */
47303     /**
47304      * @cfg {DataReader} errorReader
47305      * An Roo.data.DataReader (e.g. {@link Roo.data.XmlReader}) to be used to read data when reading validation errors on "submit" actions.
47306      * This is completely optional as there is built-in support for processing JSON.
47307      */
47308     /**
47309      * @cfg {String} url
47310      * The URL to use for form actions if one isn't supplied in the action options.
47311      */
47312     /**
47313      * @cfg {Boolean} fileUpload
47314      * Set to true if this form is a file upload.
47315      */
47316      
47317     /**
47318      * @cfg {Object} baseParams
47319      * Parameters to pass with all requests. e.g. baseParams: {id: '123', foo: 'bar'}.
47320      */
47321      /**
47322      
47323     /**
47324      * @cfg {Number} timeout Timeout for form actions in seconds (default is 30 seconds).
47325      */
47326     timeout: 30,
47327
47328     // private
47329     activeAction : null,
47330
47331     /**
47332      * @cfg {Boolean} trackResetOnLoad If set to true, form.reset() resets to the last loaded
47333      * or setValues() data instead of when the form was first created.
47334      */
47335     trackResetOnLoad : false,
47336     
47337     
47338     /**
47339      * childForms - used for multi-tab forms
47340      * @type {Array}
47341      */
47342     childForms : false,
47343     
47344     /**
47345      * allItems - full list of fields.
47346      * @type {Array}
47347      */
47348     allItems : false,
47349     
47350     /**
47351      * By default wait messages are displayed with Roo.MessageBox.wait. You can target a specific
47352      * element by passing it or its id or mask the form itself by passing in true.
47353      * @type Mixed
47354      */
47355     waitMsgTarget : false,
47356     
47357     /**
47358      * @type Boolean
47359      */
47360     disableMask : false,
47361     
47362     /**
47363      * @cfg {Boolean} errorMask (true|false) default false
47364      */
47365     errorMask : false,
47366     
47367     /**
47368      * @cfg {Number} maskOffset Default 100
47369      */
47370     maskOffset : 100,
47371
47372     // private
47373     initEl : function(el){
47374         this.el = Roo.get(el);
47375         this.id = this.el.id || Roo.id();
47376         this.el.on('submit', this.onSubmit, this);
47377         this.el.addClass('x-form');
47378     },
47379
47380     // private
47381     onSubmit : function(e){
47382         e.stopEvent();
47383     },
47384
47385     /**
47386      * Returns true if client-side validation on the form is successful.
47387      * @return Boolean
47388      */
47389     isValid : function(){
47390         var valid = true;
47391         var target = false;
47392         this.items.each(function(f){
47393             if(f.validate()){
47394                 return;
47395             }
47396             
47397             valid = false;
47398                 
47399             if(!target && f.el.isVisible(true)){
47400                 target = f;
47401             }
47402         });
47403         
47404         if(this.errorMask && !valid){
47405             Roo.form.BasicForm.popover.mask(this, target);
47406         }
47407         
47408         return valid;
47409     },
47410
47411     /**
47412      * DEPRICATED Returns true if any fields in this form have changed since their original load. 
47413      * @return Boolean
47414      */
47415     isDirty : function(){
47416         var dirty = false;
47417         this.items.each(function(f){
47418            if(f.isDirty()){
47419                dirty = true;
47420                return false;
47421            }
47422         });
47423         return dirty;
47424     },
47425     
47426     /**
47427      * Returns true if any fields in this form have changed since their original load. (New version)
47428      * @return Boolean
47429      */
47430     
47431     hasChanged : function()
47432     {
47433         var dirty = false;
47434         this.items.each(function(f){
47435            if(f.hasChanged()){
47436                dirty = true;
47437                return false;
47438            }
47439         });
47440         return dirty;
47441         
47442     },
47443     /**
47444      * Resets all hasChanged to 'false' -
47445      * The old 'isDirty' used 'original value..' however this breaks reset() and a few other things.
47446      * So hasChanged storage is only to be used for this purpose
47447      * @return Boolean
47448      */
47449     resetHasChanged : function()
47450     {
47451         this.items.each(function(f){
47452            f.resetHasChanged();
47453         });
47454         
47455     },
47456     
47457     
47458     /**
47459      * Performs a predefined action (submit or load) or custom actions you define on this form.
47460      * @param {String} actionName The name of the action type
47461      * @param {Object} options (optional) The options to pass to the action.  All of the config options listed
47462      * below are supported by both the submit and load actions unless otherwise noted (custom actions could also
47463      * accept other config options):
47464      * <pre>
47465 Property          Type             Description
47466 ----------------  ---------------  ----------------------------------------------------------------------------------
47467 url               String           The url for the action (defaults to the form's url)
47468 method            String           The form method to use (defaults to the form's method, or POST if not defined)
47469 params            String/Object    The params to pass (defaults to the form's baseParams, or none if not defined)
47470 clientValidation  Boolean          Applies to submit only.  Pass true to call form.isValid() prior to posting to
47471                                    validate the form on the client (defaults to false)
47472      * </pre>
47473      * @return {BasicForm} this
47474      */
47475     doAction : function(action, options){
47476         if(typeof action == 'string'){
47477             action = new Roo.form.Action.ACTION_TYPES[action](this, options);
47478         }
47479         if(this.fireEvent('beforeaction', this, action) !== false){
47480             this.beforeAction(action);
47481             action.run.defer(100, action);
47482         }
47483         return this;
47484     },
47485
47486     /**
47487      * Shortcut to do a submit action.
47488      * @param {Object} options The options to pass to the action (see {@link #doAction} for details)
47489      * @return {BasicForm} this
47490      */
47491     submit : function(options){
47492         this.doAction('submit', options);
47493         return this;
47494     },
47495
47496     /**
47497      * Shortcut to do a load action.
47498      * @param {Object} options The options to pass to the action (see {@link #doAction} for details)
47499      * @return {BasicForm} this
47500      */
47501     load : function(options){
47502         this.doAction('load', options);
47503         return this;
47504     },
47505
47506     /**
47507      * Persists the values in this form into the passed Roo.data.Record object in a beginEdit/endEdit block.
47508      * @param {Record} record The record to edit
47509      * @return {BasicForm} this
47510      */
47511     updateRecord : function(record){
47512         record.beginEdit();
47513         var fs = record.fields;
47514         fs.each(function(f){
47515             var field = this.findField(f.name);
47516             if(field){
47517                 record.set(f.name, field.getValue());
47518             }
47519         }, this);
47520         record.endEdit();
47521         return this;
47522     },
47523
47524     /**
47525      * Loads an Roo.data.Record into this form.
47526      * @param {Record} record The record to load
47527      * @return {BasicForm} this
47528      */
47529     loadRecord : function(record){
47530         this.setValues(record.data);
47531         return this;
47532     },
47533
47534     // private
47535     beforeAction : function(action){
47536         var o = action.options;
47537         
47538         if(!this.disableMask) {
47539             if(this.waitMsgTarget === true){
47540                 this.el.mask(o.waitMsg || "Sending", 'x-mask-loading');
47541             }else if(this.waitMsgTarget){
47542                 this.waitMsgTarget = Roo.get(this.waitMsgTarget);
47543                 this.waitMsgTarget.mask(o.waitMsg || "Sending", 'x-mask-loading');
47544             }else {
47545                 Roo.MessageBox.wait(o.waitMsg || "Sending", o.waitTitle || this.waitTitle || 'Please Wait...');
47546             }
47547         }
47548         
47549          
47550     },
47551
47552     // private
47553     afterAction : function(action, success){
47554         this.activeAction = null;
47555         var o = action.options;
47556         
47557         if(!this.disableMask) {
47558             if(this.waitMsgTarget === true){
47559                 this.el.unmask();
47560             }else if(this.waitMsgTarget){
47561                 this.waitMsgTarget.unmask();
47562             }else{
47563                 Roo.MessageBox.updateProgress(1);
47564                 Roo.MessageBox.hide();
47565             }
47566         }
47567         
47568         if(success){
47569             if(o.reset){
47570                 this.reset();
47571             }
47572             Roo.callback(o.success, o.scope, [this, action]);
47573             this.fireEvent('actioncomplete', this, action);
47574             
47575         }else{
47576             
47577             // failure condition..
47578             // we have a scenario where updates need confirming.
47579             // eg. if a locking scenario exists..
47580             // we look for { errors : { needs_confirm : true }} in the response.
47581             if (
47582                 (typeof(action.result) != 'undefined')  &&
47583                 (typeof(action.result.errors) != 'undefined')  &&
47584                 (typeof(action.result.errors.needs_confirm) != 'undefined')
47585            ){
47586                 var _t = this;
47587                 Roo.MessageBox.confirm(
47588                     "Change requires confirmation",
47589                     action.result.errorMsg,
47590                     function(r) {
47591                         if (r != 'yes') {
47592                             return;
47593                         }
47594                         _t.doAction('submit', { params :  { _submit_confirmed : 1 } }  );
47595                     }
47596                     
47597                 );
47598                 
47599                 
47600                 
47601                 return;
47602             }
47603             
47604             Roo.callback(o.failure, o.scope, [this, action]);
47605             // show an error message if no failed handler is set..
47606             if (!this.hasListener('actionfailed')) {
47607                 Roo.MessageBox.alert("Error",
47608                     (typeof(action.result) != 'undefined' && typeof(action.result.errorMsg) != 'undefined') ?
47609                         action.result.errorMsg :
47610                         "Saving Failed, please check your entries or try again"
47611                 );
47612             }
47613             
47614             this.fireEvent('actionfailed', this, action);
47615         }
47616         
47617     },
47618
47619     /**
47620      * Find a Roo.form.Field in this form by id, dataIndex, name or hiddenName
47621      * @param {String} id The value to search for
47622      * @return Field
47623      */
47624     findField : function(id){
47625         var field = this.items.get(id);
47626         if(!field){
47627             this.items.each(function(f){
47628                 if(f.isFormField && (f.dataIndex == id || f.id == id || f.getName() == id)){
47629                     field = f;
47630                     return false;
47631                 }
47632             });
47633         }
47634         return field || null;
47635     },
47636
47637     /**
47638      * Add a secondary form to this one, 
47639      * Used to provide tabbed forms. One form is primary, with hidden values 
47640      * which mirror the elements from the other forms.
47641      * 
47642      * @param {Roo.form.Form} form to add.
47643      * 
47644      */
47645     addForm : function(form)
47646     {
47647        
47648         if (this.childForms.indexOf(form) > -1) {
47649             // already added..
47650             return;
47651         }
47652         this.childForms.push(form);
47653         var n = '';
47654         Roo.each(form.allItems, function (fe) {
47655             
47656             n = typeof(fe.getName) == 'undefined' ? fe.name : fe.getName();
47657             if (this.findField(n)) { // already added..
47658                 return;
47659             }
47660             var add = new Roo.form.Hidden({
47661                 name : n
47662             });
47663             add.render(this.el);
47664             
47665             this.add( add );
47666         }, this);
47667         
47668     },
47669     /**
47670      * Mark fields in this form invalid in bulk.
47671      * @param {Array/Object} errors Either an array in the form [{id:'fieldId', msg:'The message'},...] or an object hash of {id: msg, id2: msg2}
47672      * @return {BasicForm} this
47673      */
47674     markInvalid : function(errors){
47675         if(errors instanceof Array){
47676             for(var i = 0, len = errors.length; i < len; i++){
47677                 var fieldError = errors[i];
47678                 var f = this.findField(fieldError.id);
47679                 if(f){
47680                     f.markInvalid(fieldError.msg);
47681                 }
47682             }
47683         }else{
47684             var field, id;
47685             for(id in errors){
47686                 if(typeof errors[id] != 'function' && (field = this.findField(id))){
47687                     field.markInvalid(errors[id]);
47688                 }
47689             }
47690         }
47691         Roo.each(this.childForms || [], function (f) {
47692             f.markInvalid(errors);
47693         });
47694         
47695         return this;
47696     },
47697
47698     /**
47699      * Set values for fields in this form in bulk.
47700      * @param {Array/Object} values Either an array in the form [{id:'fieldId', value:'foo'},...] or an object hash of {id: value, id2: value2}
47701      * @return {BasicForm} this
47702      */
47703     setValues : function(values){
47704         if(values instanceof Array){ // array of objects
47705             for(var i = 0, len = values.length; i < len; i++){
47706                 var v = values[i];
47707                 var f = this.findField(v.id);
47708                 if(f){
47709                     f.setValue(v.value);
47710                     if(this.trackResetOnLoad){
47711                         f.originalValue = f.getValue();
47712                     }
47713                 }
47714             }
47715         }else{ // object hash
47716             var field, id;
47717             for(id in values){
47718                 if(typeof values[id] != 'function' && (field = this.findField(id))){
47719                     
47720                     if (field.setFromData && 
47721                         field.valueField && 
47722                         field.displayField &&
47723                         // combos' with local stores can 
47724                         // be queried via setValue()
47725                         // to set their value..
47726                         (field.store && !field.store.isLocal)
47727                         ) {
47728                         // it's a combo
47729                         var sd = { };
47730                         sd[field.valueField] = typeof(values[field.hiddenName]) == 'undefined' ? '' : values[field.hiddenName];
47731                         sd[field.displayField] = typeof(values[field.name]) == 'undefined' ? '' : values[field.name];
47732                         field.setFromData(sd);
47733                         
47734                     } else {
47735                         field.setValue(values[id]);
47736                     }
47737                     
47738                     
47739                     if(this.trackResetOnLoad){
47740                         field.originalValue = field.getValue();
47741                     }
47742                 }
47743             }
47744         }
47745         this.resetHasChanged();
47746         
47747         
47748         Roo.each(this.childForms || [], function (f) {
47749             f.setValues(values);
47750             f.resetHasChanged();
47751         });
47752                 
47753         return this;
47754     },
47755  
47756     /**
47757      * Returns the fields in this form as an object with key/value pairs. If multiple fields exist with the same name
47758      * they are returned as an array.
47759      * @param {Boolean} asString
47760      * @return {Object}
47761      */
47762     getValues : function(asString){
47763         if (this.childForms) {
47764             // copy values from the child forms
47765             Roo.each(this.childForms, function (f) {
47766                 this.setValues(f.getValues());
47767             }, this);
47768         }
47769         
47770         // use formdata
47771         if (typeof(FormData) != 'undefined' && asString !== true) {
47772             var fd = (new FormData(this.el.dom)).entries();
47773             var ret = {};
47774             var ent = fd.next();
47775             while (!ent.done) {
47776                 ret[ent.value[0]] = ent.value[1]; // not sure how this will handle duplicates..
47777                 ent = fd.next();
47778             };
47779             return ret;
47780         }
47781         
47782         
47783         var fs = Roo.lib.Ajax.serializeForm(this.el.dom);
47784         if(asString === true){
47785             return fs;
47786         }
47787         return Roo.urlDecode(fs);
47788     },
47789     
47790     /**
47791      * Returns the fields in this form as an object with key/value pairs. 
47792      * This differs from getValues as it calls getValue on each child item, rather than using dom data.
47793      * @return {Object}
47794      */
47795     getFieldValues : function(with_hidden)
47796     {
47797         if (this.childForms) {
47798             // copy values from the child forms
47799             // should this call getFieldValues - probably not as we do not currently copy
47800             // hidden fields when we generate..
47801             Roo.each(this.childForms, function (f) {
47802                 this.setValues(f.getValues());
47803             }, this);
47804         }
47805         
47806         var ret = {};
47807         this.items.each(function(f){
47808             if (!f.getName()) {
47809                 return;
47810             }
47811             var v = f.getValue();
47812             if (f.inputType =='radio') {
47813                 if (typeof(ret[f.getName()]) == 'undefined') {
47814                     ret[f.getName()] = ''; // empty..
47815                 }
47816                 
47817                 if (!f.el.dom.checked) {
47818                     return;
47819                     
47820                 }
47821                 v = f.el.dom.value;
47822                 
47823             }
47824             
47825             // not sure if this supported any more..
47826             if ((typeof(v) == 'object') && f.getRawValue) {
47827                 v = f.getRawValue() ; // dates..
47828             }
47829             // combo boxes where name != hiddenName...
47830             if (f.name != f.getName()) {
47831                 ret[f.name] = f.getRawValue();
47832             }
47833             ret[f.getName()] = v;
47834         });
47835         
47836         return ret;
47837     },
47838
47839     /**
47840      * Clears all invalid messages in this form.
47841      * @return {BasicForm} this
47842      */
47843     clearInvalid : function(){
47844         this.items.each(function(f){
47845            f.clearInvalid();
47846         });
47847         
47848         Roo.each(this.childForms || [], function (f) {
47849             f.clearInvalid();
47850         });
47851         
47852         
47853         return this;
47854     },
47855
47856     /**
47857      * Resets this form.
47858      * @return {BasicForm} this
47859      */
47860     reset : function(){
47861         this.items.each(function(f){
47862             f.reset();
47863         });
47864         
47865         Roo.each(this.childForms || [], function (f) {
47866             f.reset();
47867         });
47868         this.resetHasChanged();
47869         
47870         return this;
47871     },
47872
47873     /**
47874      * Add Roo.form components to this form.
47875      * @param {Field} field1
47876      * @param {Field} field2 (optional)
47877      * @param {Field} etc (optional)
47878      * @return {BasicForm} this
47879      */
47880     add : function(){
47881         this.items.addAll(Array.prototype.slice.call(arguments, 0));
47882         return this;
47883     },
47884
47885
47886     /**
47887      * Removes a field from the items collection (does NOT remove its markup).
47888      * @param {Field} field
47889      * @return {BasicForm} this
47890      */
47891     remove : function(field){
47892         this.items.remove(field);
47893         return this;
47894     },
47895
47896     /**
47897      * Looks at the fields in this form, checks them for an id attribute,
47898      * and calls applyTo on the existing dom element with that id.
47899      * @return {BasicForm} this
47900      */
47901     render : function(){
47902         this.items.each(function(f){
47903             if(f.isFormField && !f.rendered && document.getElementById(f.id)){ // if the element exists
47904                 f.applyTo(f.id);
47905             }
47906         });
47907         return this;
47908     },
47909
47910     /**
47911      * Calls {@link Ext#apply} for all fields in this form with the passed object.
47912      * @param {Object} values
47913      * @return {BasicForm} this
47914      */
47915     applyToFields : function(o){
47916         this.items.each(function(f){
47917            Roo.apply(f, o);
47918         });
47919         return this;
47920     },
47921
47922     /**
47923      * Calls {@link Ext#applyIf} for all field in this form with the passed object.
47924      * @param {Object} values
47925      * @return {BasicForm} this
47926      */
47927     applyIfToFields : function(o){
47928         this.items.each(function(f){
47929            Roo.applyIf(f, o);
47930         });
47931         return this;
47932     }
47933 });
47934
47935 // back compat
47936 Roo.BasicForm = Roo.form.BasicForm;
47937
47938 Roo.apply(Roo.form.BasicForm, {
47939     
47940     popover : {
47941         
47942         padding : 5,
47943         
47944         isApplied : false,
47945         
47946         isMasked : false,
47947         
47948         form : false,
47949         
47950         target : false,
47951         
47952         intervalID : false,
47953         
47954         maskEl : false,
47955         
47956         apply : function()
47957         {
47958             if(this.isApplied){
47959                 return;
47960             }
47961             
47962             this.maskEl = {
47963                 top : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-top-mask" }, true),
47964                 left : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-left-mask" }, true),
47965                 bottom : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-bottom-mask" }, true),
47966                 right : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-right-mask" }, true)
47967             };
47968             
47969             this.maskEl.top.enableDisplayMode("block");
47970             this.maskEl.left.enableDisplayMode("block");
47971             this.maskEl.bottom.enableDisplayMode("block");
47972             this.maskEl.right.enableDisplayMode("block");
47973             
47974             Roo.get(document.body).on('click', function(){
47975                 this.unmask();
47976             }, this);
47977             
47978             Roo.get(document.body).on('touchstart', function(){
47979                 this.unmask();
47980             }, this);
47981             
47982             this.isApplied = true
47983         },
47984         
47985         mask : function(form, target)
47986         {
47987             this.form = form;
47988             
47989             this.target = target;
47990             
47991             if(!this.form.errorMask || !target.el){
47992                 return;
47993             }
47994             
47995             var scrollable = this.target.el.findScrollableParent() || this.target.el.findParent('div.x-layout-active-content', 100, true) || Roo.get(document.body);
47996             
47997             var ot = this.target.el.calcOffsetsTo(scrollable);
47998             
47999             var scrollTo = ot[1] - this.form.maskOffset;
48000             
48001             scrollTo = Math.min(scrollTo, scrollable.dom.scrollHeight);
48002             
48003             scrollable.scrollTo('top', scrollTo);
48004             
48005             var el = this.target.wrap || this.target.el;
48006             
48007             var box = el.getBox();
48008             
48009             this.maskEl.top.setStyle('position', 'absolute');
48010             this.maskEl.top.setStyle('z-index', 10000);
48011             this.maskEl.top.setSize(Roo.lib.Dom.getDocumentWidth(), box.y - this.padding);
48012             this.maskEl.top.setLeft(0);
48013             this.maskEl.top.setTop(0);
48014             this.maskEl.top.show();
48015             
48016             this.maskEl.left.setStyle('position', 'absolute');
48017             this.maskEl.left.setStyle('z-index', 10000);
48018             this.maskEl.left.setSize(box.x - this.padding, box.height + this.padding * 2);
48019             this.maskEl.left.setLeft(0);
48020             this.maskEl.left.setTop(box.y - this.padding);
48021             this.maskEl.left.show();
48022
48023             this.maskEl.bottom.setStyle('position', 'absolute');
48024             this.maskEl.bottom.setStyle('z-index', 10000);
48025             this.maskEl.bottom.setSize(Roo.lib.Dom.getDocumentWidth(), Roo.lib.Dom.getDocumentHeight() - box.bottom - this.padding);
48026             this.maskEl.bottom.setLeft(0);
48027             this.maskEl.bottom.setTop(box.bottom + this.padding);
48028             this.maskEl.bottom.show();
48029
48030             this.maskEl.right.setStyle('position', 'absolute');
48031             this.maskEl.right.setStyle('z-index', 10000);
48032             this.maskEl.right.setSize(Roo.lib.Dom.getDocumentWidth() - box.right - this.padding, box.height + this.padding * 2);
48033             this.maskEl.right.setLeft(box.right + this.padding);
48034             this.maskEl.right.setTop(box.y - this.padding);
48035             this.maskEl.right.show();
48036
48037             this.intervalID = window.setInterval(function() {
48038                 Roo.form.BasicForm.popover.unmask();
48039             }, 10000);
48040
48041             window.onwheel = function(){ return false;};
48042             
48043             (function(){ this.isMasked = true; }).defer(500, this);
48044             
48045         },
48046         
48047         unmask : function()
48048         {
48049             if(!this.isApplied || !this.isMasked || !this.form || !this.target || !this.form.errorMask){
48050                 return;
48051             }
48052             
48053             this.maskEl.top.setStyle('position', 'absolute');
48054             this.maskEl.top.setSize(0, 0).setXY([0, 0]);
48055             this.maskEl.top.hide();
48056
48057             this.maskEl.left.setStyle('position', 'absolute');
48058             this.maskEl.left.setSize(0, 0).setXY([0, 0]);
48059             this.maskEl.left.hide();
48060
48061             this.maskEl.bottom.setStyle('position', 'absolute');
48062             this.maskEl.bottom.setSize(0, 0).setXY([0, 0]);
48063             this.maskEl.bottom.hide();
48064
48065             this.maskEl.right.setStyle('position', 'absolute');
48066             this.maskEl.right.setSize(0, 0).setXY([0, 0]);
48067             this.maskEl.right.hide();
48068             
48069             window.onwheel = function(){ return true;};
48070             
48071             if(this.intervalID){
48072                 window.clearInterval(this.intervalID);
48073                 this.intervalID = false;
48074             }
48075             
48076             this.isMasked = false;
48077             
48078         }
48079         
48080     }
48081     
48082 });/*
48083  * Based on:
48084  * Ext JS Library 1.1.1
48085  * Copyright(c) 2006-2007, Ext JS, LLC.
48086  *
48087  * Originally Released Under LGPL - original licence link has changed is not relivant.
48088  *
48089  * Fork - LGPL
48090  * <script type="text/javascript">
48091  */
48092
48093 /**
48094  * @class Roo.form.Form
48095  * @extends Roo.form.BasicForm
48096  * Adds the ability to dynamically render forms with JavaScript to {@link Roo.form.BasicForm}.
48097  * @constructor
48098  * @param {Object} config Configuration options
48099  */
48100 Roo.form.Form = function(config){
48101     var xitems =  [];
48102     if (config.items) {
48103         xitems = config.items;
48104         delete config.items;
48105     }
48106    
48107     
48108     Roo.form.Form.superclass.constructor.call(this, null, config);
48109     this.url = this.url || this.action;
48110     if(!this.root){
48111         this.root = new Roo.form.Layout(Roo.applyIf({
48112             id: Roo.id()
48113         }, config));
48114     }
48115     this.active = this.root;
48116     /**
48117      * Array of all the buttons that have been added to this form via {@link addButton}
48118      * @type Array
48119      */
48120     this.buttons = [];
48121     this.allItems = [];
48122     this.addEvents({
48123         /**
48124          * @event clientvalidation
48125          * If the monitorValid config option is true, this event fires repetitively to notify of valid state
48126          * @param {Form} this
48127          * @param {Boolean} valid true if the form has passed client-side validation
48128          */
48129         clientvalidation: true,
48130         /**
48131          * @event rendered
48132          * Fires when the form is rendered
48133          * @param {Roo.form.Form} form
48134          */
48135         rendered : true
48136     });
48137     
48138     if (this.progressUrl) {
48139             // push a hidden field onto the list of fields..
48140             this.addxtype( {
48141                     xns: Roo.form, 
48142                     xtype : 'Hidden', 
48143                     name : 'UPLOAD_IDENTIFIER' 
48144             });
48145         }
48146         
48147     
48148     Roo.each(xitems, this.addxtype, this);
48149     
48150 };
48151
48152 Roo.extend(Roo.form.Form, Roo.form.BasicForm, {
48153     /**
48154      * @cfg {Number} labelWidth The width of labels. This property cascades to child containers.
48155      */
48156     /**
48157      * @cfg {String} itemCls A css class to apply to the x-form-item of fields. This property cascades to child containers.
48158      */
48159     /**
48160      * @cfg {String} buttonAlign Valid values are "left," "center" and "right" (defaults to "center")
48161      */
48162     buttonAlign:'center',
48163
48164     /**
48165      * @cfg {Number} minButtonWidth Minimum width of all buttons in pixels (defaults to 75)
48166      */
48167     minButtonWidth:75,
48168
48169     /**
48170      * @cfg {String} labelAlign Valid values are "left," "top" and "right" (defaults to "left").
48171      * This property cascades to child containers if not set.
48172      */
48173     labelAlign:'left',
48174
48175     /**
48176      * @cfg {Boolean} monitorValid If true the form monitors its valid state <b>client-side</b> and
48177      * fires a looping event with that state. This is required to bind buttons to the valid
48178      * state using the config value formBind:true on the button.
48179      */
48180     monitorValid : false,
48181
48182     /**
48183      * @cfg {Number} monitorPoll The milliseconds to poll valid state, ignored if monitorValid is not true (defaults to 200)
48184      */
48185     monitorPoll : 200,
48186     
48187     /**
48188      * @cfg {String} progressUrl - Url to return progress data 
48189      */
48190     
48191     progressUrl : false,
48192     /**
48193      * @cfg {boolean|FormData} formData - true to use new 'FormData' post, or set to a new FormData({dom form}) Object, if
48194      * sending a formdata with extra parameters - eg uploaded elements.
48195      */
48196     
48197     formData : false,
48198     
48199     /**
48200      * Opens a new {@link Roo.form.Column} container in the layout stack. If fields are passed after the config, the
48201      * fields are added and the column is closed. If no fields are passed the column remains open
48202      * until end() is called.
48203      * @param {Object} config The config to pass to the column
48204      * @param {Field} field1 (optional)
48205      * @param {Field} field2 (optional)
48206      * @param {Field} etc (optional)
48207      * @return Column The column container object
48208      */
48209     column : function(c){
48210         var col = new Roo.form.Column(c);
48211         this.start(col);
48212         if(arguments.length > 1){ // duplicate code required because of Opera
48213             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48214             this.end();
48215         }
48216         return col;
48217     },
48218
48219     /**
48220      * Opens a new {@link Roo.form.FieldSet} container in the layout stack. If fields are passed after the config, the
48221      * fields are added and the fieldset is closed. If no fields are passed the fieldset remains open
48222      * until end() is called.
48223      * @param {Object} config The config to pass to the fieldset
48224      * @param {Field} field1 (optional)
48225      * @param {Field} field2 (optional)
48226      * @param {Field} etc (optional)
48227      * @return FieldSet The fieldset container object
48228      */
48229     fieldset : function(c){
48230         var fs = new Roo.form.FieldSet(c);
48231         this.start(fs);
48232         if(arguments.length > 1){ // duplicate code required because of Opera
48233             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48234             this.end();
48235         }
48236         return fs;
48237     },
48238
48239     /**
48240      * Opens a new {@link Roo.form.Layout} container in the layout stack. If fields are passed after the config, the
48241      * fields are added and the container is closed. If no fields are passed the container remains open
48242      * until end() is called.
48243      * @param {Object} config The config to pass to the Layout
48244      * @param {Field} field1 (optional)
48245      * @param {Field} field2 (optional)
48246      * @param {Field} etc (optional)
48247      * @return Layout The container object
48248      */
48249     container : function(c){
48250         var l = new Roo.form.Layout(c);
48251         this.start(l);
48252         if(arguments.length > 1){ // duplicate code required because of Opera
48253             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48254             this.end();
48255         }
48256         return l;
48257     },
48258
48259     /**
48260      * Opens the passed container in the layout stack. The container can be any {@link Roo.form.Layout} or subclass.
48261      * @param {Object} container A Roo.form.Layout or subclass of Layout
48262      * @return {Form} this
48263      */
48264     start : function(c){
48265         // cascade label info
48266         Roo.applyIf(c, {'labelAlign': this.active.labelAlign, 'labelWidth': this.active.labelWidth, 'itemCls': this.active.itemCls});
48267         this.active.stack.push(c);
48268         c.ownerCt = this.active;
48269         this.active = c;
48270         return this;
48271     },
48272
48273     /**
48274      * Closes the current open container
48275      * @return {Form} this
48276      */
48277     end : function(){
48278         if(this.active == this.root){
48279             return this;
48280         }
48281         this.active = this.active.ownerCt;
48282         return this;
48283     },
48284
48285     /**
48286      * Add Roo.form components to the current open container (e.g. column, fieldset, etc.).  Fields added via this method
48287      * can also be passed with an additional property of fieldLabel, which if supplied, will provide the text to display
48288      * as the label of the field.
48289      * @param {Field} field1
48290      * @param {Field} field2 (optional)
48291      * @param {Field} etc. (optional)
48292      * @return {Form} this
48293      */
48294     add : function(){
48295         this.active.stack.push.apply(this.active.stack, arguments);
48296         this.allItems.push.apply(this.allItems,arguments);
48297         var r = [];
48298         for(var i = 0, a = arguments, len = a.length; i < len; i++) {
48299             if(a[i].isFormField){
48300                 r.push(a[i]);
48301             }
48302         }
48303         if(r.length > 0){
48304             Roo.form.Form.superclass.add.apply(this, r);
48305         }
48306         return this;
48307     },
48308     
48309
48310     
48311     
48312     
48313      /**
48314      * Find any element that has been added to a form, using it's ID or name
48315      * This can include framesets, columns etc. along with regular fields..
48316      * @param {String} id - id or name to find.
48317      
48318      * @return {Element} e - or false if nothing found.
48319      */
48320     findbyId : function(id)
48321     {
48322         var ret = false;
48323         if (!id) {
48324             return ret;
48325         }
48326         Roo.each(this.allItems, function(f){
48327             if (f.id == id || f.name == id ){
48328                 ret = f;
48329                 return false;
48330             }
48331         });
48332         return ret;
48333     },
48334
48335     
48336     
48337     /**
48338      * Render this form into the passed container. This should only be called once!
48339      * @param {String/HTMLElement/Element} container The element this component should be rendered into
48340      * @return {Form} this
48341      */
48342     render : function(ct)
48343     {
48344         
48345         
48346         
48347         ct = Roo.get(ct);
48348         var o = this.autoCreate || {
48349             tag: 'form',
48350             method : this.method || 'POST',
48351             id : this.id || Roo.id()
48352         };
48353         this.initEl(ct.createChild(o));
48354
48355         this.root.render(this.el);
48356         
48357        
48358              
48359         this.items.each(function(f){
48360             f.render('x-form-el-'+f.id);
48361         });
48362
48363         if(this.buttons.length > 0){
48364             // tables are required to maintain order and for correct IE layout
48365             var tb = this.el.createChild({cls:'x-form-btns-ct', cn: {
48366                 cls:"x-form-btns x-form-btns-"+this.buttonAlign,
48367                 html:'<table cellspacing="0"><tbody><tr></tr></tbody></table><div class="x-clear"></div>'
48368             }}, null, true);
48369             var tr = tb.getElementsByTagName('tr')[0];
48370             for(var i = 0, len = this.buttons.length; i < len; i++) {
48371                 var b = this.buttons[i];
48372                 var td = document.createElement('td');
48373                 td.className = 'x-form-btn-td';
48374                 b.render(tr.appendChild(td));
48375             }
48376         }
48377         if(this.monitorValid){ // initialize after render
48378             this.startMonitoring();
48379         }
48380         this.fireEvent('rendered', this);
48381         return this;
48382     },
48383
48384     /**
48385      * Adds a button to the footer of the form - this <b>must</b> be called before the form is rendered.
48386      * @param {String/Object} config A string becomes the button text, an object can either be a Button config
48387      * object or a valid Roo.DomHelper element config
48388      * @param {Function} handler The function called when the button is clicked
48389      * @param {Object} scope (optional) The scope of the handler function
48390      * @return {Roo.Button}
48391      */
48392     addButton : function(config, handler, scope){
48393         var bc = {
48394             handler: handler,
48395             scope: scope,
48396             minWidth: this.minButtonWidth,
48397             hideParent:true
48398         };
48399         if(typeof config == "string"){
48400             bc.text = config;
48401         }else{
48402             Roo.apply(bc, config);
48403         }
48404         var btn = new Roo.Button(null, bc);
48405         this.buttons.push(btn);
48406         return btn;
48407     },
48408
48409      /**
48410      * Adds a series of form elements (using the xtype property as the factory method.
48411      * Valid xtypes are:  TextField, TextArea .... Button, Layout, FieldSet, Column, (and 'end' to close a block)
48412      * @param {Object} config 
48413      */
48414     
48415     addxtype : function()
48416     {
48417         var ar = Array.prototype.slice.call(arguments, 0);
48418         var ret = false;
48419         for(var i = 0; i < ar.length; i++) {
48420             if (!ar[i]) {
48421                 continue; // skip -- if this happends something invalid got sent, we 
48422                 // should ignore it, as basically that interface element will not show up
48423                 // and that should be pretty obvious!!
48424             }
48425             
48426             if (Roo.form[ar[i].xtype]) {
48427                 ar[i].form = this;
48428                 var fe = Roo.factory(ar[i], Roo.form);
48429                 if (!ret) {
48430                     ret = fe;
48431                 }
48432                 fe.form = this;
48433                 if (fe.store) {
48434                     fe.store.form = this;
48435                 }
48436                 if (fe.isLayout) {  
48437                          
48438                     this.start(fe);
48439                     this.allItems.push(fe);
48440                     if (fe.items && fe.addxtype) {
48441                         fe.addxtype.apply(fe, fe.items);
48442                         delete fe.items;
48443                     }
48444                      this.end();
48445                     continue;
48446                 }
48447                 
48448                 
48449                  
48450                 this.add(fe);
48451               //  console.log('adding ' + ar[i].xtype);
48452             }
48453             if (ar[i].xtype == 'Button') {  
48454                 //console.log('adding button');
48455                 //console.log(ar[i]);
48456                 this.addButton(ar[i]);
48457                 this.allItems.push(fe);
48458                 continue;
48459             }
48460             
48461             if (ar[i].xtype == 'end') { // so we can add fieldsets... / layout etc.
48462                 alert('end is not supported on xtype any more, use items');
48463             //    this.end();
48464             //    //console.log('adding end');
48465             }
48466             
48467         }
48468         return ret;
48469     },
48470     
48471     /**
48472      * Starts monitoring of the valid state of this form. Usually this is done by passing the config
48473      * option "monitorValid"
48474      */
48475     startMonitoring : function(){
48476         if(!this.bound){
48477             this.bound = true;
48478             Roo.TaskMgr.start({
48479                 run : this.bindHandler,
48480                 interval : this.monitorPoll || 200,
48481                 scope: this
48482             });
48483         }
48484     },
48485
48486     /**
48487      * Stops monitoring of the valid state of this form
48488      */
48489     stopMonitoring : function(){
48490         this.bound = false;
48491     },
48492
48493     // private
48494     bindHandler : function(){
48495         if(!this.bound){
48496             return false; // stops binding
48497         }
48498         var valid = true;
48499         this.items.each(function(f){
48500             if(!f.isValid(true)){
48501                 valid = false;
48502                 return false;
48503             }
48504         });
48505         for(var i = 0, len = this.buttons.length; i < len; i++){
48506             var btn = this.buttons[i];
48507             if(btn.formBind === true && btn.disabled === valid){
48508                 btn.setDisabled(!valid);
48509             }
48510         }
48511         this.fireEvent('clientvalidation', this, valid);
48512     }
48513     
48514     
48515     
48516     
48517     
48518     
48519     
48520     
48521 });
48522
48523
48524 // back compat
48525 Roo.Form = Roo.form.Form;
48526 /*
48527  * Based on:
48528  * Ext JS Library 1.1.1
48529  * Copyright(c) 2006-2007, Ext JS, LLC.
48530  *
48531  * Originally Released Under LGPL - original licence link has changed is not relivant.
48532  *
48533  * Fork - LGPL
48534  * <script type="text/javascript">
48535  */
48536
48537 // as we use this in bootstrap.
48538 Roo.namespace('Roo.form');
48539  /**
48540  * @class Roo.form.Action
48541  * Internal Class used to handle form actions
48542  * @constructor
48543  * @param {Roo.form.BasicForm} el The form element or its id
48544  * @param {Object} config Configuration options
48545  */
48546
48547  
48548  
48549 // define the action interface
48550 Roo.form.Action = function(form, options){
48551     this.form = form;
48552     this.options = options || {};
48553 };
48554 /**
48555  * Client Validation Failed
48556  * @const 
48557  */
48558 Roo.form.Action.CLIENT_INVALID = 'client';
48559 /**
48560  * Server Validation Failed
48561  * @const 
48562  */
48563 Roo.form.Action.SERVER_INVALID = 'server';
48564  /**
48565  * Connect to Server Failed
48566  * @const 
48567  */
48568 Roo.form.Action.CONNECT_FAILURE = 'connect';
48569 /**
48570  * Reading Data from Server Failed
48571  * @const 
48572  */
48573 Roo.form.Action.LOAD_FAILURE = 'load';
48574
48575 Roo.form.Action.prototype = {
48576     type : 'default',
48577     failureType : undefined,
48578     response : undefined,
48579     result : undefined,
48580
48581     // interface method
48582     run : function(options){
48583
48584     },
48585
48586     // interface method
48587     success : function(response){
48588
48589     },
48590
48591     // interface method
48592     handleResponse : function(response){
48593
48594     },
48595
48596     // default connection failure
48597     failure : function(response){
48598         
48599         this.response = response;
48600         this.failureType = Roo.form.Action.CONNECT_FAILURE;
48601         this.form.afterAction(this, false);
48602     },
48603
48604     processResponse : function(response){
48605         this.response = response;
48606         if(!response.responseText){
48607             return true;
48608         }
48609         this.result = this.handleResponse(response);
48610         return this.result;
48611     },
48612
48613     // utility functions used internally
48614     getUrl : function(appendParams){
48615         var url = this.options.url || this.form.url || this.form.el.dom.action;
48616         if(appendParams){
48617             var p = this.getParams();
48618             if(p){
48619                 url += (url.indexOf('?') != -1 ? '&' : '?') + p;
48620             }
48621         }
48622         return url;
48623     },
48624
48625     getMethod : function(){
48626         return (this.options.method || this.form.method || this.form.el.dom.method || 'POST').toUpperCase();
48627     },
48628
48629     getParams : function(){
48630         var bp = this.form.baseParams;
48631         var p = this.options.params;
48632         if(p){
48633             if(typeof p == "object"){
48634                 p = Roo.urlEncode(Roo.applyIf(p, bp));
48635             }else if(typeof p == 'string' && bp){
48636                 p += '&' + Roo.urlEncode(bp);
48637             }
48638         }else if(bp){
48639             p = Roo.urlEncode(bp);
48640         }
48641         return p;
48642     },
48643
48644     createCallback : function(){
48645         return {
48646             success: this.success,
48647             failure: this.failure,
48648             scope: this,
48649             timeout: (this.form.timeout*1000),
48650             upload: this.form.fileUpload ? this.success : undefined
48651         };
48652     }
48653 };
48654
48655 Roo.form.Action.Submit = function(form, options){
48656     Roo.form.Action.Submit.superclass.constructor.call(this, form, options);
48657 };
48658
48659 Roo.extend(Roo.form.Action.Submit, Roo.form.Action, {
48660     type : 'submit',
48661
48662     haveProgress : false,
48663     uploadComplete : false,
48664     
48665     // uploadProgress indicator.
48666     uploadProgress : function()
48667     {
48668         if (!this.form.progressUrl) {
48669             return;
48670         }
48671         
48672         if (!this.haveProgress) {
48673             Roo.MessageBox.progress("Uploading", "Uploading");
48674         }
48675         if (this.uploadComplete) {
48676            Roo.MessageBox.hide();
48677            return;
48678         }
48679         
48680         this.haveProgress = true;
48681    
48682         var uid = this.form.findField('UPLOAD_IDENTIFIER').getValue();
48683         
48684         var c = new Roo.data.Connection();
48685         c.request({
48686             url : this.form.progressUrl,
48687             params: {
48688                 id : uid
48689             },
48690             method: 'GET',
48691             success : function(req){
48692                //console.log(data);
48693                 var rdata = false;
48694                 var edata;
48695                 try  {
48696                    rdata = Roo.decode(req.responseText)
48697                 } catch (e) {
48698                     Roo.log("Invalid data from server..");
48699                     Roo.log(edata);
48700                     return;
48701                 }
48702                 if (!rdata || !rdata.success) {
48703                     Roo.log(rdata);
48704                     Roo.MessageBox.alert(Roo.encode(rdata));
48705                     return;
48706                 }
48707                 var data = rdata.data;
48708                 
48709                 if (this.uploadComplete) {
48710                    Roo.MessageBox.hide();
48711                    return;
48712                 }
48713                    
48714                 if (data){
48715                     Roo.MessageBox.updateProgress(data.bytes_uploaded/data.bytes_total,
48716                        Math.floor((data.bytes_total - data.bytes_uploaded)/1000) + 'k remaining'
48717                     );
48718                 }
48719                 this.uploadProgress.defer(2000,this);
48720             },
48721        
48722             failure: function(data) {
48723                 Roo.log('progress url failed ');
48724                 Roo.log(data);
48725             },
48726             scope : this
48727         });
48728            
48729     },
48730     
48731     
48732     run : function()
48733     {
48734         // run get Values on the form, so it syncs any secondary forms.
48735         this.form.getValues();
48736         
48737         var o = this.options;
48738         var method = this.getMethod();
48739         var isPost = method == 'POST';
48740         if(o.clientValidation === false || this.form.isValid()){
48741             
48742             if (this.form.progressUrl) {
48743                 this.form.findField('UPLOAD_IDENTIFIER').setValue(
48744                     (new Date() * 1) + '' + Math.random());
48745                     
48746             } 
48747             
48748             
48749             Roo.Ajax.request(Roo.apply(this.createCallback(), {
48750                 form:this.form.el.dom,
48751                 url:this.getUrl(!isPost),
48752                 method: method,
48753                 params:isPost ? this.getParams() : null,
48754                 isUpload: this.form.fileUpload,
48755                 formData : this.form.formData
48756             }));
48757             
48758             this.uploadProgress();
48759
48760         }else if (o.clientValidation !== false){ // client validation failed
48761             this.failureType = Roo.form.Action.CLIENT_INVALID;
48762             this.form.afterAction(this, false);
48763         }
48764     },
48765
48766     success : function(response)
48767     {
48768         this.uploadComplete= true;
48769         if (this.haveProgress) {
48770             Roo.MessageBox.hide();
48771         }
48772         
48773         
48774         var result = this.processResponse(response);
48775         if(result === true || result.success){
48776             this.form.afterAction(this, true);
48777             return;
48778         }
48779         if(result.errors){
48780             this.form.markInvalid(result.errors);
48781             this.failureType = Roo.form.Action.SERVER_INVALID;
48782         }
48783         this.form.afterAction(this, false);
48784     },
48785     failure : function(response)
48786     {
48787         this.uploadComplete= true;
48788         if (this.haveProgress) {
48789             Roo.MessageBox.hide();
48790         }
48791         
48792         this.response = response;
48793         this.failureType = Roo.form.Action.CONNECT_FAILURE;
48794         this.form.afterAction(this, false);
48795     },
48796     
48797     handleResponse : function(response){
48798         if(this.form.errorReader){
48799             var rs = this.form.errorReader.read(response);
48800             var errors = [];
48801             if(rs.records){
48802                 for(var i = 0, len = rs.records.length; i < len; i++) {
48803                     var r = rs.records[i];
48804                     errors[i] = r.data;
48805                 }
48806             }
48807             if(errors.length < 1){
48808                 errors = null;
48809             }
48810             return {
48811                 success : rs.success,
48812                 errors : errors
48813             };
48814         }
48815         var ret = false;
48816         try {
48817             ret = Roo.decode(response.responseText);
48818         } catch (e) {
48819             ret = {
48820                 success: false,
48821                 errorMsg: "Failed to read server message: " + (response ? response.responseText : ' - no message'),
48822                 errors : []
48823             };
48824         }
48825         return ret;
48826         
48827     }
48828 });
48829
48830
48831 Roo.form.Action.Load = function(form, options){
48832     Roo.form.Action.Load.superclass.constructor.call(this, form, options);
48833     this.reader = this.form.reader;
48834 };
48835
48836 Roo.extend(Roo.form.Action.Load, Roo.form.Action, {
48837     type : 'load',
48838
48839     run : function(){
48840         
48841         Roo.Ajax.request(Roo.apply(
48842                 this.createCallback(), {
48843                     method:this.getMethod(),
48844                     url:this.getUrl(false),
48845                     params:this.getParams()
48846         }));
48847     },
48848
48849     success : function(response){
48850         
48851         var result = this.processResponse(response);
48852         if(result === true || !result.success || !result.data){
48853             this.failureType = Roo.form.Action.LOAD_FAILURE;
48854             this.form.afterAction(this, false);
48855             return;
48856         }
48857         this.form.clearInvalid();
48858         this.form.setValues(result.data);
48859         this.form.afterAction(this, true);
48860     },
48861
48862     handleResponse : function(response){
48863         if(this.form.reader){
48864             var rs = this.form.reader.read(response);
48865             var data = rs.records && rs.records[0] ? rs.records[0].data : null;
48866             return {
48867                 success : rs.success,
48868                 data : data
48869             };
48870         }
48871         return Roo.decode(response.responseText);
48872     }
48873 });
48874
48875 Roo.form.Action.ACTION_TYPES = {
48876     'load' : Roo.form.Action.Load,
48877     'submit' : Roo.form.Action.Submit
48878 };/*
48879  * Based on:
48880  * Ext JS Library 1.1.1
48881  * Copyright(c) 2006-2007, Ext JS, LLC.
48882  *
48883  * Originally Released Under LGPL - original licence link has changed is not relivant.
48884  *
48885  * Fork - LGPL
48886  * <script type="text/javascript">
48887  */
48888  
48889 /**
48890  * @class Roo.form.Layout
48891  * @extends Roo.Component
48892  * Creates a container for layout and rendering of fields in an {@link Roo.form.Form}.
48893  * @constructor
48894  * @param {Object} config Configuration options
48895  */
48896 Roo.form.Layout = function(config){
48897     var xitems = [];
48898     if (config.items) {
48899         xitems = config.items;
48900         delete config.items;
48901     }
48902     Roo.form.Layout.superclass.constructor.call(this, config);
48903     this.stack = [];
48904     Roo.each(xitems, this.addxtype, this);
48905      
48906 };
48907
48908 Roo.extend(Roo.form.Layout, Roo.Component, {
48909     /**
48910      * @cfg {String/Object} autoCreate
48911      * A DomHelper element spec used to autocreate the layout (defaults to {tag: 'div', cls: 'x-form-ct'})
48912      */
48913     /**
48914      * @cfg {String/Object/Function} style
48915      * A style specification string, e.g. "width:100px", or object in the form {width:"100px"}, or
48916      * a function which returns such a specification.
48917      */
48918     /**
48919      * @cfg {String} labelAlign
48920      * Valid values are "left," "top" and "right" (defaults to "left")
48921      */
48922     /**
48923      * @cfg {Number} labelWidth
48924      * Fixed width in pixels of all field labels (defaults to undefined)
48925      */
48926     /**
48927      * @cfg {Boolean} clear
48928      * True to add a clearing element at the end of this layout, equivalent to CSS clear: both (defaults to true)
48929      */
48930     clear : true,
48931     /**
48932      * @cfg {String} labelSeparator
48933      * The separator to use after field labels (defaults to ':')
48934      */
48935     labelSeparator : ':',
48936     /**
48937      * @cfg {Boolean} hideLabels
48938      * True to suppress the display of field labels in this layout (defaults to false)
48939      */
48940     hideLabels : false,
48941
48942     // private
48943     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct'},
48944     
48945     isLayout : true,
48946     
48947     // private
48948     onRender : function(ct, position){
48949         if(this.el){ // from markup
48950             this.el = Roo.get(this.el);
48951         }else {  // generate
48952             var cfg = this.getAutoCreate();
48953             this.el = ct.createChild(cfg, position);
48954         }
48955         if(this.style){
48956             this.el.applyStyles(this.style);
48957         }
48958         if(this.labelAlign){
48959             this.el.addClass('x-form-label-'+this.labelAlign);
48960         }
48961         if(this.hideLabels){
48962             this.labelStyle = "display:none";
48963             this.elementStyle = "padding-left:0;";
48964         }else{
48965             if(typeof this.labelWidth == 'number'){
48966                 this.labelStyle = "width:"+this.labelWidth+"px;";
48967                 this.elementStyle = "padding-left:"+((this.labelWidth+(typeof this.labelPad == 'number' ? this.labelPad : 5))+'px')+";";
48968             }
48969             if(this.labelAlign == 'top'){
48970                 this.labelStyle = "width:auto;";
48971                 this.elementStyle = "padding-left:0;";
48972             }
48973         }
48974         var stack = this.stack;
48975         var slen = stack.length;
48976         if(slen > 0){
48977             if(!this.fieldTpl){
48978                 var t = new Roo.Template(
48979                     '<div class="x-form-item {5}">',
48980                         '<label for="{0}" style="{2}">{1}{4}</label>',
48981                         '<div class="x-form-element" id="x-form-el-{0}" style="{3}">',
48982                         '</div>',
48983                     '</div><div class="x-form-clear-left"></div>'
48984                 );
48985                 t.disableFormats = true;
48986                 t.compile();
48987                 Roo.form.Layout.prototype.fieldTpl = t;
48988             }
48989             for(var i = 0; i < slen; i++) {
48990                 if(stack[i].isFormField){
48991                     this.renderField(stack[i]);
48992                 }else{
48993                     this.renderComponent(stack[i]);
48994                 }
48995             }
48996         }
48997         if(this.clear){
48998             this.el.createChild({cls:'x-form-clear'});
48999         }
49000     },
49001
49002     // private
49003     renderField : function(f){
49004         f.fieldEl = Roo.get(this.fieldTpl.append(this.el, [
49005                f.id, //0
49006                f.fieldLabel, //1
49007                f.labelStyle||this.labelStyle||'', //2
49008                this.elementStyle||'', //3
49009                typeof f.labelSeparator == 'undefined' ? this.labelSeparator : f.labelSeparator, //4
49010                f.itemCls||this.itemCls||''  //5
49011        ], true).getPrevSibling());
49012     },
49013
49014     // private
49015     renderComponent : function(c){
49016         c.render(c.isLayout ? this.el : this.el.createChild());    
49017     },
49018     /**
49019      * Adds a object form elements (using the xtype property as the factory method.)
49020      * Valid xtypes are:  TextField, TextArea .... Button, Layout, FieldSet, Column
49021      * @param {Object} config 
49022      */
49023     addxtype : function(o)
49024     {
49025         // create the lement.
49026         o.form = this.form;
49027         var fe = Roo.factory(o, Roo.form);
49028         this.form.allItems.push(fe);
49029         this.stack.push(fe);
49030         
49031         if (fe.isFormField) {
49032             this.form.items.add(fe);
49033         }
49034          
49035         return fe;
49036     }
49037 });
49038
49039 /**
49040  * @class Roo.form.Column
49041  * @extends Roo.form.Layout
49042  * Creates a column container for layout and rendering of fields in an {@link Roo.form.Form}.
49043  * @constructor
49044  * @param {Object} config Configuration options
49045  */
49046 Roo.form.Column = function(config){
49047     Roo.form.Column.superclass.constructor.call(this, config);
49048 };
49049
49050 Roo.extend(Roo.form.Column, Roo.form.Layout, {
49051     /**
49052      * @cfg {Number/String} width
49053      * The fixed width of the column in pixels or CSS value (defaults to "auto")
49054      */
49055     /**
49056      * @cfg {String/Object} autoCreate
49057      * A DomHelper element spec used to autocreate the column (defaults to {tag: 'div', cls: 'x-form-ct x-form-column'})
49058      */
49059
49060     // private
49061     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct x-form-column'},
49062
49063     // private
49064     onRender : function(ct, position){
49065         Roo.form.Column.superclass.onRender.call(this, ct, position);
49066         if(this.width){
49067             this.el.setWidth(this.width);
49068         }
49069     }
49070 });
49071
49072
49073 /**
49074  * @class Roo.form.Row
49075  * @extends Roo.form.Layout
49076  * Creates a row container for layout and rendering of fields in an {@link Roo.form.Form}.
49077  * @constructor
49078  * @param {Object} config Configuration options
49079  */
49080
49081  
49082 Roo.form.Row = function(config){
49083     Roo.form.Row.superclass.constructor.call(this, config);
49084 };
49085  
49086 Roo.extend(Roo.form.Row, Roo.form.Layout, {
49087       /**
49088      * @cfg {Number/String} width
49089      * The fixed width of the column in pixels or CSS value (defaults to "auto")
49090      */
49091     /**
49092      * @cfg {Number/String} height
49093      * The fixed height of the column in pixels or CSS value (defaults to "auto")
49094      */
49095     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct x-form-row'},
49096     
49097     padWidth : 20,
49098     // private
49099     onRender : function(ct, position){
49100         //console.log('row render');
49101         if(!this.rowTpl){
49102             var t = new Roo.Template(
49103                 '<div class="x-form-item {5}" style="float:left;width:{6}px">',
49104                     '<label for="{0}" style="{2}">{1}{4}</label>',
49105                     '<div class="x-form-element" id="x-form-el-{0}" style="{3}">',
49106                     '</div>',
49107                 '</div>'
49108             );
49109             t.disableFormats = true;
49110             t.compile();
49111             Roo.form.Layout.prototype.rowTpl = t;
49112         }
49113         this.fieldTpl = this.rowTpl;
49114         
49115         //console.log('lw' + this.labelWidth +', la:' + this.labelAlign);
49116         var labelWidth = 100;
49117         
49118         if ((this.labelAlign != 'top')) {
49119             if (typeof this.labelWidth == 'number') {
49120                 labelWidth = this.labelWidth
49121             }
49122             this.padWidth =  20 + labelWidth;
49123             
49124         }
49125         
49126         Roo.form.Column.superclass.onRender.call(this, ct, position);
49127         if(this.width){
49128             this.el.setWidth(this.width);
49129         }
49130         if(this.height){
49131             this.el.setHeight(this.height);
49132         }
49133     },
49134     
49135     // private
49136     renderField : function(f){
49137         f.fieldEl = this.fieldTpl.append(this.el, [
49138                f.id, f.fieldLabel,
49139                f.labelStyle||this.labelStyle||'',
49140                this.elementStyle||'',
49141                typeof f.labelSeparator == 'undefined' ? this.labelSeparator : f.labelSeparator,
49142                f.itemCls||this.itemCls||'',
49143                f.width ? f.width + this.padWidth : 160 + this.padWidth
49144        ],true);
49145     }
49146 });
49147  
49148
49149 /**
49150  * @class Roo.form.FieldSet
49151  * @extends Roo.form.Layout
49152  * Creates a fieldset container for layout and rendering of fields in an {@link Roo.form.Form}.
49153  * @constructor
49154  * @param {Object} config Configuration options
49155  */
49156 Roo.form.FieldSet = function(config){
49157     Roo.form.FieldSet.superclass.constructor.call(this, config);
49158 };
49159
49160 Roo.extend(Roo.form.FieldSet, Roo.form.Layout, {
49161     /**
49162      * @cfg {String} legend
49163      * The text to display as the legend for the FieldSet (defaults to '')
49164      */
49165     /**
49166      * @cfg {String/Object} autoCreate
49167      * A DomHelper element spec used to autocreate the fieldset (defaults to {tag: 'fieldset', cn: {tag:'legend'}})
49168      */
49169
49170     // private
49171     defaultAutoCreate : {tag: 'fieldset', cn: {tag:'legend'}},
49172
49173     // private
49174     onRender : function(ct, position){
49175         Roo.form.FieldSet.superclass.onRender.call(this, ct, position);
49176         if(this.legend){
49177             this.setLegend(this.legend);
49178         }
49179     },
49180
49181     // private
49182     setLegend : function(text){
49183         if(this.rendered){
49184             this.el.child('legend').update(text);
49185         }
49186     }
49187 });/*
49188  * Based on:
49189  * Ext JS Library 1.1.1
49190  * Copyright(c) 2006-2007, Ext JS, LLC.
49191  *
49192  * Originally Released Under LGPL - original licence link has changed is not relivant.
49193  *
49194  * Fork - LGPL
49195  * <script type="text/javascript">
49196  */
49197 /**
49198  * @class Roo.form.VTypes
49199  * Overridable validation definitions. The validations provided are basic and intended to be easily customizable and extended.
49200  * @singleton
49201  */
49202 Roo.form.VTypes = function(){
49203     // closure these in so they are only created once.
49204     var alpha = /^[a-zA-Z_]+$/;
49205     var alphanum = /^[a-zA-Z0-9_]+$/;
49206     var email = /^([\w]+)(.[\w]+)*@([\w-]+\.){1,5}([A-Za-z]){2,24}$/;
49207     var url = /(((https?)|(ftp)):\/\/([\-\w]+\.)+\w{2,3}(\/[%\-\w]+(\.\w{2,})?)*(([\w\-\.\?\\\/+@&#;`~=%!]*)(\.\w{2,})?)*\/?)/i;
49208
49209     // All these messages and functions are configurable
49210     return {
49211         /**
49212          * The function used to validate email addresses
49213          * @param {String} value The email address
49214          */
49215         'email' : function(v){
49216             return email.test(v);
49217         },
49218         /**
49219          * The error text to display when the email validation function returns false
49220          * @type String
49221          */
49222         'emailText' : 'This field should be an e-mail address in the format "user@domain.com"',
49223         /**
49224          * The keystroke filter mask to be applied on email input
49225          * @type RegExp
49226          */
49227         'emailMask' : /[a-z0-9_\.\-@]/i,
49228
49229         /**
49230          * The function used to validate URLs
49231          * @param {String} value The URL
49232          */
49233         'url' : function(v){
49234             return url.test(v);
49235         },
49236         /**
49237          * The error text to display when the url validation function returns false
49238          * @type String
49239          */
49240         'urlText' : 'This field should be a URL in the format "http:/'+'/www.domain.com"',
49241         
49242         /**
49243          * The function used to validate alpha values
49244          * @param {String} value The value
49245          */
49246         'alpha' : function(v){
49247             return alpha.test(v);
49248         },
49249         /**
49250          * The error text to display when the alpha validation function returns false
49251          * @type String
49252          */
49253         'alphaText' : 'This field should only contain letters and _',
49254         /**
49255          * The keystroke filter mask to be applied on alpha input
49256          * @type RegExp
49257          */
49258         'alphaMask' : /[a-z_]/i,
49259
49260         /**
49261          * The function used to validate alphanumeric values
49262          * @param {String} value The value
49263          */
49264         'alphanum' : function(v){
49265             return alphanum.test(v);
49266         },
49267         /**
49268          * The error text to display when the alphanumeric validation function returns false
49269          * @type String
49270          */
49271         'alphanumText' : 'This field should only contain letters, numbers and _',
49272         /**
49273          * The keystroke filter mask to be applied on alphanumeric input
49274          * @type RegExp
49275          */
49276         'alphanumMask' : /[a-z0-9_]/i
49277     };
49278 }();//<script type="text/javascript">
49279
49280 /**
49281  * @class Roo.form.FCKeditor
49282  * @extends Roo.form.TextArea
49283  * Wrapper around the FCKEditor http://www.fckeditor.net
49284  * @constructor
49285  * Creates a new FCKeditor
49286  * @param {Object} config Configuration options
49287  */
49288 Roo.form.FCKeditor = function(config){
49289     Roo.form.FCKeditor.superclass.constructor.call(this, config);
49290     this.addEvents({
49291          /**
49292          * @event editorinit
49293          * Fired when the editor is initialized - you can add extra handlers here..
49294          * @param {FCKeditor} this
49295          * @param {Object} the FCK object.
49296          */
49297         editorinit : true
49298     });
49299     
49300     
49301 };
49302 Roo.form.FCKeditor.editors = { };
49303 Roo.extend(Roo.form.FCKeditor, Roo.form.TextArea,
49304 {
49305     //defaultAutoCreate : {
49306     //    tag : "textarea",style   : "width:100px;height:60px;" ,autocomplete    : "off"
49307     //},
49308     // private
49309     /**
49310      * @cfg {Object} fck options - see fck manual for details.
49311      */
49312     fckconfig : false,
49313     
49314     /**
49315      * @cfg {Object} fck toolbar set (Basic or Default)
49316      */
49317     toolbarSet : 'Basic',
49318     /**
49319      * @cfg {Object} fck BasePath
49320      */ 
49321     basePath : '/fckeditor/',
49322     
49323     
49324     frame : false,
49325     
49326     value : '',
49327     
49328    
49329     onRender : function(ct, position)
49330     {
49331         if(!this.el){
49332             this.defaultAutoCreate = {
49333                 tag: "textarea",
49334                 style:"width:300px;height:60px;",
49335                 autocomplete: "new-password"
49336             };
49337         }
49338         Roo.form.FCKeditor.superclass.onRender.call(this, ct, position);
49339         /*
49340         if(this.grow){
49341             this.textSizeEl = Roo.DomHelper.append(document.body, {tag: "pre", cls: "x-form-grow-sizer"});
49342             if(this.preventScrollbars){
49343                 this.el.setStyle("overflow", "hidden");
49344             }
49345             this.el.setHeight(this.growMin);
49346         }
49347         */
49348         //console.log('onrender' + this.getId() );
49349         Roo.form.FCKeditor.editors[this.getId()] = this;
49350          
49351
49352         this.replaceTextarea() ;
49353         
49354     },
49355     
49356     getEditor : function() {
49357         return this.fckEditor;
49358     },
49359     /**
49360      * Sets a data value into the field and validates it.  To set the value directly without validation see {@link #setRawValue}.
49361      * @param {Mixed} value The value to set
49362      */
49363     
49364     
49365     setValue : function(value)
49366     {
49367         //console.log('setValue: ' + value);
49368         
49369         if(typeof(value) == 'undefined') { // not sure why this is happending...
49370             return;
49371         }
49372         Roo.form.FCKeditor.superclass.setValue.apply(this,[value]);
49373         
49374         //if(!this.el || !this.getEditor()) {
49375         //    this.value = value;
49376             //this.setValue.defer(100,this,[value]);    
49377         //    return;
49378         //} 
49379         
49380         if(!this.getEditor()) {
49381             return;
49382         }
49383         
49384         this.getEditor().SetData(value);
49385         
49386         //
49387
49388     },
49389
49390     /**
49391      * Returns the normalized data value (undefined or emptyText will be returned as '').  To return the raw value see {@link #getRawValue}.
49392      * @return {Mixed} value The field value
49393      */
49394     getValue : function()
49395     {
49396         
49397         if (this.frame && this.frame.dom.style.display == 'none') {
49398             return Roo.form.FCKeditor.superclass.getValue.call(this);
49399         }
49400         
49401         if(!this.el || !this.getEditor()) {
49402            
49403            // this.getValue.defer(100,this); 
49404             return this.value;
49405         }
49406        
49407         
49408         var value=this.getEditor().GetData();
49409         Roo.form.FCKeditor.superclass.setValue.apply(this,[value]);
49410         return Roo.form.FCKeditor.superclass.getValue.call(this);
49411         
49412
49413     },
49414
49415     /**
49416      * Returns the raw data value which may or may not be a valid, defined value.  To return a normalized value see {@link #getValue}.
49417      * @return {Mixed} value The field value
49418      */
49419     getRawValue : function()
49420     {
49421         if (this.frame && this.frame.dom.style.display == 'none') {
49422             return Roo.form.FCKeditor.superclass.getRawValue.call(this);
49423         }
49424         
49425         if(!this.el || !this.getEditor()) {
49426             //this.getRawValue.defer(100,this); 
49427             return this.value;
49428             return;
49429         }
49430         
49431         
49432         
49433         var value=this.getEditor().GetData();
49434         Roo.form.FCKeditor.superclass.setRawValue.apply(this,[value]);
49435         return Roo.form.FCKeditor.superclass.getRawValue.call(this);
49436          
49437     },
49438     
49439     setSize : function(w,h) {
49440         
49441         
49442         
49443         //if (this.frame && this.frame.dom.style.display == 'none') {
49444         //    Roo.form.FCKeditor.superclass.setSize.apply(this, [w, h]);
49445         //    return;
49446         //}
49447         //if(!this.el || !this.getEditor()) {
49448         //    this.setSize.defer(100,this, [w,h]); 
49449         //    return;
49450         //}
49451         
49452         
49453         
49454         Roo.form.FCKeditor.superclass.setSize.apply(this, [w, h]);
49455         
49456         this.frame.dom.setAttribute('width', w);
49457         this.frame.dom.setAttribute('height', h);
49458         this.frame.setSize(w,h);
49459         
49460     },
49461     
49462     toggleSourceEdit : function(value) {
49463         
49464       
49465          
49466         this.el.dom.style.display = value ? '' : 'none';
49467         this.frame.dom.style.display = value ?  'none' : '';
49468         
49469     },
49470     
49471     
49472     focus: function(tag)
49473     {
49474         if (this.frame.dom.style.display == 'none') {
49475             return Roo.form.FCKeditor.superclass.focus.call(this);
49476         }
49477         if(!this.el || !this.getEditor()) {
49478             this.focus.defer(100,this, [tag]); 
49479             return;
49480         }
49481         
49482         
49483         
49484         
49485         var tgs = this.getEditor().EditorDocument.getElementsByTagName(tag);
49486         this.getEditor().Focus();
49487         if (tgs.length) {
49488             if (!this.getEditor().Selection.GetSelection()) {
49489                 this.focus.defer(100,this, [tag]); 
49490                 return;
49491             }
49492             
49493             
49494             var r = this.getEditor().EditorDocument.createRange();
49495             r.setStart(tgs[0],0);
49496             r.setEnd(tgs[0],0);
49497             this.getEditor().Selection.GetSelection().removeAllRanges();
49498             this.getEditor().Selection.GetSelection().addRange(r);
49499             this.getEditor().Focus();
49500         }
49501         
49502     },
49503     
49504     
49505     
49506     replaceTextarea : function()
49507     {
49508         if ( document.getElementById( this.getId() + '___Frame' ) ) {
49509             return ;
49510         }
49511         //if ( !this.checkBrowser || this._isCompatibleBrowser() )
49512         //{
49513             // We must check the elements firstly using the Id and then the name.
49514         var oTextarea = document.getElementById( this.getId() );
49515         
49516         var colElementsByName = document.getElementsByName( this.getId() ) ;
49517          
49518         oTextarea.style.display = 'none' ;
49519
49520         if ( oTextarea.tabIndex ) {            
49521             this.TabIndex = oTextarea.tabIndex ;
49522         }
49523         
49524         this._insertHtmlBefore( this._getConfigHtml(), oTextarea ) ;
49525         this._insertHtmlBefore( this._getIFrameHtml(), oTextarea ) ;
49526         this.frame = Roo.get(this.getId() + '___Frame')
49527     },
49528     
49529     _getConfigHtml : function()
49530     {
49531         var sConfig = '' ;
49532
49533         for ( var o in this.fckconfig ) {
49534             sConfig += sConfig.length > 0  ? '&amp;' : '';
49535             sConfig += encodeURIComponent( o ) + '=' + encodeURIComponent( this.fckconfig[o] ) ;
49536         }
49537
49538         return '<input type="hidden" id="' + this.getId() + '___Config" value="' + sConfig + '" style="display:none" />' ;
49539     },
49540     
49541     
49542     _getIFrameHtml : function()
49543     {
49544         var sFile = 'fckeditor.html' ;
49545         /* no idea what this is about..
49546         try
49547         {
49548             if ( (/fcksource=true/i).test( window.top.location.search ) )
49549                 sFile = 'fckeditor.original.html' ;
49550         }
49551         catch (e) { 
49552         */
49553
49554         var sLink = this.basePath + 'editor/' + sFile + '?InstanceName=' + encodeURIComponent( this.getId() ) ;
49555         sLink += this.toolbarSet ? ( '&amp;Toolbar=' + this.toolbarSet)  : '';
49556         
49557         
49558         var html = '<iframe id="' + this.getId() +
49559             '___Frame" src="' + sLink +
49560             '" width="' + this.width +
49561             '" height="' + this.height + '"' +
49562             (this.tabIndex ?  ' tabindex="' + this.tabIndex + '"' :'' ) +
49563             ' frameborder="0" scrolling="no"></iframe>' ;
49564
49565         return html ;
49566     },
49567     
49568     _insertHtmlBefore : function( html, element )
49569     {
49570         if ( element.insertAdjacentHTML )       {
49571             // IE
49572             element.insertAdjacentHTML( 'beforeBegin', html ) ;
49573         } else { // Gecko
49574             var oRange = document.createRange() ;
49575             oRange.setStartBefore( element ) ;
49576             var oFragment = oRange.createContextualFragment( html );
49577             element.parentNode.insertBefore( oFragment, element ) ;
49578         }
49579     }
49580     
49581     
49582   
49583     
49584     
49585     
49586     
49587
49588 });
49589
49590 //Roo.reg('fckeditor', Roo.form.FCKeditor);
49591
49592 function FCKeditor_OnComplete(editorInstance){
49593     var f = Roo.form.FCKeditor.editors[editorInstance.Name];
49594     f.fckEditor = editorInstance;
49595     //console.log("loaded");
49596     f.fireEvent('editorinit', f, editorInstance);
49597
49598   
49599
49600  
49601
49602
49603
49604
49605
49606
49607
49608
49609
49610
49611
49612
49613
49614
49615
49616 //<script type="text/javascript">
49617 /**
49618  * @class Roo.form.GridField
49619  * @extends Roo.form.Field
49620  * Embed a grid (or editable grid into a form)
49621  * STATUS ALPHA
49622  * 
49623  * This embeds a grid in a form, the value of the field should be the json encoded array of rows
49624  * it needs 
49625  * xgrid.store = Roo.data.Store
49626  * xgrid.store.proxy = Roo.data.MemoryProxy (data = [] )
49627  * xgrid.store.reader = Roo.data.JsonReader 
49628  * 
49629  * 
49630  * @constructor
49631  * Creates a new GridField
49632  * @param {Object} config Configuration options
49633  */
49634 Roo.form.GridField = function(config){
49635     Roo.form.GridField.superclass.constructor.call(this, config);
49636      
49637 };
49638
49639 Roo.extend(Roo.form.GridField, Roo.form.Field,  {
49640     /**
49641      * @cfg {Number} width  - used to restrict width of grid..
49642      */
49643     width : 100,
49644     /**
49645      * @cfg {Number} height - used to restrict height of grid..
49646      */
49647     height : 50,
49648      /**
49649      * @cfg {Object} xgrid (xtype'd description of grid) { xtype : 'Grid', dataSource: .... }
49650          * 
49651          *}
49652      */
49653     xgrid : false, 
49654     /**
49655      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49656      * {tag: "input", type: "checkbox", autocomplete: "off"})
49657      */
49658    // defaultAutoCreate : { tag: 'div' },
49659     defaultAutoCreate : { tag: 'input', type: 'hidden', autocomplete: 'new-password'},
49660     /**
49661      * @cfg {String} addTitle Text to include for adding a title.
49662      */
49663     addTitle : false,
49664     //
49665     onResize : function(){
49666         Roo.form.Field.superclass.onResize.apply(this, arguments);
49667     },
49668
49669     initEvents : function(){
49670         // Roo.form.Checkbox.superclass.initEvents.call(this);
49671         // has no events...
49672        
49673     },
49674
49675
49676     getResizeEl : function(){
49677         return this.wrap;
49678     },
49679
49680     getPositionEl : function(){
49681         return this.wrap;
49682     },
49683
49684     // private
49685     onRender : function(ct, position){
49686         
49687         this.style = this.style || 'overflow: hidden; border:1px solid #c3daf9;';
49688         var style = this.style;
49689         delete this.style;
49690         
49691         Roo.form.GridField.superclass.onRender.call(this, ct, position);
49692         this.wrap = this.el.wrap({cls: ''}); // not sure why ive done thsi...
49693         this.viewEl = this.wrap.createChild({ tag: 'div' });
49694         if (style) {
49695             this.viewEl.applyStyles(style);
49696         }
49697         if (this.width) {
49698             this.viewEl.setWidth(this.width);
49699         }
49700         if (this.height) {
49701             this.viewEl.setHeight(this.height);
49702         }
49703         //if(this.inputValue !== undefined){
49704         //this.setValue(this.value);
49705         
49706         
49707         this.grid = new Roo.grid[this.xgrid.xtype](this.viewEl, this.xgrid);
49708         
49709         
49710         this.grid.render();
49711         this.grid.getDataSource().on('remove', this.refreshValue, this);
49712         this.grid.getDataSource().on('update', this.refreshValue, this);
49713         this.grid.on('afteredit', this.refreshValue, this);
49714  
49715     },
49716      
49717     
49718     /**
49719      * Sets the value of the item. 
49720      * @param {String} either an object  or a string..
49721      */
49722     setValue : function(v){
49723         //this.value = v;
49724         v = v || []; // empty set..
49725         // this does not seem smart - it really only affects memoryproxy grids..
49726         if (this.grid && this.grid.getDataSource() && typeof(v) != 'undefined') {
49727             var ds = this.grid.getDataSource();
49728             // assumes a json reader..
49729             var data = {}
49730             data[ds.reader.meta.root ] =  typeof(v) == 'string' ? Roo.decode(v) : v;
49731             ds.loadData( data);
49732         }
49733         // clear selection so it does not get stale.
49734         if (this.grid.sm) { 
49735             this.grid.sm.clearSelections();
49736         }
49737         
49738         Roo.form.GridField.superclass.setValue.call(this, v);
49739         this.refreshValue();
49740         // should load data in the grid really....
49741     },
49742     
49743     // private
49744     refreshValue: function() {
49745          var val = [];
49746         this.grid.getDataSource().each(function(r) {
49747             val.push(r.data);
49748         });
49749         this.el.dom.value = Roo.encode(val);
49750     }
49751     
49752      
49753     
49754     
49755 });/*
49756  * Based on:
49757  * Ext JS Library 1.1.1
49758  * Copyright(c) 2006-2007, Ext JS, LLC.
49759  *
49760  * Originally Released Under LGPL - original licence link has changed is not relivant.
49761  *
49762  * Fork - LGPL
49763  * <script type="text/javascript">
49764  */
49765 /**
49766  * @class Roo.form.DisplayField
49767  * @extends Roo.form.Field
49768  * A generic Field to display non-editable data.
49769  * @cfg {Boolean} closable (true|false) default false
49770  * @constructor
49771  * Creates a new Display Field item.
49772  * @param {Object} config Configuration options
49773  */
49774 Roo.form.DisplayField = function(config){
49775     Roo.form.DisplayField.superclass.constructor.call(this, config);
49776     
49777     this.addEvents({
49778         /**
49779          * @event close
49780          * Fires after the click the close btn
49781              * @param {Roo.form.DisplayField} this
49782              */
49783         close : true
49784     });
49785 };
49786
49787 Roo.extend(Roo.form.DisplayField, Roo.form.TextField,  {
49788     inputType:      'hidden',
49789     allowBlank:     true,
49790     readOnly:         true,
49791     
49792  
49793     /**
49794      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
49795      */
49796     focusClass : undefined,
49797     /**
49798      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
49799      */
49800     fieldClass: 'x-form-field',
49801     
49802      /**
49803      * @cfg {Function} valueRenderer The renderer for the field (so you can reformat output). should return raw HTML
49804      */
49805     valueRenderer: undefined,
49806     
49807     width: 100,
49808     /**
49809      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49810      * {tag: "input", type: "checkbox", autocomplete: "off"})
49811      */
49812      
49813  //   defaultAutoCreate : { tag: 'input', type: 'hidden', autocomplete: 'off'},
49814  
49815     closable : false,
49816     
49817     onResize : function(){
49818         Roo.form.DisplayField.superclass.onResize.apply(this, arguments);
49819         
49820     },
49821
49822     initEvents : function(){
49823         // Roo.form.Checkbox.superclass.initEvents.call(this);
49824         // has no events...
49825         
49826         if(this.closable){
49827             this.closeEl.on('click', this.onClose, this);
49828         }
49829        
49830     },
49831
49832
49833     getResizeEl : function(){
49834         return this.wrap;
49835     },
49836
49837     getPositionEl : function(){
49838         return this.wrap;
49839     },
49840
49841     // private
49842     onRender : function(ct, position){
49843         
49844         Roo.form.DisplayField.superclass.onRender.call(this, ct, position);
49845         //if(this.inputValue !== undefined){
49846         this.wrap = this.el.wrap();
49847         
49848         this.viewEl = this.wrap.createChild({ tag: 'div', cls: 'x-form-displayfield'});
49849         
49850         if(this.closable){
49851             this.closeEl = this.wrap.createChild({ tag: 'div', cls: 'x-dlg-close'});
49852         }
49853         
49854         if (this.bodyStyle) {
49855             this.viewEl.applyStyles(this.bodyStyle);
49856         }
49857         //this.viewEl.setStyle('padding', '2px');
49858         
49859         this.setValue(this.value);
49860         
49861     },
49862 /*
49863     // private
49864     initValue : Roo.emptyFn,
49865
49866   */
49867
49868         // private
49869     onClick : function(){
49870         
49871     },
49872
49873     /**
49874      * Sets the checked state of the checkbox.
49875      * @param {Boolean/String} checked True, 'true', '1', or 'on' to check the checkbox, any other value will uncheck it.
49876      */
49877     setValue : function(v){
49878         this.value = v;
49879         var html = this.valueRenderer ?  this.valueRenderer(v) : String.format('{0}', v);
49880         // this might be called before we have a dom element..
49881         if (!this.viewEl) {
49882             return;
49883         }
49884         this.viewEl.dom.innerHTML = html;
49885         Roo.form.DisplayField.superclass.setValue.call(this, v);
49886
49887     },
49888     
49889     onClose : function(e)
49890     {
49891         e.preventDefault();
49892         
49893         this.fireEvent('close', this);
49894     }
49895 });/*
49896  * 
49897  * Licence- LGPL
49898  * 
49899  */
49900
49901 /**
49902  * @class Roo.form.DayPicker
49903  * @extends Roo.form.Field
49904  * A Day picker show [M] [T] [W] ....
49905  * @constructor
49906  * Creates a new Day Picker
49907  * @param {Object} config Configuration options
49908  */
49909 Roo.form.DayPicker= function(config){
49910     Roo.form.DayPicker.superclass.constructor.call(this, config);
49911      
49912 };
49913
49914 Roo.extend(Roo.form.DayPicker, Roo.form.Field,  {
49915     /**
49916      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
49917      */
49918     focusClass : undefined,
49919     /**
49920      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
49921      */
49922     fieldClass: "x-form-field",
49923    
49924     /**
49925      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49926      * {tag: "input", type: "checkbox", autocomplete: "off"})
49927      */
49928     defaultAutoCreate : { tag: "input", type: 'hidden', autocomplete: "new-password"},
49929     
49930    
49931     actionMode : 'viewEl', 
49932     //
49933     // private
49934  
49935     inputType : 'hidden',
49936     
49937      
49938     inputElement: false, // real input element?
49939     basedOn: false, // ????
49940     
49941     isFormField: true, // not sure where this is needed!!!!
49942
49943     onResize : function(){
49944         Roo.form.Checkbox.superclass.onResize.apply(this, arguments);
49945         if(!this.boxLabel){
49946             this.el.alignTo(this.wrap, 'c-c');
49947         }
49948     },
49949
49950     initEvents : function(){
49951         Roo.form.Checkbox.superclass.initEvents.call(this);
49952         this.el.on("click", this.onClick,  this);
49953         this.el.on("change", this.onClick,  this);
49954     },
49955
49956
49957     getResizeEl : function(){
49958         return this.wrap;
49959     },
49960
49961     getPositionEl : function(){
49962         return this.wrap;
49963     },
49964
49965     
49966     // private
49967     onRender : function(ct, position){
49968         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
49969        
49970         this.wrap = this.el.wrap({cls: 'x-form-daypick-item '});
49971         
49972         var r1 = '<table><tr>';
49973         var r2 = '<tr class="x-form-daypick-icons">';
49974         for (var i=0; i < 7; i++) {
49975             r1+= '<td><div>' + Date.dayNames[i].substring(0,3) + '</div></td>';
49976             r2+= '<td><img class="x-menu-item-icon" src="' + Roo.BLANK_IMAGE_URL  +'"></td>';
49977         }
49978         
49979         var viewEl = this.wrap.createChild( r1 + '</tr>' + r2 + '</tr></table>');
49980         viewEl.select('img').on('click', this.onClick, this);
49981         this.viewEl = viewEl;   
49982         
49983         
49984         // this will not work on Chrome!!!
49985         this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
49986         this.el.on('propertychange', this.setFromHidden,  this);  //ie
49987         
49988         
49989           
49990
49991     },
49992
49993     // private
49994     initValue : Roo.emptyFn,
49995
49996     /**
49997      * Returns the checked state of the checkbox.
49998      * @return {Boolean} True if checked, else false
49999      */
50000     getValue : function(){
50001         return this.el.dom.value;
50002         
50003     },
50004
50005         // private
50006     onClick : function(e){ 
50007         //this.setChecked(!this.checked);
50008         Roo.get(e.target).toggleClass('x-menu-item-checked');
50009         this.refreshValue();
50010         //if(this.el.dom.checked != this.checked){
50011         //    this.setValue(this.el.dom.checked);
50012        // }
50013     },
50014     
50015     // private
50016     refreshValue : function()
50017     {
50018         var val = '';
50019         this.viewEl.select('img',true).each(function(e,i,n)  {
50020             val += e.is(".x-menu-item-checked") ? String(n) : '';
50021         });
50022         this.setValue(val, true);
50023     },
50024
50025     /**
50026      * Sets the checked state of the checkbox.
50027      * On is always based on a string comparison between inputValue and the param.
50028      * @param {Boolean/String} value - the value to set 
50029      * @param {Boolean/String} suppressEvent - whether to suppress the checkchange event.
50030      */
50031     setValue : function(v,suppressEvent){
50032         if (!this.el.dom) {
50033             return;
50034         }
50035         var old = this.el.dom.value ;
50036         this.el.dom.value = v;
50037         if (suppressEvent) {
50038             return ;
50039         }
50040          
50041         // update display..
50042         this.viewEl.select('img',true).each(function(e,i,n)  {
50043             
50044             var on = e.is(".x-menu-item-checked");
50045             var newv = v.indexOf(String(n)) > -1;
50046             if (on != newv) {
50047                 e.toggleClass('x-menu-item-checked');
50048             }
50049             
50050         });
50051         
50052         
50053         this.fireEvent('change', this, v, old);
50054         
50055         
50056     },
50057    
50058     // handle setting of hidden value by some other method!!?!?
50059     setFromHidden: function()
50060     {
50061         if(!this.el){
50062             return;
50063         }
50064         //console.log("SET FROM HIDDEN");
50065         //alert('setFrom hidden');
50066         this.setValue(this.el.dom.value);
50067     },
50068     
50069     onDestroy : function()
50070     {
50071         if(this.viewEl){
50072             Roo.get(this.viewEl).remove();
50073         }
50074          
50075         Roo.form.DayPicker.superclass.onDestroy.call(this);
50076     }
50077
50078 });/*
50079  * RooJS Library 1.1.1
50080  * Copyright(c) 2008-2011  Alan Knowles
50081  *
50082  * License - LGPL
50083  */
50084  
50085
50086 /**
50087  * @class Roo.form.ComboCheck
50088  * @extends Roo.form.ComboBox
50089  * A combobox for multiple select items.
50090  *
50091  * FIXME - could do with a reset button..
50092  * 
50093  * @constructor
50094  * Create a new ComboCheck
50095  * @param {Object} config Configuration options
50096  */
50097 Roo.form.ComboCheck = function(config){
50098     Roo.form.ComboCheck.superclass.constructor.call(this, config);
50099     // should verify some data...
50100     // like
50101     // hiddenName = required..
50102     // displayField = required
50103     // valudField == required
50104     var req= [ 'hiddenName', 'displayField', 'valueField' ];
50105     var _t = this;
50106     Roo.each(req, function(e) {
50107         if ((typeof(_t[e]) == 'undefined' ) || !_t[e].length) {
50108             throw "Roo.form.ComboCheck : missing value for: " + e;
50109         }
50110     });
50111     
50112     
50113 };
50114
50115 Roo.extend(Roo.form.ComboCheck, Roo.form.ComboBox, {
50116      
50117      
50118     editable : false,
50119      
50120     selectedClass: 'x-menu-item-checked', 
50121     
50122     // private
50123     onRender : function(ct, position){
50124         var _t = this;
50125         
50126         
50127         
50128         if(!this.tpl){
50129             var cls = 'x-combo-list';
50130
50131             
50132             this.tpl =  new Roo.Template({
50133                 html :  '<div class="'+cls+'-item x-menu-check-item">' +
50134                    '<img class="x-menu-item-icon" style="margin: 0px;" src="' + Roo.BLANK_IMAGE_URL + '">' + 
50135                    '<span>{' + this.displayField + '}</span>' +
50136                     '</div>' 
50137                 
50138             });
50139         }
50140  
50141         
50142         Roo.form.ComboCheck.superclass.onRender.call(this, ct, position);
50143         this.view.singleSelect = false;
50144         this.view.multiSelect = true;
50145         this.view.toggleSelect = true;
50146         this.pageTb.add(new Roo.Toolbar.Fill(), {
50147             
50148             text: 'Done',
50149             handler: function()
50150             {
50151                 _t.collapse();
50152             }
50153         });
50154     },
50155     
50156     onViewOver : function(e, t){
50157         // do nothing...
50158         return;
50159         
50160     },
50161     
50162     onViewClick : function(doFocus,index){
50163         return;
50164         
50165     },
50166     select: function () {
50167         //Roo.log("SELECT CALLED");
50168     },
50169      
50170     selectByValue : function(xv, scrollIntoView){
50171         var ar = this.getValueArray();
50172         var sels = [];
50173         
50174         Roo.each(ar, function(v) {
50175             if(v === undefined || v === null){
50176                 return;
50177             }
50178             var r = this.findRecord(this.valueField, v);
50179             if(r){
50180                 sels.push(this.store.indexOf(r))
50181                 
50182             }
50183         },this);
50184         this.view.select(sels);
50185         return false;
50186     },
50187     
50188     
50189     
50190     onSelect : function(record, index){
50191        // Roo.log("onselect Called");
50192        // this is only called by the clear button now..
50193         this.view.clearSelections();
50194         this.setValue('[]');
50195         if (this.value != this.valueBefore) {
50196             this.fireEvent('change', this, this.value, this.valueBefore);
50197             this.valueBefore = this.value;
50198         }
50199     },
50200     getValueArray : function()
50201     {
50202         var ar = [] ;
50203         
50204         try {
50205             //Roo.log(this.value);
50206             if (typeof(this.value) == 'undefined') {
50207                 return [];
50208             }
50209             var ar = Roo.decode(this.value);
50210             return  ar instanceof Array ? ar : []; //?? valid?
50211             
50212         } catch(e) {
50213             Roo.log(e + "\nRoo.form.ComboCheck:getValueArray  invalid data:" + this.getValue());
50214             return [];
50215         }
50216          
50217     },
50218     expand : function ()
50219     {
50220         
50221         Roo.form.ComboCheck.superclass.expand.call(this);
50222         this.valueBefore = typeof(this.value) == 'undefined' ? '' : this.value;
50223         //this.valueBefore = typeof(this.valueBefore) == 'undefined' ? '' : this.valueBefore;
50224         
50225
50226     },
50227     
50228     collapse : function(){
50229         Roo.form.ComboCheck.superclass.collapse.call(this);
50230         var sl = this.view.getSelectedIndexes();
50231         var st = this.store;
50232         var nv = [];
50233         var tv = [];
50234         var r;
50235         Roo.each(sl, function(i) {
50236             r = st.getAt(i);
50237             nv.push(r.get(this.valueField));
50238         },this);
50239         this.setValue(Roo.encode(nv));
50240         if (this.value != this.valueBefore) {
50241
50242             this.fireEvent('change', this, this.value, this.valueBefore);
50243             this.valueBefore = this.value;
50244         }
50245         
50246     },
50247     
50248     setValue : function(v){
50249         // Roo.log(v);
50250         this.value = v;
50251         
50252         var vals = this.getValueArray();
50253         var tv = [];
50254         Roo.each(vals, function(k) {
50255             var r = this.findRecord(this.valueField, k);
50256             if(r){
50257                 tv.push(r.data[this.displayField]);
50258             }else if(this.valueNotFoundText !== undefined){
50259                 tv.push( this.valueNotFoundText );
50260             }
50261         },this);
50262        // Roo.log(tv);
50263         
50264         Roo.form.ComboBox.superclass.setValue.call(this, tv.join(', '));
50265         this.hiddenField.value = v;
50266         this.value = v;
50267     }
50268     
50269 });/*
50270  * Based on:
50271  * Ext JS Library 1.1.1
50272  * Copyright(c) 2006-2007, Ext JS, LLC.
50273  *
50274  * Originally Released Under LGPL - original licence link has changed is not relivant.
50275  *
50276  * Fork - LGPL
50277  * <script type="text/javascript">
50278  */
50279  
50280 /**
50281  * @class Roo.form.Signature
50282  * @extends Roo.form.Field
50283  * Signature field.  
50284  * @constructor
50285  * 
50286  * @param {Object} config Configuration options
50287  */
50288
50289 Roo.form.Signature = function(config){
50290     Roo.form.Signature.superclass.constructor.call(this, config);
50291     
50292     this.addEvents({// not in used??
50293          /**
50294          * @event confirm
50295          * Fires when the 'confirm' icon is pressed (add a listener to enable add button)
50296              * @param {Roo.form.Signature} combo This combo box
50297              */
50298         'confirm' : true,
50299         /**
50300          * @event reset
50301          * Fires when the 'edit' icon is pressed (add a listener to enable add button)
50302              * @param {Roo.form.ComboBox} combo This combo box
50303              * @param {Roo.data.Record|false} record The data record returned from the underlying store (or false on nothing selected)
50304              */
50305         'reset' : true
50306     });
50307 };
50308
50309 Roo.extend(Roo.form.Signature, Roo.form.Field,  {
50310     /**
50311      * @cfg {Object} labels Label to use when rendering a form.
50312      * defaults to 
50313      * labels : { 
50314      *      clear : "Clear",
50315      *      confirm : "Confirm"
50316      *  }
50317      */
50318     labels : { 
50319         clear : "Clear",
50320         confirm : "Confirm"
50321     },
50322     /**
50323      * @cfg {Number} width The signature panel width (defaults to 300)
50324      */
50325     width: 300,
50326     /**
50327      * @cfg {Number} height The signature panel height (defaults to 100)
50328      */
50329     height : 100,
50330     /**
50331      * @cfg {Boolean} allowBlank False to validate that the value length > 0 (defaults to false)
50332      */
50333     allowBlank : false,
50334     
50335     //private
50336     // {Object} signPanel The signature SVG panel element (defaults to {})
50337     signPanel : {},
50338     // {Boolean} isMouseDown False to validate that the mouse down event (defaults to false)
50339     isMouseDown : false,
50340     // {Boolean} isConfirmed validate the signature is confirmed or not for submitting form (defaults to false)
50341     isConfirmed : false,
50342     // {String} signatureTmp SVG mapping string (defaults to empty string)
50343     signatureTmp : '',
50344     
50345     
50346     defaultAutoCreate : { // modified by initCompnoent..
50347         tag: "input",
50348         type:"hidden"
50349     },
50350
50351     // private
50352     onRender : function(ct, position){
50353         
50354         Roo.form.Signature.superclass.onRender.call(this, ct, position);
50355         
50356         this.wrap = this.el.wrap({
50357             cls:'x-form-signature-wrap', style : 'width: ' + this.width + 'px', cn:{cls:'x-form-signature'}
50358         });
50359         
50360         this.createToolbar(this);
50361         this.signPanel = this.wrap.createChild({
50362                 tag: 'div',
50363                 style: 'width: ' + this.width + 'px; height: ' + this.height + 'px; border: 0;'
50364             }, this.el
50365         );
50366             
50367         this.svgID = Roo.id();
50368         this.svgEl = this.signPanel.createChild({
50369               xmlns : 'http://www.w3.org/2000/svg',
50370               tag : 'svg',
50371               id : this.svgID + "-svg",
50372               width: this.width,
50373               height: this.height,
50374               viewBox: '0 0 '+this.width+' '+this.height,
50375               cn : [
50376                 {
50377                     tag: "rect",
50378                     id: this.svgID + "-svg-r",
50379                     width: this.width,
50380                     height: this.height,
50381                     fill: "#ffa"
50382                 },
50383                 {
50384                     tag: "line",
50385                     id: this.svgID + "-svg-l",
50386                     x1: "0", // start
50387                     y1: (this.height*0.8), // start set the line in 80% of height
50388                     x2: this.width, // end
50389                     y2: (this.height*0.8), // end set the line in 80% of height
50390                     'stroke': "#666",
50391                     'stroke-width': "1",
50392                     'stroke-dasharray': "3",
50393                     'shape-rendering': "crispEdges",
50394                     'pointer-events': "none"
50395                 },
50396                 {
50397                     tag: "path",
50398                     id: this.svgID + "-svg-p",
50399                     'stroke': "navy",
50400                     'stroke-width': "3",
50401                     'fill': "none",
50402                     'pointer-events': 'none'
50403                 }
50404               ]
50405         });
50406         this.createSVG();
50407         this.svgBox = this.svgEl.dom.getScreenCTM();
50408     },
50409     createSVG : function(){ 
50410         var svg = this.signPanel;
50411         var r = svg.select('#'+ this.svgID + '-svg-r', true).first().dom;
50412         var t = this;
50413
50414         r.addEventListener('mousedown', function(e) { return t.down(e); }, false);
50415         r.addEventListener('mousemove', function(e) { return t.move(e); }, false);
50416         r.addEventListener('mouseup', function(e) { return t.up(e); }, false);
50417         r.addEventListener('mouseout', function(e) { return t.up(e); }, false);
50418         r.addEventListener('touchstart', function(e) { return t.down(e); }, false);
50419         r.addEventListener('touchmove', function(e) { return t.move(e); }, false);
50420         r.addEventListener('touchend', function(e) { return t.up(e); }, false);
50421         
50422     },
50423     isTouchEvent : function(e){
50424         return e.type.match(/^touch/);
50425     },
50426     getCoords : function (e) {
50427         var pt    = this.svgEl.dom.createSVGPoint();
50428         pt.x = e.clientX; 
50429         pt.y = e.clientY;
50430         if (this.isTouchEvent(e)) {
50431             pt.x =  e.targetTouches[0].clientX;
50432             pt.y = e.targetTouches[0].clientY;
50433         }
50434         var a = this.svgEl.dom.getScreenCTM();
50435         var b = a.inverse();
50436         var mx = pt.matrixTransform(b);
50437         return mx.x + ',' + mx.y;
50438     },
50439     //mouse event headler 
50440     down : function (e) {
50441         this.signatureTmp += 'M' + this.getCoords(e) + ' ';
50442         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr('d', this.signatureTmp);
50443         
50444         this.isMouseDown = true;
50445         
50446         e.preventDefault();
50447     },
50448     move : function (e) {
50449         if (this.isMouseDown) {
50450             this.signatureTmp += 'L' + this.getCoords(e) + ' ';
50451             this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', this.signatureTmp);
50452         }
50453         
50454         e.preventDefault();
50455     },
50456     up : function (e) {
50457         this.isMouseDown = false;
50458         var sp = this.signatureTmp.split(' ');
50459         
50460         if(sp.length > 1){
50461             if(!sp[sp.length-2].match(/^L/)){
50462                 sp.pop();
50463                 sp.pop();
50464                 sp.push("");
50465                 this.signatureTmp = sp.join(" ");
50466             }
50467         }
50468         if(this.getValue() != this.signatureTmp){
50469             this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50470             this.isConfirmed = false;
50471         }
50472         e.preventDefault();
50473     },
50474     
50475     /**
50476      * Protected method that will not generally be called directly. It
50477      * is called when the editor creates its toolbar. Override this method if you need to
50478      * add custom toolbar buttons.
50479      * @param {HtmlEditor} editor
50480      */
50481     createToolbar : function(editor){
50482          function btn(id, toggle, handler){
50483             var xid = fid + '-'+ id ;
50484             return {
50485                 id : xid,
50486                 cmd : id,
50487                 cls : 'x-btn-icon x-edit-'+id,
50488                 enableToggle:toggle !== false,
50489                 scope: editor, // was editor...
50490                 handler:handler||editor.relayBtnCmd,
50491                 clickEvent:'mousedown',
50492                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
50493                 tabIndex:-1
50494             };
50495         }
50496         
50497         
50498         var tb = new Roo.Toolbar(editor.wrap.dom.firstChild);
50499         this.tb = tb;
50500         this.tb.add(
50501            {
50502                 cls : ' x-signature-btn x-signature-'+id,
50503                 scope: editor, // was editor...
50504                 handler: this.reset,
50505                 clickEvent:'mousedown',
50506                 text: this.labels.clear
50507             },
50508             {
50509                  xtype : 'Fill',
50510                  xns: Roo.Toolbar
50511             }, 
50512             {
50513                 cls : '  x-signature-btn x-signature-'+id,
50514                 scope: editor, // was editor...
50515                 handler: this.confirmHandler,
50516                 clickEvent:'mousedown',
50517                 text: this.labels.confirm
50518             }
50519         );
50520     
50521     },
50522     //public
50523     /**
50524      * when user is clicked confirm then show this image.....
50525      * 
50526      * @return {String} Image Data URI
50527      */
50528     getImageDataURI : function(){
50529         var svg = this.svgEl.dom.parentNode.innerHTML;
50530         var src = 'data:image/svg+xml;base64,'+window.btoa(svg);
50531         return src; 
50532     },
50533     /**
50534      * 
50535      * @return {Boolean} this.isConfirmed
50536      */
50537     getConfirmed : function(){
50538         return this.isConfirmed;
50539     },
50540     /**
50541      * 
50542      * @return {Number} this.width
50543      */
50544     getWidth : function(){
50545         return this.width;
50546     },
50547     /**
50548      * 
50549      * @return {Number} this.height
50550      */
50551     getHeight : function(){
50552         return this.height;
50553     },
50554     // private
50555     getSignature : function(){
50556         return this.signatureTmp;
50557     },
50558     // private
50559     reset : function(){
50560         this.signatureTmp = '';
50561         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50562         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', '');
50563         this.isConfirmed = false;
50564         Roo.form.Signature.superclass.reset.call(this);
50565     },
50566     setSignature : function(s){
50567         this.signatureTmp = s;
50568         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50569         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', s);
50570         this.setValue(s);
50571         this.isConfirmed = false;
50572         Roo.form.Signature.superclass.reset.call(this);
50573     }, 
50574     test : function(){
50575 //        Roo.log(this.signPanel.dom.contentWindow.up())
50576     },
50577     //private
50578     setConfirmed : function(){
50579         
50580         
50581         
50582 //        Roo.log(Roo.get(this.signPanel.dom.contentWindow.r).attr('fill', '#cfc'));
50583     },
50584     // private
50585     confirmHandler : function(){
50586         if(!this.getSignature()){
50587             return;
50588         }
50589         
50590         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#cfc');
50591         this.setValue(this.getSignature());
50592         this.isConfirmed = true;
50593         
50594         this.fireEvent('confirm', this);
50595     },
50596     // private
50597     // Subclasses should provide the validation implementation by overriding this
50598     validateValue : function(value){
50599         if(this.allowBlank){
50600             return true;
50601         }
50602         
50603         if(this.isConfirmed){
50604             return true;
50605         }
50606         return false;
50607     }
50608 });/*
50609  * Based on:
50610  * Ext JS Library 1.1.1
50611  * Copyright(c) 2006-2007, Ext JS, LLC.
50612  *
50613  * Originally Released Under LGPL - original licence link has changed is not relivant.
50614  *
50615  * Fork - LGPL
50616  * <script type="text/javascript">
50617  */
50618  
50619
50620 /**
50621  * @class Roo.form.ComboBox
50622  * @extends Roo.form.TriggerField
50623  * A combobox control with support for autocomplete, remote-loading, paging and many other features.
50624  * @constructor
50625  * Create a new ComboBox.
50626  * @param {Object} config Configuration options
50627  */
50628 Roo.form.Select = function(config){
50629     Roo.form.Select.superclass.constructor.call(this, config);
50630      
50631 };
50632
50633 Roo.extend(Roo.form.Select , Roo.form.ComboBox, {
50634     /**
50635      * @cfg {String/HTMLElement/Element} transform The id, DOM node or element of an existing select to convert to a ComboBox
50636      */
50637     /**
50638      * @cfg {Boolean} lazyRender True to prevent the ComboBox from rendering until requested (should always be used when
50639      * rendering into an Roo.Editor, defaults to false)
50640      */
50641     /**
50642      * @cfg {Boolean/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to:
50643      * {tag: "input", type: "text", size: "24", autocomplete: "off"})
50644      */
50645     /**
50646      * @cfg {Roo.data.Store} store The data store to which this combo is bound (defaults to undefined)
50647      */
50648     /**
50649      * @cfg {String} title If supplied, a header element is created containing this text and added into the top of
50650      * the dropdown list (defaults to undefined, with no header element)
50651      */
50652
50653      /**
50654      * @cfg {String/Roo.Template} tpl The template to use to render the output
50655      */
50656      
50657     // private
50658     defaultAutoCreate : {tag: "select"  },
50659     /**
50660      * @cfg {Number} listWidth The width in pixels of the dropdown list (defaults to the width of the ComboBox field)
50661      */
50662     listWidth: undefined,
50663     /**
50664      * @cfg {String} displayField The underlying data field name to bind to this CombBox (defaults to undefined if
50665      * mode = 'remote' or 'text' if mode = 'local')
50666      */
50667     displayField: undefined,
50668     /**
50669      * @cfg {String} valueField The underlying data value name to bind to this CombBox (defaults to undefined if
50670      * mode = 'remote' or 'value' if mode = 'local'). 
50671      * Note: use of a valueField requires the user make a selection
50672      * in order for a value to be mapped.
50673      */
50674     valueField: undefined,
50675     
50676     
50677     /**
50678      * @cfg {String} hiddenName If specified, a hidden form field with this name is dynamically generated to store the
50679      * field's data value (defaults to the underlying DOM element's name)
50680      */
50681     hiddenName: undefined,
50682     /**
50683      * @cfg {String} listClass CSS class to apply to the dropdown list element (defaults to '')
50684      */
50685     listClass: '',
50686     /**
50687      * @cfg {String} selectedClass CSS class to apply to the selected item in the dropdown list (defaults to 'x-combo-selected')
50688      */
50689     selectedClass: 'x-combo-selected',
50690     /**
50691      * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
50692      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-arrow-trigger'
50693      * which displays a downward arrow icon).
50694      */
50695     triggerClass : 'x-form-arrow-trigger',
50696     /**
50697      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
50698      */
50699     shadow:'sides',
50700     /**
50701      * @cfg {String} listAlign A valid anchor position value. See {@link Roo.Element#alignTo} for details on supported
50702      * anchor positions (defaults to 'tl-bl')
50703      */
50704     listAlign: 'tl-bl?',
50705     /**
50706      * @cfg {Number} maxHeight The maximum height in pixels of the dropdown list before scrollbars are shown (defaults to 300)
50707      */
50708     maxHeight: 300,
50709     /**
50710      * @cfg {String} triggerAction The action to execute when the trigger field is activated.  Use 'all' to run the
50711      * query specified by the allQuery config option (defaults to 'query')
50712      */
50713     triggerAction: 'query',
50714     /**
50715      * @cfg {Number} minChars The minimum number of characters the user must type before autocomplete and typeahead activate
50716      * (defaults to 4, does not apply if editable = false)
50717      */
50718     minChars : 4,
50719     /**
50720      * @cfg {Boolean} typeAhead True to populate and autoselect the remainder of the text being typed after a configurable
50721      * delay (typeAheadDelay) if it matches a known value (defaults to false)
50722      */
50723     typeAhead: false,
50724     /**
50725      * @cfg {Number} queryDelay The length of time in milliseconds to delay between the start of typing and sending the
50726      * query to filter the dropdown list (defaults to 500 if mode = 'remote' or 10 if mode = 'local')
50727      */
50728     queryDelay: 500,
50729     /**
50730      * @cfg {Number} pageSize If greater than 0, a paging toolbar is displayed in the footer of the dropdown list and the
50731      * filter queries will execute with page start and limit parameters.  Only applies when mode = 'remote' (defaults to 0)
50732      */
50733     pageSize: 0,
50734     /**
50735      * @cfg {Boolean} selectOnFocus True to select any existing text in the field immediately on focus.  Only applies
50736      * when editable = true (defaults to false)
50737      */
50738     selectOnFocus:false,
50739     /**
50740      * @cfg {String} queryParam Name of the query as it will be passed on the querystring (defaults to 'query')
50741      */
50742     queryParam: 'query',
50743     /**
50744      * @cfg {String} loadingText The text to display in the dropdown list while data is loading.  Only applies
50745      * when mode = 'remote' (defaults to 'Loading...')
50746      */
50747     loadingText: 'Loading...',
50748     /**
50749      * @cfg {Boolean} resizable True to add a resize handle to the bottom of the dropdown list (defaults to false)
50750      */
50751     resizable: false,
50752     /**
50753      * @cfg {Number} handleHeight The height in pixels of the dropdown list resize handle if resizable = true (defaults to 8)
50754      */
50755     handleHeight : 8,
50756     /**
50757      * @cfg {Boolean} editable False to prevent the user from typing text directly into the field, just like a
50758      * traditional select (defaults to true)
50759      */
50760     editable: true,
50761     /**
50762      * @cfg {String} allQuery The text query to send to the server to return all records for the list with no filtering (defaults to '')
50763      */
50764     allQuery: '',
50765     /**
50766      * @cfg {String} mode Set to 'local' if the ComboBox loads local data (defaults to 'remote' which loads from the server)
50767      */
50768     mode: 'remote',
50769     /**
50770      * @cfg {Number} minListWidth The minimum width of the dropdown list in pixels (defaults to 70, will be ignored if
50771      * listWidth has a higher value)
50772      */
50773     minListWidth : 70,
50774     /**
50775      * @cfg {Boolean} forceSelection True to restrict the selected value to one of the values in the list, false to
50776      * allow the user to set arbitrary text into the field (defaults to false)
50777      */
50778     forceSelection:false,
50779     /**
50780      * @cfg {Number} typeAheadDelay The length of time in milliseconds to wait until the typeahead text is displayed
50781      * if typeAhead = true (defaults to 250)
50782      */
50783     typeAheadDelay : 250,
50784     /**
50785      * @cfg {String} valueNotFoundText When using a name/value combo, if the value passed to setValue is not found in
50786      * the store, valueNotFoundText will be displayed as the field text if defined (defaults to undefined)
50787      */
50788     valueNotFoundText : undefined,
50789     
50790     /**
50791      * @cfg {String} defaultValue The value displayed after loading the store.
50792      */
50793     defaultValue: '',
50794     
50795     /**
50796      * @cfg {Boolean} blockFocus Prevents all focus calls, so it can work with things like HTML edtor bar
50797      */
50798     blockFocus : false,
50799     
50800     /**
50801      * @cfg {Boolean} disableClear Disable showing of clear button.
50802      */
50803     disableClear : false,
50804     /**
50805      * @cfg {Boolean} alwaysQuery  Disable caching of results, and always send query
50806      */
50807     alwaysQuery : false,
50808     
50809     //private
50810     addicon : false,
50811     editicon: false,
50812     
50813     // element that contains real text value.. (when hidden is used..)
50814      
50815     // private
50816     onRender : function(ct, position){
50817         Roo.form.Field.prototype.onRender.call(this, ct, position);
50818         
50819         if(this.store){
50820             this.store.on('beforeload', this.onBeforeLoad, this);
50821             this.store.on('load', this.onLoad, this);
50822             this.store.on('loadexception', this.onLoadException, this);
50823             this.store.load({});
50824         }
50825         
50826         
50827         
50828     },
50829
50830     // private
50831     initEvents : function(){
50832         //Roo.form.ComboBox.superclass.initEvents.call(this);
50833  
50834     },
50835
50836     onDestroy : function(){
50837        
50838         if(this.store){
50839             this.store.un('beforeload', this.onBeforeLoad, this);
50840             this.store.un('load', this.onLoad, this);
50841             this.store.un('loadexception', this.onLoadException, this);
50842         }
50843         //Roo.form.ComboBox.superclass.onDestroy.call(this);
50844     },
50845
50846     // private
50847     fireKey : function(e){
50848         if(e.isNavKeyPress() && !this.list.isVisible()){
50849             this.fireEvent("specialkey", this, e);
50850         }
50851     },
50852
50853     // private
50854     onResize: function(w, h){
50855         
50856         return; 
50857     
50858         
50859     },
50860
50861     /**
50862      * Allow or prevent the user from directly editing the field text.  If false is passed,
50863      * the user will only be able to select from the items defined in the dropdown list.  This method
50864      * is the runtime equivalent of setting the 'editable' config option at config time.
50865      * @param {Boolean} value True to allow the user to directly edit the field text
50866      */
50867     setEditable : function(value){
50868          
50869     },
50870
50871     // private
50872     onBeforeLoad : function(){
50873         
50874         Roo.log("Select before load");
50875         return;
50876     
50877         this.innerList.update(this.loadingText ?
50878                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
50879         //this.restrictHeight();
50880         this.selectedIndex = -1;
50881     },
50882
50883     // private
50884     onLoad : function(){
50885
50886     
50887         var dom = this.el.dom;
50888         dom.innerHTML = '';
50889          var od = dom.ownerDocument;
50890          
50891         if (this.emptyText) {
50892             var op = od.createElement('option');
50893             op.setAttribute('value', '');
50894             op.innerHTML = String.format('{0}', this.emptyText);
50895             dom.appendChild(op);
50896         }
50897         if(this.store.getCount() > 0){
50898            
50899             var vf = this.valueField;
50900             var df = this.displayField;
50901             this.store.data.each(function(r) {
50902                 // which colmsn to use... testing - cdoe / title..
50903                 var op = od.createElement('option');
50904                 op.setAttribute('value', r.data[vf]);
50905                 op.innerHTML = String.format('{0}', r.data[df]);
50906                 dom.appendChild(op);
50907             });
50908             if (typeof(this.defaultValue != 'undefined')) {
50909                 this.setValue(this.defaultValue);
50910             }
50911             
50912              
50913         }else{
50914             //this.onEmptyResults();
50915         }
50916         //this.el.focus();
50917     },
50918     // private
50919     onLoadException : function()
50920     {
50921         dom.innerHTML = '';
50922             
50923         Roo.log("Select on load exception");
50924         return;
50925     
50926         this.collapse();
50927         Roo.log(this.store.reader.jsonData);
50928         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
50929             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
50930         }
50931         
50932         
50933     },
50934     // private
50935     onTypeAhead : function(){
50936          
50937     },
50938
50939     // private
50940     onSelect : function(record, index){
50941         Roo.log('on select?');
50942         return;
50943         if(this.fireEvent('beforeselect', this, record, index) !== false){
50944             this.setFromData(index > -1 ? record.data : false);
50945             this.collapse();
50946             this.fireEvent('select', this, record, index);
50947         }
50948     },
50949
50950     /**
50951      * Returns the currently selected field value or empty string if no value is set.
50952      * @return {String} value The selected value
50953      */
50954     getValue : function(){
50955         var dom = this.el.dom;
50956         this.value = dom.options[dom.selectedIndex].value;
50957         return this.value;
50958         
50959     },
50960
50961     /**
50962      * Clears any text/value currently set in the field
50963      */
50964     clearValue : function(){
50965         this.value = '';
50966         this.el.dom.selectedIndex = this.emptyText ? 0 : -1;
50967         
50968     },
50969
50970     /**
50971      * Sets the specified value into the field.  If the value finds a match, the corresponding record text
50972      * will be displayed in the field.  If the value does not match the data value of an existing item,
50973      * and the valueNotFoundText config option is defined, it will be displayed as the default field text.
50974      * Otherwise the field will be blank (although the value will still be set).
50975      * @param {String} value The value to match
50976      */
50977     setValue : function(v){
50978         var d = this.el.dom;
50979         for (var i =0; i < d.options.length;i++) {
50980             if (v == d.options[i].value) {
50981                 d.selectedIndex = i;
50982                 this.value = v;
50983                 return;
50984             }
50985         }
50986         this.clearValue();
50987     },
50988     /**
50989      * @property {Object} the last set data for the element
50990      */
50991     
50992     lastData : false,
50993     /**
50994      * Sets the value of the field based on a object which is related to the record format for the store.
50995      * @param {Object} value the value to set as. or false on reset?
50996      */
50997     setFromData : function(o){
50998         Roo.log('setfrom data?');
50999          
51000         
51001         
51002     },
51003     // private
51004     reset : function(){
51005         this.clearValue();
51006     },
51007     // private
51008     findRecord : function(prop, value){
51009         
51010         return false;
51011     
51012         var record;
51013         if(this.store.getCount() > 0){
51014             this.store.each(function(r){
51015                 if(r.data[prop] == value){
51016                     record = r;
51017                     return false;
51018                 }
51019                 return true;
51020             });
51021         }
51022         return record;
51023     },
51024     
51025     getName: function()
51026     {
51027         // returns hidden if it's set..
51028         if (!this.rendered) {return ''};
51029         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
51030         
51031     },
51032      
51033
51034     
51035
51036     // private
51037     onEmptyResults : function(){
51038         Roo.log('empty results');
51039         //this.collapse();
51040     },
51041
51042     /**
51043      * Returns true if the dropdown list is expanded, else false.
51044      */
51045     isExpanded : function(){
51046         return false;
51047     },
51048
51049     /**
51050      * Select an item in the dropdown list by its data value. This function does NOT cause the select event to fire.
51051      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
51052      * @param {String} value The data value of the item to select
51053      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
51054      * selected item if it is not currently in view (defaults to true)
51055      * @return {Boolean} True if the value matched an item in the list, else false
51056      */
51057     selectByValue : function(v, scrollIntoView){
51058         Roo.log('select By Value');
51059         return false;
51060     
51061         if(v !== undefined && v !== null){
51062             var r = this.findRecord(this.valueField || this.displayField, v);
51063             if(r){
51064                 this.select(this.store.indexOf(r), scrollIntoView);
51065                 return true;
51066             }
51067         }
51068         return false;
51069     },
51070
51071     /**
51072      * Select an item in the dropdown list by its numeric index in the list. This function does NOT cause the select event to fire.
51073      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
51074      * @param {Number} index The zero-based index of the list item to select
51075      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
51076      * selected item if it is not currently in view (defaults to true)
51077      */
51078     select : function(index, scrollIntoView){
51079         Roo.log('select ');
51080         return  ;
51081         
51082         this.selectedIndex = index;
51083         this.view.select(index);
51084         if(scrollIntoView !== false){
51085             var el = this.view.getNode(index);
51086             if(el){
51087                 this.innerList.scrollChildIntoView(el, false);
51088             }
51089         }
51090     },
51091
51092       
51093
51094     // private
51095     validateBlur : function(){
51096         
51097         return;
51098         
51099     },
51100
51101     // private
51102     initQuery : function(){
51103         this.doQuery(this.getRawValue());
51104     },
51105
51106     // private
51107     doForce : function(){
51108         if(this.el.dom.value.length > 0){
51109             this.el.dom.value =
51110                 this.lastSelectionText === undefined ? '' : this.lastSelectionText;
51111              
51112         }
51113     },
51114
51115     /**
51116      * Execute a query to filter the dropdown list.  Fires the beforequery event prior to performing the
51117      * query allowing the query action to be canceled if needed.
51118      * @param {String} query The SQL query to execute
51119      * @param {Boolean} forceAll True to force the query to execute even if there are currently fewer characters
51120      * in the field than the minimum specified by the minChars config option.  It also clears any filter previously
51121      * saved in the current store (defaults to false)
51122      */
51123     doQuery : function(q, forceAll){
51124         
51125         Roo.log('doQuery?');
51126         if(q === undefined || q === null){
51127             q = '';
51128         }
51129         var qe = {
51130             query: q,
51131             forceAll: forceAll,
51132             combo: this,
51133             cancel:false
51134         };
51135         if(this.fireEvent('beforequery', qe)===false || qe.cancel){
51136             return false;
51137         }
51138         q = qe.query;
51139         forceAll = qe.forceAll;
51140         if(forceAll === true || (q.length >= this.minChars)){
51141             if(this.lastQuery != q || this.alwaysQuery){
51142                 this.lastQuery = q;
51143                 if(this.mode == 'local'){
51144                     this.selectedIndex = -1;
51145                     if(forceAll){
51146                         this.store.clearFilter();
51147                     }else{
51148                         this.store.filter(this.displayField, q);
51149                     }
51150                     this.onLoad();
51151                 }else{
51152                     this.store.baseParams[this.queryParam] = q;
51153                     this.store.load({
51154                         params: this.getParams(q)
51155                     });
51156                     this.expand();
51157                 }
51158             }else{
51159                 this.selectedIndex = -1;
51160                 this.onLoad();   
51161             }
51162         }
51163     },
51164
51165     // private
51166     getParams : function(q){
51167         var p = {};
51168         //p[this.queryParam] = q;
51169         if(this.pageSize){
51170             p.start = 0;
51171             p.limit = this.pageSize;
51172         }
51173         return p;
51174     },
51175
51176     /**
51177      * Hides the dropdown list if it is currently expanded. Fires the 'collapse' event on completion.
51178      */
51179     collapse : function(){
51180         
51181     },
51182
51183     // private
51184     collapseIf : function(e){
51185         
51186     },
51187
51188     /**
51189      * Expands the dropdown list if it is currently hidden. Fires the 'expand' event on completion.
51190      */
51191     expand : function(){
51192         
51193     } ,
51194
51195     // private
51196      
51197
51198     /** 
51199     * @cfg {Boolean} grow 
51200     * @hide 
51201     */
51202     /** 
51203     * @cfg {Number} growMin 
51204     * @hide 
51205     */
51206     /** 
51207     * @cfg {Number} growMax 
51208     * @hide 
51209     */
51210     /**
51211      * @hide
51212      * @method autoSize
51213      */
51214     
51215     setWidth : function()
51216     {
51217         
51218     },
51219     getResizeEl : function(){
51220         return this.el;
51221     }
51222 });//<script type="text/javasscript">
51223  
51224
51225 /**
51226  * @class Roo.DDView
51227  * A DnD enabled version of Roo.View.
51228  * @param {Element/String} container The Element in which to create the View.
51229  * @param {String} tpl The template string used to create the markup for each element of the View
51230  * @param {Object} config The configuration properties. These include all the config options of
51231  * {@link Roo.View} plus some specific to this class.<br>
51232  * <p>
51233  * Drag/drop is implemented by adding {@link Roo.data.Record}s to the target DDView. If copying is
51234  * not being performed, the original {@link Roo.data.Record} is removed from the source DDView.<br>
51235  * <p>
51236  * The following extra CSS rules are needed to provide insertion point highlighting:<pre><code>
51237 .x-view-drag-insert-above {
51238         border-top:1px dotted #3366cc;
51239 }
51240 .x-view-drag-insert-below {
51241         border-bottom:1px dotted #3366cc;
51242 }
51243 </code></pre>
51244  * 
51245  */
51246  
51247 Roo.DDView = function(container, tpl, config) {
51248     Roo.DDView.superclass.constructor.apply(this, arguments);
51249     this.getEl().setStyle("outline", "0px none");
51250     this.getEl().unselectable();
51251     if (this.dragGroup) {
51252                 this.setDraggable(this.dragGroup.split(","));
51253     }
51254     if (this.dropGroup) {
51255                 this.setDroppable(this.dropGroup.split(","));
51256     }
51257     if (this.deletable) {
51258         this.setDeletable();
51259     }
51260     this.isDirtyFlag = false;
51261         this.addEvents({
51262                 "drop" : true
51263         });
51264 };
51265
51266 Roo.extend(Roo.DDView, Roo.View, {
51267 /**     @cfg {String/Array} dragGroup The ddgroup name(s) for the View's DragZone. */
51268 /**     @cfg {String/Array} dropGroup The ddgroup name(s) for the View's DropZone. */
51269 /**     @cfg {Boolean} copy Causes drag operations to copy nodes rather than move. */
51270 /**     @cfg {Boolean} allowCopy Causes ctrl/drag operations to copy nodes rather than move. */
51271
51272         isFormField: true,
51273
51274         reset: Roo.emptyFn,
51275         
51276         clearInvalid: Roo.form.Field.prototype.clearInvalid,
51277
51278         validate: function() {
51279                 return true;
51280         },
51281         
51282         destroy: function() {
51283                 this.purgeListeners();
51284                 this.getEl.removeAllListeners();
51285                 this.getEl().remove();
51286                 if (this.dragZone) {
51287                         if (this.dragZone.destroy) {
51288                                 this.dragZone.destroy();
51289                         }
51290                 }
51291                 if (this.dropZone) {
51292                         if (this.dropZone.destroy) {
51293                                 this.dropZone.destroy();
51294                         }
51295                 }
51296         },
51297
51298 /**     Allows this class to be an Roo.form.Field so it can be found using {@link Roo.form.BasicForm#findField}. */
51299         getName: function() {
51300                 return this.name;
51301         },
51302
51303 /**     Loads the View from a JSON string representing the Records to put into the Store. */
51304         setValue: function(v) {
51305                 if (!this.store) {
51306                         throw "DDView.setValue(). DDView must be constructed with a valid Store";
51307                 }
51308                 var data = {};
51309                 data[this.store.reader.meta.root] = v ? [].concat(v) : [];
51310                 this.store.proxy = new Roo.data.MemoryProxy(data);
51311                 this.store.load();
51312         },
51313
51314 /**     @return {String} a parenthesised list of the ids of the Records in the View. */
51315         getValue: function() {
51316                 var result = '(';
51317                 this.store.each(function(rec) {
51318                         result += rec.id + ',';
51319                 });
51320                 return result.substr(0, result.length - 1) + ')';
51321         },
51322         
51323         getIds: function() {
51324                 var i = 0, result = new Array(this.store.getCount());
51325                 this.store.each(function(rec) {
51326                         result[i++] = rec.id;
51327                 });
51328                 return result;
51329         },
51330         
51331         isDirty: function() {
51332                 return this.isDirtyFlag;
51333         },
51334
51335 /**
51336  *      Part of the Roo.dd.DropZone interface. If no target node is found, the
51337  *      whole Element becomes the target, and this causes the drop gesture to append.
51338  */
51339     getTargetFromEvent : function(e) {
51340                 var target = e.getTarget();
51341                 while ((target !== null) && (target.parentNode != this.el.dom)) {
51342                 target = target.parentNode;
51343                 }
51344                 if (!target) {
51345                         target = this.el.dom.lastChild || this.el.dom;
51346                 }
51347                 return target;
51348     },
51349
51350 /**
51351  *      Create the drag data which consists of an object which has the property "ddel" as
51352  *      the drag proxy element. 
51353  */
51354     getDragData : function(e) {
51355         var target = this.findItemFromChild(e.getTarget());
51356                 if(target) {
51357                         this.handleSelection(e);
51358                         var selNodes = this.getSelectedNodes();
51359             var dragData = {
51360                 source: this,
51361                 copy: this.copy || (this.allowCopy && e.ctrlKey),
51362                 nodes: selNodes,
51363                 records: []
51364                         };
51365                         var selectedIndices = this.getSelectedIndexes();
51366                         for (var i = 0; i < selectedIndices.length; i++) {
51367                                 dragData.records.push(this.store.getAt(selectedIndices[i]));
51368                         }
51369                         if (selNodes.length == 1) {
51370                                 dragData.ddel = target.cloneNode(true); // the div element
51371                         } else {
51372                                 var div = document.createElement('div'); // create the multi element drag "ghost"
51373                                 div.className = 'multi-proxy';
51374                                 for (var i = 0, len = selNodes.length; i < len; i++) {
51375                                         div.appendChild(selNodes[i].cloneNode(true));
51376                                 }
51377                                 dragData.ddel = div;
51378                         }
51379             //console.log(dragData)
51380             //console.log(dragData.ddel.innerHTML)
51381                         return dragData;
51382                 }
51383         //console.log('nodragData')
51384                 return false;
51385     },
51386     
51387 /**     Specify to which ddGroup items in this DDView may be dragged. */
51388     setDraggable: function(ddGroup) {
51389         if (ddGroup instanceof Array) {
51390                 Roo.each(ddGroup, this.setDraggable, this);
51391                 return;
51392         }
51393         if (this.dragZone) {
51394                 this.dragZone.addToGroup(ddGroup);
51395         } else {
51396                         this.dragZone = new Roo.dd.DragZone(this.getEl(), {
51397                                 containerScroll: true,
51398                                 ddGroup: ddGroup 
51399
51400                         });
51401 //                      Draggability implies selection. DragZone's mousedown selects the element.
51402                         if (!this.multiSelect) { this.singleSelect = true; }
51403
51404 //                      Wire the DragZone's handlers up to methods in *this*
51405                         this.dragZone.getDragData = this.getDragData.createDelegate(this);
51406                 }
51407     },
51408
51409 /**     Specify from which ddGroup this DDView accepts drops. */
51410     setDroppable: function(ddGroup) {
51411         if (ddGroup instanceof Array) {
51412                 Roo.each(ddGroup, this.setDroppable, this);
51413                 return;
51414         }
51415         if (this.dropZone) {
51416                 this.dropZone.addToGroup(ddGroup);
51417         } else {
51418                         this.dropZone = new Roo.dd.DropZone(this.getEl(), {
51419                                 containerScroll: true,
51420                                 ddGroup: ddGroup
51421                         });
51422
51423 //                      Wire the DropZone's handlers up to methods in *this*
51424                         this.dropZone.getTargetFromEvent = this.getTargetFromEvent.createDelegate(this);
51425                         this.dropZone.onNodeEnter = this.onNodeEnter.createDelegate(this);
51426                         this.dropZone.onNodeOver = this.onNodeOver.createDelegate(this);
51427                         this.dropZone.onNodeOut = this.onNodeOut.createDelegate(this);
51428                         this.dropZone.onNodeDrop = this.onNodeDrop.createDelegate(this);
51429                 }
51430     },
51431
51432 /**     Decide whether to drop above or below a View node. */
51433     getDropPoint : function(e, n, dd){
51434         if (n == this.el.dom) { return "above"; }
51435                 var t = Roo.lib.Dom.getY(n), b = t + n.offsetHeight;
51436                 var c = t + (b - t) / 2;
51437                 var y = Roo.lib.Event.getPageY(e);
51438                 if(y <= c) {
51439                         return "above";
51440                 }else{
51441                         return "below";
51442                 }
51443     },
51444
51445     onNodeEnter : function(n, dd, e, data){
51446                 return false;
51447     },
51448     
51449     onNodeOver : function(n, dd, e, data){
51450                 var pt = this.getDropPoint(e, n, dd);
51451                 // set the insert point style on the target node
51452                 var dragElClass = this.dropNotAllowed;
51453                 if (pt) {
51454                         var targetElClass;
51455                         if (pt == "above"){
51456                                 dragElClass = n.previousSibling ? "x-tree-drop-ok-between" : "x-tree-drop-ok-above";
51457                                 targetElClass = "x-view-drag-insert-above";
51458                         } else {
51459                                 dragElClass = n.nextSibling ? "x-tree-drop-ok-between" : "x-tree-drop-ok-below";
51460                                 targetElClass = "x-view-drag-insert-below";
51461                         }
51462                         if (this.lastInsertClass != targetElClass){
51463                                 Roo.fly(n).replaceClass(this.lastInsertClass, targetElClass);
51464                                 this.lastInsertClass = targetElClass;
51465                         }
51466                 }
51467                 return dragElClass;
51468         },
51469
51470     onNodeOut : function(n, dd, e, data){
51471                 this.removeDropIndicators(n);
51472     },
51473
51474     onNodeDrop : function(n, dd, e, data){
51475         if (this.fireEvent("drop", this, n, dd, e, data) === false) {
51476                 return false;
51477         }
51478         var pt = this.getDropPoint(e, n, dd);
51479                 var insertAt = (n == this.el.dom) ? this.nodes.length : n.nodeIndex;
51480                 if (pt == "below") { insertAt++; }
51481                 for (var i = 0; i < data.records.length; i++) {
51482                         var r = data.records[i];
51483                         var dup = this.store.getById(r.id);
51484                         if (dup && (dd != this.dragZone)) {
51485                                 Roo.fly(this.getNode(this.store.indexOf(dup))).frame("red", 1);
51486                         } else {
51487                                 if (data.copy) {
51488                                         this.store.insert(insertAt++, r.copy());
51489                                 } else {
51490                                         data.source.isDirtyFlag = true;
51491                                         r.store.remove(r);
51492                                         this.store.insert(insertAt++, r);
51493                                 }
51494                                 this.isDirtyFlag = true;
51495                         }
51496                 }
51497                 this.dragZone.cachedTarget = null;
51498                 return true;
51499     },
51500
51501     removeDropIndicators : function(n){
51502                 if(n){
51503                         Roo.fly(n).removeClass([
51504                                 "x-view-drag-insert-above",
51505                                 "x-view-drag-insert-below"]);
51506                         this.lastInsertClass = "_noclass";
51507                 }
51508     },
51509
51510 /**
51511  *      Utility method. Add a delete option to the DDView's context menu.
51512  *      @param {String} imageUrl The URL of the "delete" icon image.
51513  */
51514         setDeletable: function(imageUrl) {
51515                 if (!this.singleSelect && !this.multiSelect) {
51516                         this.singleSelect = true;
51517                 }
51518                 var c = this.getContextMenu();
51519                 this.contextMenu.on("itemclick", function(item) {
51520                         switch (item.id) {
51521                                 case "delete":
51522                                         this.remove(this.getSelectedIndexes());
51523                                         break;
51524                         }
51525                 }, this);
51526                 this.contextMenu.add({
51527                         icon: imageUrl,
51528                         id: "delete",
51529                         text: 'Delete'
51530                 });
51531         },
51532         
51533 /**     Return the context menu for this DDView. */
51534         getContextMenu: function() {
51535                 if (!this.contextMenu) {
51536 //                      Create the View's context menu
51537                         this.contextMenu = new Roo.menu.Menu({
51538                                 id: this.id + "-contextmenu"
51539                         });
51540                         this.el.on("contextmenu", this.showContextMenu, this);
51541                 }
51542                 return this.contextMenu;
51543         },
51544         
51545         disableContextMenu: function() {
51546                 if (this.contextMenu) {
51547                         this.el.un("contextmenu", this.showContextMenu, this);
51548                 }
51549         },
51550
51551         showContextMenu: function(e, item) {
51552         item = this.findItemFromChild(e.getTarget());
51553                 if (item) {
51554                         e.stopEvent();
51555                         this.select(this.getNode(item), this.multiSelect && e.ctrlKey, true);
51556                         this.contextMenu.showAt(e.getXY());
51557             }
51558     },
51559
51560 /**
51561  *      Remove {@link Roo.data.Record}s at the specified indices.
51562  *      @param {Array/Number} selectedIndices The index (or Array of indices) of Records to remove.
51563  */
51564     remove: function(selectedIndices) {
51565                 selectedIndices = [].concat(selectedIndices);
51566                 for (var i = 0; i < selectedIndices.length; i++) {
51567                         var rec = this.store.getAt(selectedIndices[i]);
51568                         this.store.remove(rec);
51569                 }
51570     },
51571
51572 /**
51573  *      Double click fires the event, but also, if this is draggable, and there is only one other
51574  *      related DropZone, it transfers the selected node.
51575  */
51576     onDblClick : function(e){
51577         var item = this.findItemFromChild(e.getTarget());
51578         if(item){
51579             if (this.fireEvent("dblclick", this, this.indexOf(item), item, e) === false) {
51580                 return false;
51581             }
51582             if (this.dragGroup) {
51583                     var targets = Roo.dd.DragDropMgr.getRelated(this.dragZone, true);
51584                     while (targets.indexOf(this.dropZone) > -1) {
51585                             targets.remove(this.dropZone);
51586                                 }
51587                     if (targets.length == 1) {
51588                                         this.dragZone.cachedTarget = null;
51589                         var el = Roo.get(targets[0].getEl());
51590                         var box = el.getBox(true);
51591                         targets[0].onNodeDrop(el.dom, {
51592                                 target: el.dom,
51593                                 xy: [box.x, box.y + box.height - 1]
51594                         }, null, this.getDragData(e));
51595                     }
51596                 }
51597         }
51598     },
51599     
51600     handleSelection: function(e) {
51601                 this.dragZone.cachedTarget = null;
51602         var item = this.findItemFromChild(e.getTarget());
51603         if (!item) {
51604                 this.clearSelections(true);
51605                 return;
51606         }
51607                 if (item && (this.multiSelect || this.singleSelect)){
51608                         if(this.multiSelect && e.shiftKey && (!e.ctrlKey) && this.lastSelection){
51609                                 this.select(this.getNodes(this.indexOf(this.lastSelection), item.nodeIndex), false);
51610                         }else if (this.isSelected(this.getNode(item)) && e.ctrlKey){
51611                                 this.unselect(item);
51612                         } else {
51613                                 this.select(item, this.multiSelect && e.ctrlKey);
51614                                 this.lastSelection = item;
51615                         }
51616                 }
51617     },
51618
51619     onItemClick : function(item, index, e){
51620                 if(this.fireEvent("beforeclick", this, index, item, e) === false){
51621                         return false;
51622                 }
51623                 return true;
51624     },
51625
51626     unselect : function(nodeInfo, suppressEvent){
51627                 var node = this.getNode(nodeInfo);
51628                 if(node && this.isSelected(node)){
51629                         if(this.fireEvent("beforeselect", this, node, this.selections) !== false){
51630                                 Roo.fly(node).removeClass(this.selectedClass);
51631                                 this.selections.remove(node);
51632                                 if(!suppressEvent){
51633                                         this.fireEvent("selectionchange", this, this.selections);
51634                                 }
51635                         }
51636                 }
51637     }
51638 });
51639 /*
51640  * Based on:
51641  * Ext JS Library 1.1.1
51642  * Copyright(c) 2006-2007, Ext JS, LLC.
51643  *
51644  * Originally Released Under LGPL - original licence link has changed is not relivant.
51645  *
51646  * Fork - LGPL
51647  * <script type="text/javascript">
51648  */
51649  
51650 /**
51651  * @class Roo.LayoutManager
51652  * @extends Roo.util.Observable
51653  * Base class for layout managers.
51654  */
51655 Roo.LayoutManager = function(container, config){
51656     Roo.LayoutManager.superclass.constructor.call(this);
51657     this.el = Roo.get(container);
51658     // ie scrollbar fix
51659     if(this.el.dom == document.body && Roo.isIE && !config.allowScroll){
51660         document.body.scroll = "no";
51661     }else if(this.el.dom != document.body && this.el.getStyle('position') == 'static'){
51662         this.el.position('relative');
51663     }
51664     this.id = this.el.id;
51665     this.el.addClass("x-layout-container");
51666     /** false to disable window resize monitoring @type Boolean */
51667     this.monitorWindowResize = true;
51668     this.regions = {};
51669     this.addEvents({
51670         /**
51671          * @event layout
51672          * Fires when a layout is performed. 
51673          * @param {Roo.LayoutManager} this
51674          */
51675         "layout" : true,
51676         /**
51677          * @event regionresized
51678          * Fires when the user resizes a region. 
51679          * @param {Roo.LayoutRegion} region The resized region
51680          * @param {Number} newSize The new size (width for east/west, height for north/south)
51681          */
51682         "regionresized" : true,
51683         /**
51684          * @event regioncollapsed
51685          * Fires when a region is collapsed. 
51686          * @param {Roo.LayoutRegion} region The collapsed region
51687          */
51688         "regioncollapsed" : true,
51689         /**
51690          * @event regionexpanded
51691          * Fires when a region is expanded.  
51692          * @param {Roo.LayoutRegion} region The expanded region
51693          */
51694         "regionexpanded" : true
51695     });
51696     this.updating = false;
51697     Roo.EventManager.onWindowResize(this.onWindowResize, this, true);
51698 };
51699
51700 Roo.extend(Roo.LayoutManager, Roo.util.Observable, {
51701     /**
51702      * Returns true if this layout is currently being updated
51703      * @return {Boolean}
51704      */
51705     isUpdating : function(){
51706         return this.updating; 
51707     },
51708     
51709     /**
51710      * Suspend the LayoutManager from doing auto-layouts while
51711      * making multiple add or remove calls
51712      */
51713     beginUpdate : function(){
51714         this.updating = true;    
51715     },
51716     
51717     /**
51718      * Restore auto-layouts and optionally disable the manager from performing a layout
51719      * @param {Boolean} noLayout true to disable a layout update 
51720      */
51721     endUpdate : function(noLayout){
51722         this.updating = false;
51723         if(!noLayout){
51724             this.layout();
51725         }    
51726     },
51727     
51728     layout: function(){
51729         
51730     },
51731     
51732     onRegionResized : function(region, newSize){
51733         this.fireEvent("regionresized", region, newSize);
51734         this.layout();
51735     },
51736     
51737     onRegionCollapsed : function(region){
51738         this.fireEvent("regioncollapsed", region);
51739     },
51740     
51741     onRegionExpanded : function(region){
51742         this.fireEvent("regionexpanded", region);
51743     },
51744         
51745     /**
51746      * Returns the size of the current view. This method normalizes document.body and element embedded layouts and
51747      * performs box-model adjustments.
51748      * @return {Object} The size as an object {width: (the width), height: (the height)}
51749      */
51750     getViewSize : function(){
51751         var size;
51752         if(this.el.dom != document.body){
51753             size = this.el.getSize();
51754         }else{
51755             size = {width: Roo.lib.Dom.getViewWidth(), height: Roo.lib.Dom.getViewHeight()};
51756         }
51757         size.width -= this.el.getBorderWidth("lr")-this.el.getPadding("lr");
51758         size.height -= this.el.getBorderWidth("tb")-this.el.getPadding("tb");
51759         return size;
51760     },
51761     
51762     /**
51763      * Returns the Element this layout is bound to.
51764      * @return {Roo.Element}
51765      */
51766     getEl : function(){
51767         return this.el;
51768     },
51769     
51770     /**
51771      * Returns the specified region.
51772      * @param {String} target The region key ('center', 'north', 'south', 'east' or 'west')
51773      * @return {Roo.LayoutRegion}
51774      */
51775     getRegion : function(target){
51776         return this.regions[target.toLowerCase()];
51777     },
51778     
51779     onWindowResize : function(){
51780         if(this.monitorWindowResize){
51781             this.layout();
51782         }
51783     }
51784 });/*
51785  * Based on:
51786  * Ext JS Library 1.1.1
51787  * Copyright(c) 2006-2007, Ext JS, LLC.
51788  *
51789  * Originally Released Under LGPL - original licence link has changed is not relivant.
51790  *
51791  * Fork - LGPL
51792  * <script type="text/javascript">
51793  */
51794 /**
51795  * @class Roo.BorderLayout
51796  * @extends Roo.LayoutManager
51797  * This class represents a common layout manager used in desktop applications. For screenshots and more details,
51798  * please see: <br><br>
51799  * <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>
51800  * <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>
51801  * Example:
51802  <pre><code>
51803  var layout = new Roo.BorderLayout(document.body, {
51804     north: {
51805         initialSize: 25,
51806         titlebar: false
51807     },
51808     west: {
51809         split:true,
51810         initialSize: 200,
51811         minSize: 175,
51812         maxSize: 400,
51813         titlebar: true,
51814         collapsible: true
51815     },
51816     east: {
51817         split:true,
51818         initialSize: 202,
51819         minSize: 175,
51820         maxSize: 400,
51821         titlebar: true,
51822         collapsible: true
51823     },
51824     south: {
51825         split:true,
51826         initialSize: 100,
51827         minSize: 100,
51828         maxSize: 200,
51829         titlebar: true,
51830         collapsible: true
51831     },
51832     center: {
51833         titlebar: true,
51834         autoScroll:true,
51835         resizeTabs: true,
51836         minTabWidth: 50,
51837         preferredTabWidth: 150
51838     }
51839 });
51840
51841 // shorthand
51842 var CP = Roo.ContentPanel;
51843
51844 layout.beginUpdate();
51845 layout.add("north", new CP("north", "North"));
51846 layout.add("south", new CP("south", {title: "South", closable: true}));
51847 layout.add("west", new CP("west", {title: "West"}));
51848 layout.add("east", new CP("autoTabs", {title: "Auto Tabs", closable: true}));
51849 layout.add("center", new CP("center1", {title: "Close Me", closable: true}));
51850 layout.add("center", new CP("center2", {title: "Center Panel", closable: false}));
51851 layout.getRegion("center").showPanel("center1");
51852 layout.endUpdate();
51853 </code></pre>
51854
51855 <b>The container the layout is rendered into can be either the body element or any other element.
51856 If it is not the body element, the container needs to either be an absolute positioned element,
51857 or you will need to add "position:relative" to the css of the container.  You will also need to specify
51858 the container size if it is not the body element.</b>
51859
51860 * @constructor
51861 * Create a new BorderLayout
51862 * @param {String/HTMLElement/Element} container The container this layout is bound to
51863 * @param {Object} config Configuration options
51864  */
51865 Roo.BorderLayout = function(container, config){
51866     config = config || {};
51867     Roo.BorderLayout.superclass.constructor.call(this, container, config);
51868     this.factory = config.factory || Roo.BorderLayout.RegionFactory;
51869     for(var i = 0, len = this.factory.validRegions.length; i < len; i++) {
51870         var target = this.factory.validRegions[i];
51871         if(config[target]){
51872             this.addRegion(target, config[target]);
51873         }
51874     }
51875 };
51876
51877 Roo.extend(Roo.BorderLayout, Roo.LayoutManager, {
51878     /**
51879      * Creates and adds a new region if it doesn't already exist.
51880      * @param {String} target The target region key (north, south, east, west or center).
51881      * @param {Object} config The regions config object
51882      * @return {BorderLayoutRegion} The new region
51883      */
51884     addRegion : function(target, config){
51885         if(!this.regions[target]){
51886             var r = this.factory.create(target, this, config);
51887             this.bindRegion(target, r);
51888         }
51889         return this.regions[target];
51890     },
51891
51892     // private (kinda)
51893     bindRegion : function(name, r){
51894         this.regions[name] = r;
51895         r.on("visibilitychange", this.layout, this);
51896         r.on("paneladded", this.layout, this);
51897         r.on("panelremoved", this.layout, this);
51898         r.on("invalidated", this.layout, this);
51899         r.on("resized", this.onRegionResized, this);
51900         r.on("collapsed", this.onRegionCollapsed, this);
51901         r.on("expanded", this.onRegionExpanded, this);
51902     },
51903
51904     /**
51905      * Performs a layout update.
51906      */
51907     layout : function(){
51908         if(this.updating) {
51909             return;
51910         }
51911         var size = this.getViewSize();
51912         var w = size.width;
51913         var h = size.height;
51914         var centerW = w;
51915         var centerH = h;
51916         var centerY = 0;
51917         var centerX = 0;
51918         //var x = 0, y = 0;
51919
51920         var rs = this.regions;
51921         var north = rs["north"];
51922         var south = rs["south"]; 
51923         var west = rs["west"];
51924         var east = rs["east"];
51925         var center = rs["center"];
51926         //if(this.hideOnLayout){ // not supported anymore
51927             //c.el.setStyle("display", "none");
51928         //}
51929         if(north && north.isVisible()){
51930             var b = north.getBox();
51931             var m = north.getMargins();
51932             b.width = w - (m.left+m.right);
51933             b.x = m.left;
51934             b.y = m.top;
51935             centerY = b.height + b.y + m.bottom;
51936             centerH -= centerY;
51937             north.updateBox(this.safeBox(b));
51938         }
51939         if(south && south.isVisible()){
51940             var b = south.getBox();
51941             var m = south.getMargins();
51942             b.width = w - (m.left+m.right);
51943             b.x = m.left;
51944             var totalHeight = (b.height + m.top + m.bottom);
51945             b.y = h - totalHeight + m.top;
51946             centerH -= totalHeight;
51947             south.updateBox(this.safeBox(b));
51948         }
51949         if(west && west.isVisible()){
51950             var b = west.getBox();
51951             var m = west.getMargins();
51952             b.height = centerH - (m.top+m.bottom);
51953             b.x = m.left;
51954             b.y = centerY + m.top;
51955             var totalWidth = (b.width + m.left + m.right);
51956             centerX += totalWidth;
51957             centerW -= totalWidth;
51958             west.updateBox(this.safeBox(b));
51959         }
51960         if(east && east.isVisible()){
51961             var b = east.getBox();
51962             var m = east.getMargins();
51963             b.height = centerH - (m.top+m.bottom);
51964             var totalWidth = (b.width + m.left + m.right);
51965             b.x = w - totalWidth + m.left;
51966             b.y = centerY + m.top;
51967             centerW -= totalWidth;
51968             east.updateBox(this.safeBox(b));
51969         }
51970         if(center){
51971             var m = center.getMargins();
51972             var centerBox = {
51973                 x: centerX + m.left,
51974                 y: centerY + m.top,
51975                 width: centerW - (m.left+m.right),
51976                 height: centerH - (m.top+m.bottom)
51977             };
51978             //if(this.hideOnLayout){
51979                 //center.el.setStyle("display", "block");
51980             //}
51981             center.updateBox(this.safeBox(centerBox));
51982         }
51983         this.el.repaint();
51984         this.fireEvent("layout", this);
51985     },
51986
51987     // private
51988     safeBox : function(box){
51989         box.width = Math.max(0, box.width);
51990         box.height = Math.max(0, box.height);
51991         return box;
51992     },
51993
51994     /**
51995      * Adds a ContentPanel (or subclass) to this layout.
51996      * @param {String} target The target region key (north, south, east, west or center).
51997      * @param {Roo.ContentPanel} panel The panel to add
51998      * @return {Roo.ContentPanel} The added panel
51999      */
52000     add : function(target, panel){
52001          
52002         target = target.toLowerCase();
52003         return this.regions[target].add(panel);
52004     },
52005
52006     /**
52007      * Remove a ContentPanel (or subclass) to this layout.
52008      * @param {String} target The target region key (north, south, east, west or center).
52009      * @param {Number/String/Roo.ContentPanel} panel The index, id or panel to remove
52010      * @return {Roo.ContentPanel} The removed panel
52011      */
52012     remove : function(target, panel){
52013         target = target.toLowerCase();
52014         return this.regions[target].remove(panel);
52015     },
52016
52017     /**
52018      * Searches all regions for a panel with the specified id
52019      * @param {String} panelId
52020      * @return {Roo.ContentPanel} The panel or null if it wasn't found
52021      */
52022     findPanel : function(panelId){
52023         var rs = this.regions;
52024         for(var target in rs){
52025             if(typeof rs[target] != "function"){
52026                 var p = rs[target].getPanel(panelId);
52027                 if(p){
52028                     return p;
52029                 }
52030             }
52031         }
52032         return null;
52033     },
52034
52035     /**
52036      * Searches all regions for a panel with the specified id and activates (shows) it.
52037      * @param {String/ContentPanel} panelId The panels id or the panel itself
52038      * @return {Roo.ContentPanel} The shown panel or null
52039      */
52040     showPanel : function(panelId) {
52041       var rs = this.regions;
52042       for(var target in rs){
52043          var r = rs[target];
52044          if(typeof r != "function"){
52045             if(r.hasPanel(panelId)){
52046                return r.showPanel(panelId);
52047             }
52048          }
52049       }
52050       return null;
52051    },
52052
52053    /**
52054      * Restores this layout's state using Roo.state.Manager or the state provided by the passed provider.
52055      * @param {Roo.state.Provider} provider (optional) An alternate state provider
52056      */
52057     restoreState : function(provider){
52058         if(!provider){
52059             provider = Roo.state.Manager;
52060         }
52061         var sm = new Roo.LayoutStateManager();
52062         sm.init(this, provider);
52063     },
52064
52065     /**
52066      * Adds a batch of multiple ContentPanels dynamically by passing a special regions config object.  This config
52067      * object should contain properties for each region to add ContentPanels to, and each property's value should be
52068      * a valid ContentPanel config object.  Example:
52069      * <pre><code>
52070 // Create the main layout
52071 var layout = new Roo.BorderLayout('main-ct', {
52072     west: {
52073         split:true,
52074         minSize: 175,
52075         titlebar: true
52076     },
52077     center: {
52078         title:'Components'
52079     }
52080 }, 'main-ct');
52081
52082 // Create and add multiple ContentPanels at once via configs
52083 layout.batchAdd({
52084    west: {
52085        id: 'source-files',
52086        autoCreate:true,
52087        title:'Ext Source Files',
52088        autoScroll:true,
52089        fitToFrame:true
52090    },
52091    center : {
52092        el: cview,
52093        autoScroll:true,
52094        fitToFrame:true,
52095        toolbar: tb,
52096        resizeEl:'cbody'
52097    }
52098 });
52099 </code></pre>
52100      * @param {Object} regions An object containing ContentPanel configs by region name
52101      */
52102     batchAdd : function(regions){
52103         this.beginUpdate();
52104         for(var rname in regions){
52105             var lr = this.regions[rname];
52106             if(lr){
52107                 this.addTypedPanels(lr, regions[rname]);
52108             }
52109         }
52110         this.endUpdate();
52111     },
52112
52113     // private
52114     addTypedPanels : function(lr, ps){
52115         if(typeof ps == 'string'){
52116             lr.add(new Roo.ContentPanel(ps));
52117         }
52118         else if(ps instanceof Array){
52119             for(var i =0, len = ps.length; i < len; i++){
52120                 this.addTypedPanels(lr, ps[i]);
52121             }
52122         }
52123         else if(!ps.events){ // raw config?
52124             var el = ps.el;
52125             delete ps.el; // prevent conflict
52126             lr.add(new Roo.ContentPanel(el || Roo.id(), ps));
52127         }
52128         else {  // panel object assumed!
52129             lr.add(ps);
52130         }
52131     },
52132     /**
52133      * Adds a xtype elements to the layout.
52134      * <pre><code>
52135
52136 layout.addxtype({
52137        xtype : 'ContentPanel',
52138        region: 'west',
52139        items: [ .... ]
52140    }
52141 );
52142
52143 layout.addxtype({
52144         xtype : 'NestedLayoutPanel',
52145         region: 'west',
52146         layout: {
52147            center: { },
52148            west: { }   
52149         },
52150         items : [ ... list of content panels or nested layout panels.. ]
52151    }
52152 );
52153 </code></pre>
52154      * @param {Object} cfg Xtype definition of item to add.
52155      */
52156     addxtype : function(cfg)
52157     {
52158         // basically accepts a pannel...
52159         // can accept a layout region..!?!?
52160         //Roo.log('Roo.BorderLayout add ' + cfg.xtype)
52161         
52162         if (!cfg.xtype.match(/Panel$/)) {
52163             return false;
52164         }
52165         var ret = false;
52166         
52167         if (typeof(cfg.region) == 'undefined') {
52168             Roo.log("Failed to add Panel, region was not set");
52169             Roo.log(cfg);
52170             return false;
52171         }
52172         var region = cfg.region;
52173         delete cfg.region;
52174         
52175           
52176         var xitems = [];
52177         if (cfg.items) {
52178             xitems = cfg.items;
52179             delete cfg.items;
52180         }
52181         var nb = false;
52182         
52183         switch(cfg.xtype) 
52184         {
52185             case 'ContentPanel':  // ContentPanel (el, cfg)
52186             case 'ScrollPanel':  // ContentPanel (el, cfg)
52187             case 'ViewPanel': 
52188                 if(cfg.autoCreate) {
52189                     ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52190                 } else {
52191                     var el = this.el.createChild();
52192                     ret = new Roo[cfg.xtype](el, cfg); // new panel!!!!!
52193                 }
52194                 
52195                 this.add(region, ret);
52196                 break;
52197             
52198             
52199             case 'TreePanel': // our new panel!
52200                 cfg.el = this.el.createChild();
52201                 ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52202                 this.add(region, ret);
52203                 break;
52204             
52205             case 'NestedLayoutPanel': 
52206                 // create a new Layout (which is  a Border Layout...
52207                 var el = this.el.createChild();
52208                 var clayout = cfg.layout;
52209                 delete cfg.layout;
52210                 clayout.items   = clayout.items  || [];
52211                 // replace this exitems with the clayout ones..
52212                 xitems = clayout.items;
52213                  
52214                 
52215                 if (region == 'center' && this.active && this.getRegion('center').panels.length < 1) {
52216                     cfg.background = false;
52217                 }
52218                 var layout = new Roo.BorderLayout(el, clayout);
52219                 
52220                 ret = new Roo[cfg.xtype](layout, cfg); // new panel!!!!!
52221                 //console.log('adding nested layout panel '  + cfg.toSource());
52222                 this.add(region, ret);
52223                 nb = {}; /// find first...
52224                 break;
52225                 
52226             case 'GridPanel': 
52227             
52228                 // needs grid and region
52229                 
52230                 //var el = this.getRegion(region).el.createChild();
52231                 var el = this.el.createChild();
52232                 // create the grid first...
52233                 
52234                 var grid = new Roo.grid[cfg.grid.xtype](el, cfg.grid);
52235                 delete cfg.grid;
52236                 if (region == 'center' && this.active ) {
52237                     cfg.background = false;
52238                 }
52239                 ret = new Roo[cfg.xtype](grid, cfg); // new panel!!!!!
52240                 
52241                 this.add(region, ret);
52242                 if (cfg.background) {
52243                     ret.on('activate', function(gp) {
52244                         if (!gp.grid.rendered) {
52245                             gp.grid.render();
52246                         }
52247                     });
52248                 } else {
52249                     grid.render();
52250                 }
52251                 break;
52252            
52253            
52254            
52255                 
52256                 
52257                 
52258             default:
52259                 if (typeof(Roo[cfg.xtype]) != 'undefined') {
52260                     
52261                     ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52262                     this.add(region, ret);
52263                 } else {
52264                 
52265                     alert("Can not add '" + cfg.xtype + "' to BorderLayout");
52266                     return null;
52267                 }
52268                 
52269              // GridPanel (grid, cfg)
52270             
52271         }
52272         this.beginUpdate();
52273         // add children..
52274         var region = '';
52275         var abn = {};
52276         Roo.each(xitems, function(i)  {
52277             region = nb && i.region ? i.region : false;
52278             
52279             var add = ret.addxtype(i);
52280            
52281             if (region) {
52282                 nb[region] = nb[region] == undefined ? 0 : nb[region]+1;
52283                 if (!i.background) {
52284                     abn[region] = nb[region] ;
52285                 }
52286             }
52287             
52288         });
52289         this.endUpdate();
52290
52291         // make the last non-background panel active..
52292         //if (nb) { Roo.log(abn); }
52293         if (nb) {
52294             
52295             for(var r in abn) {
52296                 region = this.getRegion(r);
52297                 if (region) {
52298                     // tried using nb[r], but it does not work..
52299                      
52300                     region.showPanel(abn[r]);
52301                    
52302                 }
52303             }
52304         }
52305         return ret;
52306         
52307     }
52308 });
52309
52310 /**
52311  * Shortcut for creating a new BorderLayout object and adding one or more ContentPanels to it in a single step, handling
52312  * the beginUpdate and endUpdate calls internally.  The key to this method is the <b>panels</b> property that can be
52313  * provided with each region config, which allows you to add ContentPanel configs in addition to the region configs
52314  * during creation.  The following code is equivalent to the constructor-based example at the beginning of this class:
52315  * <pre><code>
52316 // shorthand
52317 var CP = Roo.ContentPanel;
52318
52319 var layout = Roo.BorderLayout.create({
52320     north: {
52321         initialSize: 25,
52322         titlebar: false,
52323         panels: [new CP("north", "North")]
52324     },
52325     west: {
52326         split:true,
52327         initialSize: 200,
52328         minSize: 175,
52329         maxSize: 400,
52330         titlebar: true,
52331         collapsible: true,
52332         panels: [new CP("west", {title: "West"})]
52333     },
52334     east: {
52335         split:true,
52336         initialSize: 202,
52337         minSize: 175,
52338         maxSize: 400,
52339         titlebar: true,
52340         collapsible: true,
52341         panels: [new CP("autoTabs", {title: "Auto Tabs", closable: true})]
52342     },
52343     south: {
52344         split:true,
52345         initialSize: 100,
52346         minSize: 100,
52347         maxSize: 200,
52348         titlebar: true,
52349         collapsible: true,
52350         panels: [new CP("south", {title: "South", closable: true})]
52351     },
52352     center: {
52353         titlebar: true,
52354         autoScroll:true,
52355         resizeTabs: true,
52356         minTabWidth: 50,
52357         preferredTabWidth: 150,
52358         panels: [
52359             new CP("center1", {title: "Close Me", closable: true}),
52360             new CP("center2", {title: "Center Panel", closable: false})
52361         ]
52362     }
52363 }, document.body);
52364
52365 layout.getRegion("center").showPanel("center1");
52366 </code></pre>
52367  * @param config
52368  * @param targetEl
52369  */
52370 Roo.BorderLayout.create = function(config, targetEl){
52371     var layout = new Roo.BorderLayout(targetEl || document.body, config);
52372     layout.beginUpdate();
52373     var regions = Roo.BorderLayout.RegionFactory.validRegions;
52374     for(var j = 0, jlen = regions.length; j < jlen; j++){
52375         var lr = regions[j];
52376         if(layout.regions[lr] && config[lr].panels){
52377             var r = layout.regions[lr];
52378             var ps = config[lr].panels;
52379             layout.addTypedPanels(r, ps);
52380         }
52381     }
52382     layout.endUpdate();
52383     return layout;
52384 };
52385
52386 // private
52387 Roo.BorderLayout.RegionFactory = {
52388     // private
52389     validRegions : ["north","south","east","west","center"],
52390
52391     // private
52392     create : function(target, mgr, config){
52393         target = target.toLowerCase();
52394         if(config.lightweight || config.basic){
52395             return new Roo.BasicLayoutRegion(mgr, config, target);
52396         }
52397         switch(target){
52398             case "north":
52399                 return new Roo.NorthLayoutRegion(mgr, config);
52400             case "south":
52401                 return new Roo.SouthLayoutRegion(mgr, config);
52402             case "east":
52403                 return new Roo.EastLayoutRegion(mgr, config);
52404             case "west":
52405                 return new Roo.WestLayoutRegion(mgr, config);
52406             case "center":
52407                 return new Roo.CenterLayoutRegion(mgr, config);
52408         }
52409         throw 'Layout region "'+target+'" not supported.';
52410     }
52411 };/*
52412  * Based on:
52413  * Ext JS Library 1.1.1
52414  * Copyright(c) 2006-2007, Ext JS, LLC.
52415  *
52416  * Originally Released Under LGPL - original licence link has changed is not relivant.
52417  *
52418  * Fork - LGPL
52419  * <script type="text/javascript">
52420  */
52421  
52422 /**
52423  * @class Roo.BasicLayoutRegion
52424  * @extends Roo.util.Observable
52425  * This class represents a lightweight region in a layout manager. This region does not move dom nodes
52426  * and does not have a titlebar, tabs or any other features. All it does is size and position 
52427  * panels. To create a BasicLayoutRegion, add lightweight:true or basic:true to your regions config.
52428  */
52429 Roo.BasicLayoutRegion = function(mgr, config, pos, skipConfig){
52430     this.mgr = mgr;
52431     this.position  = pos;
52432     this.events = {
52433         /**
52434          * @scope Roo.BasicLayoutRegion
52435          */
52436         
52437         /**
52438          * @event beforeremove
52439          * Fires before a panel is removed (or closed). To cancel the removal set "e.cancel = true" on the event argument.
52440          * @param {Roo.LayoutRegion} this
52441          * @param {Roo.ContentPanel} panel The panel
52442          * @param {Object} e The cancel event object
52443          */
52444         "beforeremove" : true,
52445         /**
52446          * @event invalidated
52447          * Fires when the layout for this region is changed.
52448          * @param {Roo.LayoutRegion} this
52449          */
52450         "invalidated" : true,
52451         /**
52452          * @event visibilitychange
52453          * Fires when this region is shown or hidden 
52454          * @param {Roo.LayoutRegion} this
52455          * @param {Boolean} visibility true or false
52456          */
52457         "visibilitychange" : true,
52458         /**
52459          * @event paneladded
52460          * Fires when a panel is added. 
52461          * @param {Roo.LayoutRegion} this
52462          * @param {Roo.ContentPanel} panel The panel
52463          */
52464         "paneladded" : true,
52465         /**
52466          * @event panelremoved
52467          * Fires when a panel is removed. 
52468          * @param {Roo.LayoutRegion} this
52469          * @param {Roo.ContentPanel} panel The panel
52470          */
52471         "panelremoved" : true,
52472         /**
52473          * @event beforecollapse
52474          * Fires when this region before collapse.
52475          * @param {Roo.LayoutRegion} this
52476          */
52477         "beforecollapse" : true,
52478         /**
52479          * @event collapsed
52480          * Fires when this region is collapsed.
52481          * @param {Roo.LayoutRegion} this
52482          */
52483         "collapsed" : true,
52484         /**
52485          * @event expanded
52486          * Fires when this region is expanded.
52487          * @param {Roo.LayoutRegion} this
52488          */
52489         "expanded" : true,
52490         /**
52491          * @event slideshow
52492          * Fires when this region is slid into view.
52493          * @param {Roo.LayoutRegion} this
52494          */
52495         "slideshow" : true,
52496         /**
52497          * @event slidehide
52498          * Fires when this region slides out of view. 
52499          * @param {Roo.LayoutRegion} this
52500          */
52501         "slidehide" : true,
52502         /**
52503          * @event panelactivated
52504          * Fires when a panel is activated. 
52505          * @param {Roo.LayoutRegion} this
52506          * @param {Roo.ContentPanel} panel The activated panel
52507          */
52508         "panelactivated" : true,
52509         /**
52510          * @event resized
52511          * Fires when the user resizes this region. 
52512          * @param {Roo.LayoutRegion} this
52513          * @param {Number} newSize The new size (width for east/west, height for north/south)
52514          */
52515         "resized" : true
52516     };
52517     /** A collection of panels in this region. @type Roo.util.MixedCollection */
52518     this.panels = new Roo.util.MixedCollection();
52519     this.panels.getKey = this.getPanelId.createDelegate(this);
52520     this.box = null;
52521     this.activePanel = null;
52522     // ensure listeners are added...
52523     
52524     if (config.listeners || config.events) {
52525         Roo.BasicLayoutRegion.superclass.constructor.call(this, {
52526             listeners : config.listeners || {},
52527             events : config.events || {}
52528         });
52529     }
52530     
52531     if(skipConfig !== true){
52532         this.applyConfig(config);
52533     }
52534 };
52535
52536 Roo.extend(Roo.BasicLayoutRegion, Roo.util.Observable, {
52537     getPanelId : function(p){
52538         return p.getId();
52539     },
52540     
52541     applyConfig : function(config){
52542         this.margins = config.margins || this.margins || {top: 0, left: 0, right:0, bottom: 0};
52543         this.config = config;
52544         
52545     },
52546     
52547     /**
52548      * Resizes the region to the specified size. For vertical regions (west, east) this adjusts 
52549      * the width, for horizontal (north, south) the height.
52550      * @param {Number} newSize The new width or height
52551      */
52552     resizeTo : function(newSize){
52553         var el = this.el ? this.el :
52554                  (this.activePanel ? this.activePanel.getEl() : null);
52555         if(el){
52556             switch(this.position){
52557                 case "east":
52558                 case "west":
52559                     el.setWidth(newSize);
52560                     this.fireEvent("resized", this, newSize);
52561                 break;
52562                 case "north":
52563                 case "south":
52564                     el.setHeight(newSize);
52565                     this.fireEvent("resized", this, newSize);
52566                 break;                
52567             }
52568         }
52569     },
52570     
52571     getBox : function(){
52572         return this.activePanel ? this.activePanel.getEl().getBox(false, true) : null;
52573     },
52574     
52575     getMargins : function(){
52576         return this.margins;
52577     },
52578     
52579     updateBox : function(box){
52580         this.box = box;
52581         var el = this.activePanel.getEl();
52582         el.dom.style.left = box.x + "px";
52583         el.dom.style.top = box.y + "px";
52584         this.activePanel.setSize(box.width, box.height);
52585     },
52586     
52587     /**
52588      * Returns the container element for this region.
52589      * @return {Roo.Element}
52590      */
52591     getEl : function(){
52592         return this.activePanel;
52593     },
52594     
52595     /**
52596      * Returns true if this region is currently visible.
52597      * @return {Boolean}
52598      */
52599     isVisible : function(){
52600         return this.activePanel ? true : false;
52601     },
52602     
52603     setActivePanel : function(panel){
52604         panel = this.getPanel(panel);
52605         if(this.activePanel && this.activePanel != panel){
52606             this.activePanel.setActiveState(false);
52607             this.activePanel.getEl().setLeftTop(-10000,-10000);
52608         }
52609         this.activePanel = panel;
52610         panel.setActiveState(true);
52611         if(this.box){
52612             panel.setSize(this.box.width, this.box.height);
52613         }
52614         this.fireEvent("panelactivated", this, panel);
52615         this.fireEvent("invalidated");
52616     },
52617     
52618     /**
52619      * Show the specified panel.
52620      * @param {Number/String/ContentPanel} panelId The panels index, id or the panel itself
52621      * @return {Roo.ContentPanel} The shown panel or null
52622      */
52623     showPanel : function(panel){
52624         if(panel = this.getPanel(panel)){
52625             this.setActivePanel(panel);
52626         }
52627         return panel;
52628     },
52629     
52630     /**
52631      * Get the active panel for this region.
52632      * @return {Roo.ContentPanel} The active panel or null
52633      */
52634     getActivePanel : function(){
52635         return this.activePanel;
52636     },
52637     
52638     /**
52639      * Add the passed ContentPanel(s)
52640      * @param {ContentPanel...} panel The ContentPanel(s) to add (you can pass more than one)
52641      * @return {Roo.ContentPanel} The panel added (if only one was added)
52642      */
52643     add : function(panel){
52644         if(arguments.length > 1){
52645             for(var i = 0, len = arguments.length; i < len; i++) {
52646                 this.add(arguments[i]);
52647             }
52648             return null;
52649         }
52650         if(this.hasPanel(panel)){
52651             this.showPanel(panel);
52652             return panel;
52653         }
52654         var el = panel.getEl();
52655         if(el.dom.parentNode != this.mgr.el.dom){
52656             this.mgr.el.dom.appendChild(el.dom);
52657         }
52658         if(panel.setRegion){
52659             panel.setRegion(this);
52660         }
52661         this.panels.add(panel);
52662         el.setStyle("position", "absolute");
52663         if(!panel.background){
52664             this.setActivePanel(panel);
52665             if(this.config.initialSize && this.panels.getCount()==1){
52666                 this.resizeTo(this.config.initialSize);
52667             }
52668         }
52669         this.fireEvent("paneladded", this, panel);
52670         return panel;
52671     },
52672     
52673     /**
52674      * Returns true if the panel is in this region.
52675      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52676      * @return {Boolean}
52677      */
52678     hasPanel : function(panel){
52679         if(typeof panel == "object"){ // must be panel obj
52680             panel = panel.getId();
52681         }
52682         return this.getPanel(panel) ? true : false;
52683     },
52684     
52685     /**
52686      * Removes the specified panel. If preservePanel is not true (either here or in the config), the panel is destroyed.
52687      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52688      * @param {Boolean} preservePanel Overrides the config preservePanel option
52689      * @return {Roo.ContentPanel} The panel that was removed
52690      */
52691     remove : function(panel, preservePanel){
52692         panel = this.getPanel(panel);
52693         if(!panel){
52694             return null;
52695         }
52696         var e = {};
52697         this.fireEvent("beforeremove", this, panel, e);
52698         if(e.cancel === true){
52699             return null;
52700         }
52701         var panelId = panel.getId();
52702         this.panels.removeKey(panelId);
52703         return panel;
52704     },
52705     
52706     /**
52707      * Returns the panel specified or null if it's not in this region.
52708      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52709      * @return {Roo.ContentPanel}
52710      */
52711     getPanel : function(id){
52712         if(typeof id == "object"){ // must be panel obj
52713             return id;
52714         }
52715         return this.panels.get(id);
52716     },
52717     
52718     /**
52719      * Returns this regions position (north/south/east/west/center).
52720      * @return {String} 
52721      */
52722     getPosition: function(){
52723         return this.position;    
52724     }
52725 });/*
52726  * Based on:
52727  * Ext JS Library 1.1.1
52728  * Copyright(c) 2006-2007, Ext JS, LLC.
52729  *
52730  * Originally Released Under LGPL - original licence link has changed is not relivant.
52731  *
52732  * Fork - LGPL
52733  * <script type="text/javascript">
52734  */
52735  
52736 /**
52737  * @class Roo.LayoutRegion
52738  * @extends Roo.BasicLayoutRegion
52739  * This class represents a region in a layout manager.
52740  * @cfg {Boolean}   collapsible     False to disable collapsing (defaults to true)
52741  * @cfg {Boolean}   collapsed       True to set the initial display to collapsed (defaults to false)
52742  * @cfg {Boolean}   floatable       False to disable floating (defaults to true)
52743  * @cfg {Object}    margins         Margins for the element (defaults to {top: 0, left: 0, right:0, bottom: 0})
52744  * @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})
52745  * @cfg {String}    tabPosition     (top|bottom) "top" or "bottom" (defaults to "bottom")
52746  * @cfg {String}    collapsedTitle  Optional string message to display in the collapsed block of a north or south region
52747  * @cfg {Boolean}   alwaysShowTabs  True to always display tabs even when there is only 1 panel (defaults to false)
52748  * @cfg {Boolean}   autoScroll      True to enable overflow scrolling (defaults to false)
52749  * @cfg {Boolean}   titlebar        True to display a title bar (defaults to true)
52750  * @cfg {String}    title           The title for the region (overrides panel titles)
52751  * @cfg {Boolean}   animate         True to animate expand/collapse (defaults to false)
52752  * @cfg {Boolean}   autoHide        False to disable auto hiding when the mouse leaves the "floated" region (defaults to true)
52753  * @cfg {Boolean}   preservePanels  True to preserve removed panels so they can be readded later (defaults to false)
52754  * @cfg {Boolean}   closeOnTab      True to place the close icon on the tabs instead of the region titlebar (defaults to false)
52755  * @cfg {Boolean}   hideTabs        True to hide the tab strip (defaults to false)
52756  * @cfg {Boolean}   resizeTabs      True to enable automatic tab resizing. This will resize the tabs so they are all the same size and fit within
52757  *                      the space available, similar to FireFox 1.5 tabs (defaults to false)
52758  * @cfg {Number}    minTabWidth     The minimum tab width (defaults to 40)
52759  * @cfg {Number}    preferredTabWidth The preferred tab width (defaults to 150)
52760  * @cfg {Boolean}   showPin         True to show a pin button
52761  * @cfg {Boolean}   hidden          True to start the region hidden (defaults to false)
52762  * @cfg {Boolean}   hideWhenEmpty   True to hide the region when it has no panels
52763  * @cfg {Boolean}   disableTabTips  True to disable tab tooltips
52764  * @cfg {Number}    width           For East/West panels
52765  * @cfg {Number}    height          For North/South panels
52766  * @cfg {Boolean}   split           To show the splitter
52767  * @cfg {Boolean}   toolbar         xtype configuration for a toolbar - shows on right of tabbar
52768  */
52769 Roo.LayoutRegion = function(mgr, config, pos){
52770     Roo.LayoutRegion.superclass.constructor.call(this, mgr, config, pos, true);
52771     var dh = Roo.DomHelper;
52772     /** This region's container element 
52773     * @type Roo.Element */
52774     this.el = dh.append(mgr.el.dom, {tag: "div", cls: "x-layout-panel x-layout-panel-" + this.position}, true);
52775     /** This region's title element 
52776     * @type Roo.Element */
52777
52778     this.titleEl = dh.append(this.el.dom, {tag: "div", unselectable: "on", cls: "x-unselectable x-layout-panel-hd x-layout-title-"+this.position, children:[
52779         {tag: "span", cls: "x-unselectable x-layout-panel-hd-text", unselectable: "on", html: "&#160;"},
52780         {tag: "div", cls: "x-unselectable x-layout-panel-hd-tools", unselectable: "on"}
52781     ]}, true);
52782     this.titleEl.enableDisplayMode();
52783     /** This region's title text element 
52784     * @type HTMLElement */
52785     this.titleTextEl = this.titleEl.dom.firstChild;
52786     this.tools = Roo.get(this.titleEl.dom.childNodes[1], true);
52787     this.closeBtn = this.createTool(this.tools.dom, "x-layout-close");
52788     this.closeBtn.enableDisplayMode();
52789     this.closeBtn.on("click", this.closeClicked, this);
52790     this.closeBtn.hide();
52791
52792     this.createBody(config);
52793     this.visible = true;
52794     this.collapsed = false;
52795
52796     if(config.hideWhenEmpty){
52797         this.hide();
52798         this.on("paneladded", this.validateVisibility, this);
52799         this.on("panelremoved", this.validateVisibility, this);
52800     }
52801     this.applyConfig(config);
52802 };
52803
52804 Roo.extend(Roo.LayoutRegion, Roo.BasicLayoutRegion, {
52805
52806     createBody : function(){
52807         /** This region's body element 
52808         * @type Roo.Element */
52809         this.bodyEl = this.el.createChild({tag: "div", cls: "x-layout-panel-body"});
52810     },
52811
52812     applyConfig : function(c){
52813         if(c.collapsible && this.position != "center" && !this.collapsedEl){
52814             var dh = Roo.DomHelper;
52815             if(c.titlebar !== false){
52816                 this.collapseBtn = this.createTool(this.tools.dom, "x-layout-collapse-"+this.position);
52817                 this.collapseBtn.on("click", this.collapse, this);
52818                 this.collapseBtn.enableDisplayMode();
52819
52820                 if(c.showPin === true || this.showPin){
52821                     this.stickBtn = this.createTool(this.tools.dom, "x-layout-stick");
52822                     this.stickBtn.enableDisplayMode();
52823                     this.stickBtn.on("click", this.expand, this);
52824                     this.stickBtn.hide();
52825                 }
52826             }
52827             /** This region's collapsed element
52828             * @type Roo.Element */
52829             this.collapsedEl = dh.append(this.mgr.el.dom, {cls: "x-layout-collapsed x-layout-collapsed-"+this.position, children:[
52830                 {cls: "x-layout-collapsed-tools", children:[{cls: "x-layout-ctools-inner"}]}
52831             ]}, true);
52832             if(c.floatable !== false){
52833                this.collapsedEl.addClassOnOver("x-layout-collapsed-over");
52834                this.collapsedEl.on("click", this.collapseClick, this);
52835             }
52836
52837             if(c.collapsedTitle && (this.position == "north" || this.position== "south")) {
52838                 this.collapsedTitleTextEl = dh.append(this.collapsedEl.dom, {tag: "div", cls: "x-unselectable x-layout-panel-hd-text",
52839                    id: "message", unselectable: "on", style:{"float":"left"}});
52840                this.collapsedTitleTextEl.innerHTML = c.collapsedTitle;
52841              }
52842             this.expandBtn = this.createTool(this.collapsedEl.dom.firstChild.firstChild, "x-layout-expand-"+this.position);
52843             this.expandBtn.on("click", this.expand, this);
52844         }
52845         if(this.collapseBtn){
52846             this.collapseBtn.setVisible(c.collapsible == true);
52847         }
52848         this.cmargins = c.cmargins || this.cmargins ||
52849                          (this.position == "west" || this.position == "east" ?
52850                              {top: 0, left: 2, right:2, bottom: 0} :
52851                              {top: 2, left: 0, right:0, bottom: 2});
52852         this.margins = c.margins || this.margins || {top: 0, left: 0, right:0, bottom: 0};
52853         this.bottomTabs = c.tabPosition != "top";
52854         this.autoScroll = c.autoScroll || false;
52855         if(this.autoScroll){
52856             this.bodyEl.setStyle("overflow", "auto");
52857         }else{
52858             this.bodyEl.setStyle("overflow", "hidden");
52859         }
52860         //if(c.titlebar !== false){
52861             if((!c.titlebar && !c.title) || c.titlebar === false){
52862                 this.titleEl.hide();
52863             }else{
52864                 this.titleEl.show();
52865                 if(c.title){
52866                     this.titleTextEl.innerHTML = c.title;
52867                 }
52868             }
52869         //}
52870         this.duration = c.duration || .30;
52871         this.slideDuration = c.slideDuration || .45;
52872         this.config = c;
52873         if(c.collapsed){
52874             this.collapse(true);
52875         }
52876         if(c.hidden){
52877             this.hide();
52878         }
52879     },
52880     /**
52881      * Returns true if this region is currently visible.
52882      * @return {Boolean}
52883      */
52884     isVisible : function(){
52885         return this.visible;
52886     },
52887
52888     /**
52889      * Updates the title for collapsed north/south regions (used with {@link #collapsedTitle} config option)
52890      * @param {String} title (optional) The title text (accepts HTML markup, defaults to the numeric character reference for a non-breaking space, "&amp;#160;")
52891      */
52892     setCollapsedTitle : function(title){
52893         title = title || "&#160;";
52894         if(this.collapsedTitleTextEl){
52895             this.collapsedTitleTextEl.innerHTML = title;
52896         }
52897     },
52898
52899     getBox : function(){
52900         var b;
52901         if(!this.collapsed){
52902             b = this.el.getBox(false, true);
52903         }else{
52904             b = this.collapsedEl.getBox(false, true);
52905         }
52906         return b;
52907     },
52908
52909     getMargins : function(){
52910         return this.collapsed ? this.cmargins : this.margins;
52911     },
52912
52913     highlight : function(){
52914         this.el.addClass("x-layout-panel-dragover");
52915     },
52916
52917     unhighlight : function(){
52918         this.el.removeClass("x-layout-panel-dragover");
52919     },
52920
52921     updateBox : function(box){
52922         this.box = box;
52923         if(!this.collapsed){
52924             this.el.dom.style.left = box.x + "px";
52925             this.el.dom.style.top = box.y + "px";
52926             this.updateBody(box.width, box.height);
52927         }else{
52928             this.collapsedEl.dom.style.left = box.x + "px";
52929             this.collapsedEl.dom.style.top = box.y + "px";
52930             this.collapsedEl.setSize(box.width, box.height);
52931         }
52932         if(this.tabs){
52933             this.tabs.autoSizeTabs();
52934         }
52935     },
52936
52937     updateBody : function(w, h){
52938         if(w !== null){
52939             this.el.setWidth(w);
52940             w -= this.el.getBorderWidth("rl");
52941             if(this.config.adjustments){
52942                 w += this.config.adjustments[0];
52943             }
52944         }
52945         if(h !== null){
52946             this.el.setHeight(h);
52947             h = this.titleEl && this.titleEl.isDisplayed() ? h - (this.titleEl.getHeight()||0) : h;
52948             h -= this.el.getBorderWidth("tb");
52949             if(this.config.adjustments){
52950                 h += this.config.adjustments[1];
52951             }
52952             this.bodyEl.setHeight(h);
52953             if(this.tabs){
52954                 h = this.tabs.syncHeight(h);
52955             }
52956         }
52957         if(this.panelSize){
52958             w = w !== null ? w : this.panelSize.width;
52959             h = h !== null ? h : this.panelSize.height;
52960         }
52961         if(this.activePanel){
52962             var el = this.activePanel.getEl();
52963             w = w !== null ? w : el.getWidth();
52964             h = h !== null ? h : el.getHeight();
52965             this.panelSize = {width: w, height: h};
52966             this.activePanel.setSize(w, h);
52967         }
52968         if(Roo.isIE && this.tabs){
52969             this.tabs.el.repaint();
52970         }
52971     },
52972
52973     /**
52974      * Returns the container element for this region.
52975      * @return {Roo.Element}
52976      */
52977     getEl : function(){
52978         return this.el;
52979     },
52980
52981     /**
52982      * Hides this region.
52983      */
52984     hide : function(){
52985         if(!this.collapsed){
52986             this.el.dom.style.left = "-2000px";
52987             this.el.hide();
52988         }else{
52989             this.collapsedEl.dom.style.left = "-2000px";
52990             this.collapsedEl.hide();
52991         }
52992         this.visible = false;
52993         this.fireEvent("visibilitychange", this, false);
52994     },
52995
52996     /**
52997      * Shows this region if it was previously hidden.
52998      */
52999     show : function(){
53000         if(!this.collapsed){
53001             this.el.show();
53002         }else{
53003             this.collapsedEl.show();
53004         }
53005         this.visible = true;
53006         this.fireEvent("visibilitychange", this, true);
53007     },
53008
53009     closeClicked : function(){
53010         if(this.activePanel){
53011             this.remove(this.activePanel);
53012         }
53013     },
53014
53015     collapseClick : function(e){
53016         if(this.isSlid){
53017            e.stopPropagation();
53018            this.slideIn();
53019         }else{
53020            e.stopPropagation();
53021            this.slideOut();
53022         }
53023     },
53024
53025     /**
53026      * Collapses this region.
53027      * @param {Boolean} skipAnim (optional) true to collapse the element without animation (if animate is true)
53028      */
53029     collapse : function(skipAnim, skipCheck){
53030         if(this.collapsed) {
53031             return;
53032         }
53033         
53034         if(skipCheck || this.fireEvent("beforecollapse", this) != false){
53035             
53036             this.collapsed = true;
53037             if(this.split){
53038                 this.split.el.hide();
53039             }
53040             if(this.config.animate && skipAnim !== true){
53041                 this.fireEvent("invalidated", this);
53042                 this.animateCollapse();
53043             }else{
53044                 this.el.setLocation(-20000,-20000);
53045                 this.el.hide();
53046                 this.collapsedEl.show();
53047                 this.fireEvent("collapsed", this);
53048                 this.fireEvent("invalidated", this);
53049             }
53050         }
53051         
53052     },
53053
53054     animateCollapse : function(){
53055         // overridden
53056     },
53057
53058     /**
53059      * Expands this region if it was previously collapsed.
53060      * @param {Roo.EventObject} e The event that triggered the expand (or null if calling manually)
53061      * @param {Boolean} skipAnim (optional) true to expand the element without animation (if animate is true)
53062      */
53063     expand : function(e, skipAnim){
53064         if(e) {
53065             e.stopPropagation();
53066         }
53067         if(!this.collapsed || this.el.hasActiveFx()) {
53068             return;
53069         }
53070         if(this.isSlid){
53071             this.afterSlideIn();
53072             skipAnim = true;
53073         }
53074         this.collapsed = false;
53075         if(this.config.animate && skipAnim !== true){
53076             this.animateExpand();
53077         }else{
53078             this.el.show();
53079             if(this.split){
53080                 this.split.el.show();
53081             }
53082             this.collapsedEl.setLocation(-2000,-2000);
53083             this.collapsedEl.hide();
53084             this.fireEvent("invalidated", this);
53085             this.fireEvent("expanded", this);
53086         }
53087     },
53088
53089     animateExpand : function(){
53090         // overridden
53091     },
53092
53093     initTabs : function()
53094     {
53095         this.bodyEl.setStyle("overflow", "hidden");
53096         var ts = new Roo.TabPanel(
53097                 this.bodyEl.dom,
53098                 {
53099                     tabPosition: this.bottomTabs ? 'bottom' : 'top',
53100                     disableTooltips: this.config.disableTabTips,
53101                     toolbar : this.config.toolbar
53102                 }
53103         );
53104         if(this.config.hideTabs){
53105             ts.stripWrap.setDisplayed(false);
53106         }
53107         this.tabs = ts;
53108         ts.resizeTabs = this.config.resizeTabs === true;
53109         ts.minTabWidth = this.config.minTabWidth || 40;
53110         ts.maxTabWidth = this.config.maxTabWidth || 250;
53111         ts.preferredTabWidth = this.config.preferredTabWidth || 150;
53112         ts.monitorResize = false;
53113         ts.bodyEl.setStyle("overflow", this.config.autoScroll ? "auto" : "hidden");
53114         ts.bodyEl.addClass('x-layout-tabs-body');
53115         this.panels.each(this.initPanelAsTab, this);
53116     },
53117
53118     initPanelAsTab : function(panel){
53119         var ti = this.tabs.addTab(panel.getEl().id, panel.getTitle(), null,
53120                     this.config.closeOnTab && panel.isClosable());
53121         if(panel.tabTip !== undefined){
53122             ti.setTooltip(panel.tabTip);
53123         }
53124         ti.on("activate", function(){
53125               this.setActivePanel(panel);
53126         }, this);
53127         if(this.config.closeOnTab){
53128             ti.on("beforeclose", function(t, e){
53129                 e.cancel = true;
53130                 this.remove(panel);
53131             }, this);
53132         }
53133         return ti;
53134     },
53135
53136     updatePanelTitle : function(panel, title){
53137         if(this.activePanel == panel){
53138             this.updateTitle(title);
53139         }
53140         if(this.tabs){
53141             var ti = this.tabs.getTab(panel.getEl().id);
53142             ti.setText(title);
53143             if(panel.tabTip !== undefined){
53144                 ti.setTooltip(panel.tabTip);
53145             }
53146         }
53147     },
53148
53149     updateTitle : function(title){
53150         if(this.titleTextEl && !this.config.title){
53151             this.titleTextEl.innerHTML = (typeof title != "undefined" && title.length > 0 ? title : "&#160;");
53152         }
53153     },
53154
53155     setActivePanel : function(panel){
53156         panel = this.getPanel(panel);
53157         if(this.activePanel && this.activePanel != panel){
53158             this.activePanel.setActiveState(false);
53159         }
53160         this.activePanel = panel;
53161         panel.setActiveState(true);
53162         if(this.panelSize){
53163             panel.setSize(this.panelSize.width, this.panelSize.height);
53164         }
53165         if(this.closeBtn){
53166             this.closeBtn.setVisible(!this.config.closeOnTab && !this.isSlid && panel.isClosable());
53167         }
53168         this.updateTitle(panel.getTitle());
53169         if(this.tabs){
53170             this.fireEvent("invalidated", this);
53171         }
53172         this.fireEvent("panelactivated", this, panel);
53173     },
53174
53175     /**
53176      * Shows the specified panel.
53177      * @param {Number/String/ContentPanel} panelId The panel's index, id or the panel itself
53178      * @return {Roo.ContentPanel} The shown panel, or null if a panel could not be found from panelId
53179      */
53180     showPanel : function(panel)
53181     {
53182         panel = this.getPanel(panel);
53183         if(panel){
53184             if(this.tabs){
53185                 var tab = this.tabs.getTab(panel.getEl().id);
53186                 if(tab.isHidden()){
53187                     this.tabs.unhideTab(tab.id);
53188                 }
53189                 tab.activate();
53190             }else{
53191                 this.setActivePanel(panel);
53192             }
53193         }
53194         return panel;
53195     },
53196
53197     /**
53198      * Get the active panel for this region.
53199      * @return {Roo.ContentPanel} The active panel or null
53200      */
53201     getActivePanel : function(){
53202         return this.activePanel;
53203     },
53204
53205     validateVisibility : function(){
53206         if(this.panels.getCount() < 1){
53207             this.updateTitle("&#160;");
53208             this.closeBtn.hide();
53209             this.hide();
53210         }else{
53211             if(!this.isVisible()){
53212                 this.show();
53213             }
53214         }
53215     },
53216
53217     /**
53218      * Adds the passed ContentPanel(s) to this region.
53219      * @param {ContentPanel...} panel The ContentPanel(s) to add (you can pass more than one)
53220      * @return {Roo.ContentPanel} The panel added (if only one was added; null otherwise)
53221      */
53222     add : function(panel){
53223         if(arguments.length > 1){
53224             for(var i = 0, len = arguments.length; i < len; i++) {
53225                 this.add(arguments[i]);
53226             }
53227             return null;
53228         }
53229         if(this.hasPanel(panel)){
53230             this.showPanel(panel);
53231             return panel;
53232         }
53233         panel.setRegion(this);
53234         this.panels.add(panel);
53235         if(this.panels.getCount() == 1 && !this.config.alwaysShowTabs){
53236             this.bodyEl.dom.appendChild(panel.getEl().dom);
53237             if(panel.background !== true){
53238                 this.setActivePanel(panel);
53239             }
53240             this.fireEvent("paneladded", this, panel);
53241             return panel;
53242         }
53243         if(!this.tabs){
53244             this.initTabs();
53245         }else{
53246             this.initPanelAsTab(panel);
53247         }
53248         if(panel.background !== true){
53249             this.tabs.activate(panel.getEl().id);
53250         }
53251         this.fireEvent("paneladded", this, panel);
53252         return panel;
53253     },
53254
53255     /**
53256      * Hides the tab for the specified panel.
53257      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53258      */
53259     hidePanel : function(panel){
53260         if(this.tabs && (panel = this.getPanel(panel))){
53261             this.tabs.hideTab(panel.getEl().id);
53262         }
53263     },
53264
53265     /**
53266      * Unhides the tab for a previously hidden panel.
53267      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53268      */
53269     unhidePanel : function(panel){
53270         if(this.tabs && (panel = this.getPanel(panel))){
53271             this.tabs.unhideTab(panel.getEl().id);
53272         }
53273     },
53274
53275     clearPanels : function(){
53276         while(this.panels.getCount() > 0){
53277              this.remove(this.panels.first());
53278         }
53279     },
53280
53281     /**
53282      * Removes the specified panel. If preservePanel is not true (either here or in the config), the panel is destroyed.
53283      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53284      * @param {Boolean} preservePanel Overrides the config preservePanel option
53285      * @return {Roo.ContentPanel} The panel that was removed
53286      */
53287     remove : function(panel, preservePanel){
53288         panel = this.getPanel(panel);
53289         if(!panel){
53290             return null;
53291         }
53292         var e = {};
53293         this.fireEvent("beforeremove", this, panel, e);
53294         if(e.cancel === true){
53295             return null;
53296         }
53297         preservePanel = (typeof preservePanel != "undefined" ? preservePanel : (this.config.preservePanels === true || panel.preserve === true));
53298         var panelId = panel.getId();
53299         this.panels.removeKey(panelId);
53300         if(preservePanel){
53301             document.body.appendChild(panel.getEl().dom);
53302         }
53303         if(this.tabs){
53304             this.tabs.removeTab(panel.getEl().id);
53305         }else if (!preservePanel){
53306             this.bodyEl.dom.removeChild(panel.getEl().dom);
53307         }
53308         if(this.panels.getCount() == 1 && this.tabs && !this.config.alwaysShowTabs){
53309             var p = this.panels.first();
53310             var tempEl = document.createElement("div"); // temp holder to keep IE from deleting the node
53311             tempEl.appendChild(p.getEl().dom);
53312             this.bodyEl.update("");
53313             this.bodyEl.dom.appendChild(p.getEl().dom);
53314             tempEl = null;
53315             this.updateTitle(p.getTitle());
53316             this.tabs = null;
53317             this.bodyEl.setStyle("overflow", this.config.autoScroll ? "auto" : "hidden");
53318             this.setActivePanel(p);
53319         }
53320         panel.setRegion(null);
53321         if(this.activePanel == panel){
53322             this.activePanel = null;
53323         }
53324         if(this.config.autoDestroy !== false && preservePanel !== true){
53325             try{panel.destroy();}catch(e){}
53326         }
53327         this.fireEvent("panelremoved", this, panel);
53328         return panel;
53329     },
53330
53331     /**
53332      * Returns the TabPanel component used by this region
53333      * @return {Roo.TabPanel}
53334      */
53335     getTabs : function(){
53336         return this.tabs;
53337     },
53338
53339     createTool : function(parentEl, className){
53340         var btn = Roo.DomHelper.append(parentEl, {tag: "div", cls: "x-layout-tools-button",
53341             children: [{tag: "div", cls: "x-layout-tools-button-inner " + className, html: "&#160;"}]}, true);
53342         btn.addClassOnOver("x-layout-tools-button-over");
53343         return btn;
53344     }
53345 });/*
53346  * Based on:
53347  * Ext JS Library 1.1.1
53348  * Copyright(c) 2006-2007, Ext JS, LLC.
53349  *
53350  * Originally Released Under LGPL - original licence link has changed is not relivant.
53351  *
53352  * Fork - LGPL
53353  * <script type="text/javascript">
53354  */
53355  
53356
53357
53358 /**
53359  * @class Roo.SplitLayoutRegion
53360  * @extends Roo.LayoutRegion
53361  * Adds a splitbar and other (private) useful functionality to a {@link Roo.LayoutRegion}.
53362  */
53363 Roo.SplitLayoutRegion = function(mgr, config, pos, cursor){
53364     this.cursor = cursor;
53365     Roo.SplitLayoutRegion.superclass.constructor.call(this, mgr, config, pos);
53366 };
53367
53368 Roo.extend(Roo.SplitLayoutRegion, Roo.LayoutRegion, {
53369     splitTip : "Drag to resize.",
53370     collapsibleSplitTip : "Drag to resize. Double click to hide.",
53371     useSplitTips : false,
53372
53373     applyConfig : function(config){
53374         Roo.SplitLayoutRegion.superclass.applyConfig.call(this, config);
53375         if(config.split){
53376             if(!this.split){
53377                 var splitEl = Roo.DomHelper.append(this.mgr.el.dom, 
53378                         {tag: "div", id: this.el.id + "-split", cls: "x-layout-split x-layout-split-"+this.position, html: "&#160;"});
53379                 /** The SplitBar for this region 
53380                 * @type Roo.SplitBar */
53381                 this.split = new Roo.SplitBar(splitEl, this.el, this.orientation);
53382                 this.split.on("moved", this.onSplitMove, this);
53383                 this.split.useShim = config.useShim === true;
53384                 this.split.getMaximumSize = this[this.position == 'north' || this.position == 'south' ? 'getVMaxSize' : 'getHMaxSize'].createDelegate(this);
53385                 if(this.useSplitTips){
53386                     this.split.el.dom.title = config.collapsible ? this.collapsibleSplitTip : this.splitTip;
53387                 }
53388                 if(config.collapsible){
53389                     this.split.el.on("dblclick", this.collapse,  this);
53390                 }
53391             }
53392             if(typeof config.minSize != "undefined"){
53393                 this.split.minSize = config.minSize;
53394             }
53395             if(typeof config.maxSize != "undefined"){
53396                 this.split.maxSize = config.maxSize;
53397             }
53398             if(config.hideWhenEmpty || config.hidden || config.collapsed){
53399                 this.hideSplitter();
53400             }
53401         }
53402     },
53403
53404     getHMaxSize : function(){
53405          var cmax = this.config.maxSize || 10000;
53406          var center = this.mgr.getRegion("center");
53407          return Math.min(cmax, (this.el.getWidth()+center.getEl().getWidth())-center.getMinWidth());
53408     },
53409
53410     getVMaxSize : function(){
53411          var cmax = this.config.maxSize || 10000;
53412          var center = this.mgr.getRegion("center");
53413          return Math.min(cmax, (this.el.getHeight()+center.getEl().getHeight())-center.getMinHeight());
53414     },
53415
53416     onSplitMove : function(split, newSize){
53417         this.fireEvent("resized", this, newSize);
53418     },
53419     
53420     /** 
53421      * Returns the {@link Roo.SplitBar} for this region.
53422      * @return {Roo.SplitBar}
53423      */
53424     getSplitBar : function(){
53425         return this.split;
53426     },
53427     
53428     hide : function(){
53429         this.hideSplitter();
53430         Roo.SplitLayoutRegion.superclass.hide.call(this);
53431     },
53432
53433     hideSplitter : function(){
53434         if(this.split){
53435             this.split.el.setLocation(-2000,-2000);
53436             this.split.el.hide();
53437         }
53438     },
53439
53440     show : function(){
53441         if(this.split){
53442             this.split.el.show();
53443         }
53444         Roo.SplitLayoutRegion.superclass.show.call(this);
53445     },
53446     
53447     beforeSlide: function(){
53448         if(Roo.isGecko){// firefox overflow auto bug workaround
53449             this.bodyEl.clip();
53450             if(this.tabs) {
53451                 this.tabs.bodyEl.clip();
53452             }
53453             if(this.activePanel){
53454                 this.activePanel.getEl().clip();
53455                 
53456                 if(this.activePanel.beforeSlide){
53457                     this.activePanel.beforeSlide();
53458                 }
53459             }
53460         }
53461     },
53462     
53463     afterSlide : function(){
53464         if(Roo.isGecko){// firefox overflow auto bug workaround
53465             this.bodyEl.unclip();
53466             if(this.tabs) {
53467                 this.tabs.bodyEl.unclip();
53468             }
53469             if(this.activePanel){
53470                 this.activePanel.getEl().unclip();
53471                 if(this.activePanel.afterSlide){
53472                     this.activePanel.afterSlide();
53473                 }
53474             }
53475         }
53476     },
53477
53478     initAutoHide : function(){
53479         if(this.autoHide !== false){
53480             if(!this.autoHideHd){
53481                 var st = new Roo.util.DelayedTask(this.slideIn, this);
53482                 this.autoHideHd = {
53483                     "mouseout": function(e){
53484                         if(!e.within(this.el, true)){
53485                             st.delay(500);
53486                         }
53487                     },
53488                     "mouseover" : function(e){
53489                         st.cancel();
53490                     },
53491                     scope : this
53492                 };
53493             }
53494             this.el.on(this.autoHideHd);
53495         }
53496     },
53497
53498     clearAutoHide : function(){
53499         if(this.autoHide !== false){
53500             this.el.un("mouseout", this.autoHideHd.mouseout);
53501             this.el.un("mouseover", this.autoHideHd.mouseover);
53502         }
53503     },
53504
53505     clearMonitor : function(){
53506         Roo.get(document).un("click", this.slideInIf, this);
53507     },
53508
53509     // these names are backwards but not changed for compat
53510     slideOut : function(){
53511         if(this.isSlid || this.el.hasActiveFx()){
53512             return;
53513         }
53514         this.isSlid = true;
53515         if(this.collapseBtn){
53516             this.collapseBtn.hide();
53517         }
53518         this.closeBtnState = this.closeBtn.getStyle('display');
53519         this.closeBtn.hide();
53520         if(this.stickBtn){
53521             this.stickBtn.show();
53522         }
53523         this.el.show();
53524         this.el.alignTo(this.collapsedEl, this.getCollapseAnchor());
53525         this.beforeSlide();
53526         this.el.setStyle("z-index", 10001);
53527         this.el.slideIn(this.getSlideAnchor(), {
53528             callback: function(){
53529                 this.afterSlide();
53530                 this.initAutoHide();
53531                 Roo.get(document).on("click", this.slideInIf, this);
53532                 this.fireEvent("slideshow", this);
53533             },
53534             scope: this,
53535             block: true
53536         });
53537     },
53538
53539     afterSlideIn : function(){
53540         this.clearAutoHide();
53541         this.isSlid = false;
53542         this.clearMonitor();
53543         this.el.setStyle("z-index", "");
53544         if(this.collapseBtn){
53545             this.collapseBtn.show();
53546         }
53547         this.closeBtn.setStyle('display', this.closeBtnState);
53548         if(this.stickBtn){
53549             this.stickBtn.hide();
53550         }
53551         this.fireEvent("slidehide", this);
53552     },
53553
53554     slideIn : function(cb){
53555         if(!this.isSlid || this.el.hasActiveFx()){
53556             Roo.callback(cb);
53557             return;
53558         }
53559         this.isSlid = false;
53560         this.beforeSlide();
53561         this.el.slideOut(this.getSlideAnchor(), {
53562             callback: function(){
53563                 this.el.setLeftTop(-10000, -10000);
53564                 this.afterSlide();
53565                 this.afterSlideIn();
53566                 Roo.callback(cb);
53567             },
53568             scope: this,
53569             block: true
53570         });
53571     },
53572     
53573     slideInIf : function(e){
53574         if(!e.within(this.el)){
53575             this.slideIn();
53576         }
53577     },
53578
53579     animateCollapse : function(){
53580         this.beforeSlide();
53581         this.el.setStyle("z-index", 20000);
53582         var anchor = this.getSlideAnchor();
53583         this.el.slideOut(anchor, {
53584             callback : function(){
53585                 this.el.setStyle("z-index", "");
53586                 this.collapsedEl.slideIn(anchor, {duration:.3});
53587                 this.afterSlide();
53588                 this.el.setLocation(-10000,-10000);
53589                 this.el.hide();
53590                 this.fireEvent("collapsed", this);
53591             },
53592             scope: this,
53593             block: true
53594         });
53595     },
53596
53597     animateExpand : function(){
53598         this.beforeSlide();
53599         this.el.alignTo(this.collapsedEl, this.getCollapseAnchor(), this.getExpandAdj());
53600         this.el.setStyle("z-index", 20000);
53601         this.collapsedEl.hide({
53602             duration:.1
53603         });
53604         this.el.slideIn(this.getSlideAnchor(), {
53605             callback : function(){
53606                 this.el.setStyle("z-index", "");
53607                 this.afterSlide();
53608                 if(this.split){
53609                     this.split.el.show();
53610                 }
53611                 this.fireEvent("invalidated", this);
53612                 this.fireEvent("expanded", this);
53613             },
53614             scope: this,
53615             block: true
53616         });
53617     },
53618
53619     anchors : {
53620         "west" : "left",
53621         "east" : "right",
53622         "north" : "top",
53623         "south" : "bottom"
53624     },
53625
53626     sanchors : {
53627         "west" : "l",
53628         "east" : "r",
53629         "north" : "t",
53630         "south" : "b"
53631     },
53632
53633     canchors : {
53634         "west" : "tl-tr",
53635         "east" : "tr-tl",
53636         "north" : "tl-bl",
53637         "south" : "bl-tl"
53638     },
53639
53640     getAnchor : function(){
53641         return this.anchors[this.position];
53642     },
53643
53644     getCollapseAnchor : function(){
53645         return this.canchors[this.position];
53646     },
53647
53648     getSlideAnchor : function(){
53649         return this.sanchors[this.position];
53650     },
53651
53652     getAlignAdj : function(){
53653         var cm = this.cmargins;
53654         switch(this.position){
53655             case "west":
53656                 return [0, 0];
53657             break;
53658             case "east":
53659                 return [0, 0];
53660             break;
53661             case "north":
53662                 return [0, 0];
53663             break;
53664             case "south":
53665                 return [0, 0];
53666             break;
53667         }
53668     },
53669
53670     getExpandAdj : function(){
53671         var c = this.collapsedEl, cm = this.cmargins;
53672         switch(this.position){
53673             case "west":
53674                 return [-(cm.right+c.getWidth()+cm.left), 0];
53675             break;
53676             case "east":
53677                 return [cm.right+c.getWidth()+cm.left, 0];
53678             break;
53679             case "north":
53680                 return [0, -(cm.top+cm.bottom+c.getHeight())];
53681             break;
53682             case "south":
53683                 return [0, cm.top+cm.bottom+c.getHeight()];
53684             break;
53685         }
53686     }
53687 });/*
53688  * Based on:
53689  * Ext JS Library 1.1.1
53690  * Copyright(c) 2006-2007, Ext JS, LLC.
53691  *
53692  * Originally Released Under LGPL - original licence link has changed is not relivant.
53693  *
53694  * Fork - LGPL
53695  * <script type="text/javascript">
53696  */
53697 /*
53698  * These classes are private internal classes
53699  */
53700 Roo.CenterLayoutRegion = function(mgr, config){
53701     Roo.LayoutRegion.call(this, mgr, config, "center");
53702     this.visible = true;
53703     this.minWidth = config.minWidth || 20;
53704     this.minHeight = config.minHeight || 20;
53705 };
53706
53707 Roo.extend(Roo.CenterLayoutRegion, Roo.LayoutRegion, {
53708     hide : function(){
53709         // center panel can't be hidden
53710     },
53711     
53712     show : function(){
53713         // center panel can't be hidden
53714     },
53715     
53716     getMinWidth: function(){
53717         return this.minWidth;
53718     },
53719     
53720     getMinHeight: function(){
53721         return this.minHeight;
53722     }
53723 });
53724
53725
53726 Roo.NorthLayoutRegion = function(mgr, config){
53727     Roo.LayoutRegion.call(this, mgr, config, "north", "n-resize");
53728     if(this.split){
53729         this.split.placement = Roo.SplitBar.TOP;
53730         this.split.orientation = Roo.SplitBar.VERTICAL;
53731         this.split.el.addClass("x-layout-split-v");
53732     }
53733     var size = config.initialSize || config.height;
53734     if(typeof size != "undefined"){
53735         this.el.setHeight(size);
53736     }
53737 };
53738 Roo.extend(Roo.NorthLayoutRegion, Roo.SplitLayoutRegion, {
53739     orientation: Roo.SplitBar.VERTICAL,
53740     getBox : function(){
53741         if(this.collapsed){
53742             return this.collapsedEl.getBox();
53743         }
53744         var box = this.el.getBox();
53745         if(this.split){
53746             box.height += this.split.el.getHeight();
53747         }
53748         return box;
53749     },
53750     
53751     updateBox : function(box){
53752         if(this.split && !this.collapsed){
53753             box.height -= this.split.el.getHeight();
53754             this.split.el.setLeft(box.x);
53755             this.split.el.setTop(box.y+box.height);
53756             this.split.el.setWidth(box.width);
53757         }
53758         if(this.collapsed){
53759             this.updateBody(box.width, null);
53760         }
53761         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53762     }
53763 });
53764
53765 Roo.SouthLayoutRegion = function(mgr, config){
53766     Roo.SplitLayoutRegion.call(this, mgr, config, "south", "s-resize");
53767     if(this.split){
53768         this.split.placement = Roo.SplitBar.BOTTOM;
53769         this.split.orientation = Roo.SplitBar.VERTICAL;
53770         this.split.el.addClass("x-layout-split-v");
53771     }
53772     var size = config.initialSize || config.height;
53773     if(typeof size != "undefined"){
53774         this.el.setHeight(size);
53775     }
53776 };
53777 Roo.extend(Roo.SouthLayoutRegion, Roo.SplitLayoutRegion, {
53778     orientation: Roo.SplitBar.VERTICAL,
53779     getBox : function(){
53780         if(this.collapsed){
53781             return this.collapsedEl.getBox();
53782         }
53783         var box = this.el.getBox();
53784         if(this.split){
53785             var sh = this.split.el.getHeight();
53786             box.height += sh;
53787             box.y -= sh;
53788         }
53789         return box;
53790     },
53791     
53792     updateBox : function(box){
53793         if(this.split && !this.collapsed){
53794             var sh = this.split.el.getHeight();
53795             box.height -= sh;
53796             box.y += sh;
53797             this.split.el.setLeft(box.x);
53798             this.split.el.setTop(box.y-sh);
53799             this.split.el.setWidth(box.width);
53800         }
53801         if(this.collapsed){
53802             this.updateBody(box.width, null);
53803         }
53804         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53805     }
53806 });
53807
53808 Roo.EastLayoutRegion = function(mgr, config){
53809     Roo.SplitLayoutRegion.call(this, mgr, config, "east", "e-resize");
53810     if(this.split){
53811         this.split.placement = Roo.SplitBar.RIGHT;
53812         this.split.orientation = Roo.SplitBar.HORIZONTAL;
53813         this.split.el.addClass("x-layout-split-h");
53814     }
53815     var size = config.initialSize || config.width;
53816     if(typeof size != "undefined"){
53817         this.el.setWidth(size);
53818     }
53819 };
53820 Roo.extend(Roo.EastLayoutRegion, Roo.SplitLayoutRegion, {
53821     orientation: Roo.SplitBar.HORIZONTAL,
53822     getBox : function(){
53823         if(this.collapsed){
53824             return this.collapsedEl.getBox();
53825         }
53826         var box = this.el.getBox();
53827         if(this.split){
53828             var sw = this.split.el.getWidth();
53829             box.width += sw;
53830             box.x -= sw;
53831         }
53832         return box;
53833     },
53834
53835     updateBox : function(box){
53836         if(this.split && !this.collapsed){
53837             var sw = this.split.el.getWidth();
53838             box.width -= sw;
53839             this.split.el.setLeft(box.x);
53840             this.split.el.setTop(box.y);
53841             this.split.el.setHeight(box.height);
53842             box.x += sw;
53843         }
53844         if(this.collapsed){
53845             this.updateBody(null, box.height);
53846         }
53847         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53848     }
53849 });
53850
53851 Roo.WestLayoutRegion = function(mgr, config){
53852     Roo.SplitLayoutRegion.call(this, mgr, config, "west", "w-resize");
53853     if(this.split){
53854         this.split.placement = Roo.SplitBar.LEFT;
53855         this.split.orientation = Roo.SplitBar.HORIZONTAL;
53856         this.split.el.addClass("x-layout-split-h");
53857     }
53858     var size = config.initialSize || config.width;
53859     if(typeof size != "undefined"){
53860         this.el.setWidth(size);
53861     }
53862 };
53863 Roo.extend(Roo.WestLayoutRegion, Roo.SplitLayoutRegion, {
53864     orientation: Roo.SplitBar.HORIZONTAL,
53865     getBox : function(){
53866         if(this.collapsed){
53867             return this.collapsedEl.getBox();
53868         }
53869         var box = this.el.getBox();
53870         if(this.split){
53871             box.width += this.split.el.getWidth();
53872         }
53873         return box;
53874     },
53875     
53876     updateBox : function(box){
53877         if(this.split && !this.collapsed){
53878             var sw = this.split.el.getWidth();
53879             box.width -= sw;
53880             this.split.el.setLeft(box.x+box.width);
53881             this.split.el.setTop(box.y);
53882             this.split.el.setHeight(box.height);
53883         }
53884         if(this.collapsed){
53885             this.updateBody(null, box.height);
53886         }
53887         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53888     }
53889 });
53890 /*
53891  * Based on:
53892  * Ext JS Library 1.1.1
53893  * Copyright(c) 2006-2007, Ext JS, LLC.
53894  *
53895  * Originally Released Under LGPL - original licence link has changed is not relivant.
53896  *
53897  * Fork - LGPL
53898  * <script type="text/javascript">
53899  */
53900  
53901  
53902 /*
53903  * Private internal class for reading and applying state
53904  */
53905 Roo.LayoutStateManager = function(layout){
53906      // default empty state
53907      this.state = {
53908         north: {},
53909         south: {},
53910         east: {},
53911         west: {}       
53912     };
53913 };
53914
53915 Roo.LayoutStateManager.prototype = {
53916     init : function(layout, provider){
53917         this.provider = provider;
53918         var state = provider.get(layout.id+"-layout-state");
53919         if(state){
53920             var wasUpdating = layout.isUpdating();
53921             if(!wasUpdating){
53922                 layout.beginUpdate();
53923             }
53924             for(var key in state){
53925                 if(typeof state[key] != "function"){
53926                     var rstate = state[key];
53927                     var r = layout.getRegion(key);
53928                     if(r && rstate){
53929                         if(rstate.size){
53930                             r.resizeTo(rstate.size);
53931                         }
53932                         if(rstate.collapsed == true){
53933                             r.collapse(true);
53934                         }else{
53935                             r.expand(null, true);
53936                         }
53937                     }
53938                 }
53939             }
53940             if(!wasUpdating){
53941                 layout.endUpdate();
53942             }
53943             this.state = state; 
53944         }
53945         this.layout = layout;
53946         layout.on("regionresized", this.onRegionResized, this);
53947         layout.on("regioncollapsed", this.onRegionCollapsed, this);
53948         layout.on("regionexpanded", this.onRegionExpanded, this);
53949     },
53950     
53951     storeState : function(){
53952         this.provider.set(this.layout.id+"-layout-state", this.state);
53953     },
53954     
53955     onRegionResized : function(region, newSize){
53956         this.state[region.getPosition()].size = newSize;
53957         this.storeState();
53958     },
53959     
53960     onRegionCollapsed : function(region){
53961         this.state[region.getPosition()].collapsed = true;
53962         this.storeState();
53963     },
53964     
53965     onRegionExpanded : function(region){
53966         this.state[region.getPosition()].collapsed = false;
53967         this.storeState();
53968     }
53969 };/*
53970  * Based on:
53971  * Ext JS Library 1.1.1
53972  * Copyright(c) 2006-2007, Ext JS, LLC.
53973  *
53974  * Originally Released Under LGPL - original licence link has changed is not relivant.
53975  *
53976  * Fork - LGPL
53977  * <script type="text/javascript">
53978  */
53979 /**
53980  * @class Roo.ContentPanel
53981  * @extends Roo.util.Observable
53982  * A basic ContentPanel element.
53983  * @cfg {Boolean}   fitToFrame    True for this panel to adjust its size to fit when the region resizes  (defaults to false)
53984  * @cfg {Boolean}   fitContainer   When using {@link #fitToFrame} and {@link #resizeEl}, you can also fit the parent container  (defaults to false)
53985  * @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
53986  * @cfg {Boolean}   closable      True if the panel can be closed/removed
53987  * @cfg {Boolean}   background    True if the panel should not be activated when it is added (defaults to false)
53988  * @cfg {String/HTMLElement/Element} resizeEl An element to resize if {@link #fitToFrame} is true (instead of this panel's element)
53989  * @cfg {Toolbar}   toolbar       A toolbar for this panel
53990  * @cfg {Boolean} autoScroll    True to scroll overflow in this panel (use with {@link #fitToFrame})
53991  * @cfg {String} title          The title for this panel
53992  * @cfg {Array} adjustments     Values to <b>add</b> to the width/height when doing a {@link #fitToFrame} (default is [0, 0])
53993  * @cfg {String} url            Calls {@link #setUrl} with this value
53994  * @cfg {String} region         (center|north|south|east|west) which region to put this panel on (when used with xtype constructors)
53995  * @cfg {String/Object} params  When used with {@link #url}, calls {@link #setUrl} with this value
53996  * @cfg {Boolean} loadOnce      When used with {@link #url}, calls {@link #setUrl} with this value
53997  * @cfg {String}    content        Raw content to fill content panel with (uses setContent on construction.)
53998
53999  * @constructor
54000  * Create a new ContentPanel.
54001  * @param {String/HTMLElement/Roo.Element} el The container element for this panel
54002  * @param {String/Object} config A string to set only the title or a config object
54003  * @param {String} content (optional) Set the HTML content for this panel
54004  * @param {String} region (optional) Used by xtype constructors to add to regions. (values center,east,west,south,north)
54005  */
54006 Roo.ContentPanel = function(el, config, content){
54007     
54008      
54009     /*
54010     if(el.autoCreate || el.xtype){ // xtype is available if this is called from factory
54011         config = el;
54012         el = Roo.id();
54013     }
54014     if (config && config.parentLayout) { 
54015         el = config.parentLayout.el.createChild(); 
54016     }
54017     */
54018     if(el.autoCreate){ // xtype is available if this is called from factory
54019         config = el;
54020         el = Roo.id();
54021     }
54022     this.el = Roo.get(el);
54023     if(!this.el && config && config.autoCreate){
54024         if(typeof config.autoCreate == "object"){
54025             if(!config.autoCreate.id){
54026                 config.autoCreate.id = config.id||el;
54027             }
54028             this.el = Roo.DomHelper.append(document.body,
54029                         config.autoCreate, true);
54030         }else{
54031             this.el = Roo.DomHelper.append(document.body,
54032                         {tag: "div", cls: "x-layout-inactive-content", id: config.id||el}, true);
54033         }
54034     }
54035     this.closable = false;
54036     this.loaded = false;
54037     this.active = false;
54038     if(typeof config == "string"){
54039         this.title = config;
54040     }else{
54041         Roo.apply(this, config);
54042     }
54043     
54044     if (this.toolbar && !this.toolbar.el && this.toolbar.xtype) {
54045         this.wrapEl = this.el.wrap();
54046         this.toolbar.container = this.el.insertSibling(false, 'before');
54047         this.toolbar = new Roo.Toolbar(this.toolbar);
54048     }
54049     
54050     // xtype created footer. - not sure if will work as we normally have to render first..
54051     if (this.footer && !this.footer.el && this.footer.xtype) {
54052         if (!this.wrapEl) {
54053             this.wrapEl = this.el.wrap();
54054         }
54055     
54056         this.footer.container = this.wrapEl.createChild();
54057          
54058         this.footer = Roo.factory(this.footer, Roo);
54059         
54060     }
54061     
54062     if(this.resizeEl){
54063         this.resizeEl = Roo.get(this.resizeEl, true);
54064     }else{
54065         this.resizeEl = this.el;
54066     }
54067     // handle view.xtype
54068     
54069  
54070     
54071     
54072     this.addEvents({
54073         /**
54074          * @event activate
54075          * Fires when this panel is activated. 
54076          * @param {Roo.ContentPanel} this
54077          */
54078         "activate" : true,
54079         /**
54080          * @event deactivate
54081          * Fires when this panel is activated. 
54082          * @param {Roo.ContentPanel} this
54083          */
54084         "deactivate" : true,
54085
54086         /**
54087          * @event resize
54088          * Fires when this panel is resized if fitToFrame is true.
54089          * @param {Roo.ContentPanel} this
54090          * @param {Number} width The width after any component adjustments
54091          * @param {Number} height The height after any component adjustments
54092          */
54093         "resize" : true,
54094         
54095          /**
54096          * @event render
54097          * Fires when this tab is created
54098          * @param {Roo.ContentPanel} this
54099          */
54100         "render" : true
54101          
54102         
54103     });
54104     
54105
54106     
54107     
54108     if(this.autoScroll){
54109         this.resizeEl.setStyle("overflow", "auto");
54110     } else {
54111         // fix randome scrolling
54112         this.el.on('scroll', function() {
54113             Roo.log('fix random scolling');
54114             this.scrollTo('top',0); 
54115         });
54116     }
54117     content = content || this.content;
54118     if(content){
54119         this.setContent(content);
54120     }
54121     if(config && config.url){
54122         this.setUrl(this.url, this.params, this.loadOnce);
54123     }
54124     
54125     
54126     
54127     Roo.ContentPanel.superclass.constructor.call(this);
54128     
54129     if (this.view && typeof(this.view.xtype) != 'undefined') {
54130         this.view.el = this.el.appendChild(document.createElement("div"));
54131         this.view = Roo.factory(this.view); 
54132         this.view.render  &&  this.view.render(false, '');  
54133     }
54134     
54135     
54136     this.fireEvent('render', this);
54137 };
54138
54139 Roo.extend(Roo.ContentPanel, Roo.util.Observable, {
54140     tabTip:'',
54141     setRegion : function(region){
54142         this.region = region;
54143         if(region){
54144            this.el.replaceClass("x-layout-inactive-content", "x-layout-active-content");
54145         }else{
54146            this.el.replaceClass("x-layout-active-content", "x-layout-inactive-content");
54147         } 
54148     },
54149     
54150     /**
54151      * Returns the toolbar for this Panel if one was configured. 
54152      * @return {Roo.Toolbar} 
54153      */
54154     getToolbar : function(){
54155         return this.toolbar;
54156     },
54157     
54158     setActiveState : function(active){
54159         this.active = active;
54160         if(!active){
54161             this.fireEvent("deactivate", this);
54162         }else{
54163             this.fireEvent("activate", this);
54164         }
54165     },
54166     /**
54167      * Updates this panel's element
54168      * @param {String} content The new content
54169      * @param {Boolean} loadScripts (optional) true to look for and process scripts
54170     */
54171     setContent : function(content, loadScripts){
54172         this.el.update(content, loadScripts);
54173     },
54174
54175     ignoreResize : function(w, h){
54176         if(this.lastSize && this.lastSize.width == w && this.lastSize.height == h){
54177             return true;
54178         }else{
54179             this.lastSize = {width: w, height: h};
54180             return false;
54181         }
54182     },
54183     /**
54184      * Get the {@link Roo.UpdateManager} for this panel. Enables you to perform Ajax updates.
54185      * @return {Roo.UpdateManager} The UpdateManager
54186      */
54187     getUpdateManager : function(){
54188         return this.el.getUpdateManager();
54189     },
54190      /**
54191      * Loads this content panel immediately with content from XHR. Note: to delay loading until the panel is activated, use {@link #setUrl}.
54192      * @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:
54193 <pre><code>
54194 panel.load({
54195     url: "your-url.php",
54196     params: {param1: "foo", param2: "bar"}, // or a URL encoded string
54197     callback: yourFunction,
54198     scope: yourObject, //(optional scope)
54199     discardUrl: false,
54200     nocache: false,
54201     text: "Loading...",
54202     timeout: 30,
54203     scripts: false
54204 });
54205 </code></pre>
54206      * The only required property is <i>url</i>. The optional properties <i>nocache</i>, <i>text</i> and <i>scripts</i>
54207      * 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.
54208      * @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}
54209      * @param {Function} callback (optional) Callback when transaction is complete -- called with signature (oElement, bSuccess, oResponse)
54210      * @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.
54211      * @return {Roo.ContentPanel} this
54212      */
54213     load : function(){
54214         var um = this.el.getUpdateManager();
54215         um.update.apply(um, arguments);
54216         return this;
54217     },
54218
54219
54220     /**
54221      * 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.
54222      * @param {String/Function} url The URL to load the content from or a function to call to get the URL
54223      * @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)
54224      * @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)
54225      * @return {Roo.UpdateManager} The UpdateManager
54226      */
54227     setUrl : function(url, params, loadOnce){
54228         if(this.refreshDelegate){
54229             this.removeListener("activate", this.refreshDelegate);
54230         }
54231         this.refreshDelegate = this._handleRefresh.createDelegate(this, [url, params, loadOnce]);
54232         this.on("activate", this.refreshDelegate);
54233         return this.el.getUpdateManager();
54234     },
54235     
54236     _handleRefresh : function(url, params, loadOnce){
54237         if(!loadOnce || !this.loaded){
54238             var updater = this.el.getUpdateManager();
54239             updater.update(url, params, this._setLoaded.createDelegate(this));
54240         }
54241     },
54242     
54243     _setLoaded : function(){
54244         this.loaded = true;
54245     }, 
54246     
54247     /**
54248      * Returns this panel's id
54249      * @return {String} 
54250      */
54251     getId : function(){
54252         return this.el.id;
54253     },
54254     
54255     /** 
54256      * Returns this panel's element - used by regiosn to add.
54257      * @return {Roo.Element} 
54258      */
54259     getEl : function(){
54260         return this.wrapEl || this.el;
54261     },
54262     
54263     adjustForComponents : function(width, height)
54264     {
54265         //Roo.log('adjustForComponents ');
54266         if(this.resizeEl != this.el){
54267             width -= this.el.getFrameWidth('lr');
54268             height -= this.el.getFrameWidth('tb');
54269         }
54270         if(this.toolbar){
54271             var te = this.toolbar.getEl();
54272             height -= te.getHeight();
54273             te.setWidth(width);
54274         }
54275         if(this.footer){
54276             var te = this.footer.getEl();
54277             //Roo.log("footer:" + te.getHeight());
54278             
54279             height -= te.getHeight();
54280             te.setWidth(width);
54281         }
54282         
54283         
54284         if(this.adjustments){
54285             width += this.adjustments[0];
54286             height += this.adjustments[1];
54287         }
54288         return {"width": width, "height": height};
54289     },
54290     
54291     setSize : function(width, height){
54292         if(this.fitToFrame && !this.ignoreResize(width, height)){
54293             if(this.fitContainer && this.resizeEl != this.el){
54294                 this.el.setSize(width, height);
54295             }
54296             var size = this.adjustForComponents(width, height);
54297             this.resizeEl.setSize(this.autoWidth ? "auto" : size.width, this.autoHeight ? "auto" : size.height);
54298             this.fireEvent('resize', this, size.width, size.height);
54299         }
54300     },
54301     
54302     /**
54303      * Returns this panel's title
54304      * @return {String} 
54305      */
54306     getTitle : function(){
54307         return this.title;
54308     },
54309     
54310     /**
54311      * Set this panel's title
54312      * @param {String} title
54313      */
54314     setTitle : function(title){
54315         this.title = title;
54316         if(this.region){
54317             this.region.updatePanelTitle(this, title);
54318         }
54319     },
54320     
54321     /**
54322      * Returns true is this panel was configured to be closable
54323      * @return {Boolean} 
54324      */
54325     isClosable : function(){
54326         return this.closable;
54327     },
54328     
54329     beforeSlide : function(){
54330         this.el.clip();
54331         this.resizeEl.clip();
54332     },
54333     
54334     afterSlide : function(){
54335         this.el.unclip();
54336         this.resizeEl.unclip();
54337     },
54338     
54339     /**
54340      *   Force a content refresh from the URL specified in the {@link #setUrl} method.
54341      *   Will fail silently if the {@link #setUrl} method has not been called.
54342      *   This does not activate the panel, just updates its content.
54343      */
54344     refresh : function(){
54345         if(this.refreshDelegate){
54346            this.loaded = false;
54347            this.refreshDelegate();
54348         }
54349     },
54350     
54351     /**
54352      * Destroys this panel
54353      */
54354     destroy : function(){
54355         this.el.removeAllListeners();
54356         var tempEl = document.createElement("span");
54357         tempEl.appendChild(this.el.dom);
54358         tempEl.innerHTML = "";
54359         this.el.remove();
54360         this.el = null;
54361     },
54362     
54363     /**
54364      * form - if the content panel contains a form - this is a reference to it.
54365      * @type {Roo.form.Form}
54366      */
54367     form : false,
54368     /**
54369      * view - if the content panel contains a view (Roo.DatePicker / Roo.View / Roo.JsonView)
54370      *    This contains a reference to it.
54371      * @type {Roo.View}
54372      */
54373     view : false,
54374     
54375       /**
54376      * Adds a xtype elements to the panel - currently only supports Forms, View, JsonView.
54377      * <pre><code>
54378
54379 layout.addxtype({
54380        xtype : 'Form',
54381        items: [ .... ]
54382    }
54383 );
54384
54385 </code></pre>
54386      * @param {Object} cfg Xtype definition of item to add.
54387      */
54388     
54389     addxtype : function(cfg) {
54390         // add form..
54391         if (cfg.xtype.match(/^Form$/)) {
54392             
54393             var el;
54394             //if (this.footer) {
54395             //    el = this.footer.container.insertSibling(false, 'before');
54396             //} else {
54397                 el = this.el.createChild();
54398             //}
54399
54400             this.form = new  Roo.form.Form(cfg);
54401             
54402             
54403             if ( this.form.allItems.length) {
54404                 this.form.render(el.dom);
54405             }
54406             return this.form;
54407         }
54408         // should only have one of theses..
54409         if ([ 'View', 'JsonView', 'DatePicker'].indexOf(cfg.xtype) > -1) {
54410             // views.. should not be just added - used named prop 'view''
54411             
54412             cfg.el = this.el.appendChild(document.createElement("div"));
54413             // factory?
54414             
54415             var ret = new Roo.factory(cfg);
54416              
54417              ret.render && ret.render(false, ''); // render blank..
54418             this.view = ret;
54419             return ret;
54420         }
54421         return false;
54422     }
54423 });
54424
54425 /**
54426  * @class Roo.GridPanel
54427  * @extends Roo.ContentPanel
54428  * @constructor
54429  * Create a new GridPanel.
54430  * @param {Roo.grid.Grid} grid The grid for this panel
54431  * @param {String/Object} config A string to set only the panel's title, or a config object
54432  */
54433 Roo.GridPanel = function(grid, config){
54434     
54435   
54436     this.wrapper = Roo.DomHelper.append(document.body, // wrapper for IE7 strict & safari scroll issue
54437         {tag: "div", cls: "x-layout-grid-wrapper x-layout-inactive-content"}, true);
54438         
54439     this.wrapper.dom.appendChild(grid.getGridEl().dom);
54440     
54441     Roo.GridPanel.superclass.constructor.call(this, this.wrapper, config);
54442     
54443     if(this.toolbar){
54444         this.toolbar.el.insertBefore(this.wrapper.dom.firstChild);
54445     }
54446     // xtype created footer. - not sure if will work as we normally have to render first..
54447     if (this.footer && !this.footer.el && this.footer.xtype) {
54448         
54449         this.footer.container = this.grid.getView().getFooterPanel(true);
54450         this.footer.dataSource = this.grid.dataSource;
54451         this.footer = Roo.factory(this.footer, Roo);
54452         
54453     }
54454     
54455     grid.monitorWindowResize = false; // turn off autosizing
54456     grid.autoHeight = false;
54457     grid.autoWidth = false;
54458     this.grid = grid;
54459     this.grid.getGridEl().replaceClass("x-layout-inactive-content", "x-layout-component-panel");
54460 };
54461
54462 Roo.extend(Roo.GridPanel, Roo.ContentPanel, {
54463     getId : function(){
54464         return this.grid.id;
54465     },
54466     
54467     /**
54468      * Returns the grid for this panel
54469      * @return {Roo.grid.Grid} 
54470      */
54471     getGrid : function(){
54472         return this.grid;    
54473     },
54474     
54475     setSize : function(width, height){
54476         if(!this.ignoreResize(width, height)){
54477             var grid = this.grid;
54478             var size = this.adjustForComponents(width, height);
54479             grid.getGridEl().setSize(size.width, size.height);
54480             grid.autoSize();
54481         }
54482     },
54483     
54484     beforeSlide : function(){
54485         this.grid.getView().scroller.clip();
54486     },
54487     
54488     afterSlide : function(){
54489         this.grid.getView().scroller.unclip();
54490     },
54491     
54492     destroy : function(){
54493         this.grid.destroy();
54494         delete this.grid;
54495         Roo.GridPanel.superclass.destroy.call(this); 
54496     }
54497 });
54498
54499
54500 /**
54501  * @class Roo.NestedLayoutPanel
54502  * @extends Roo.ContentPanel
54503  * @constructor
54504  * Create a new NestedLayoutPanel.
54505  * 
54506  * 
54507  * @param {Roo.BorderLayout} layout The layout for this panel
54508  * @param {String/Object} config A string to set only the title or a config object
54509  */
54510 Roo.NestedLayoutPanel = function(layout, config)
54511 {
54512     // construct with only one argument..
54513     /* FIXME - implement nicer consturctors
54514     if (layout.layout) {
54515         config = layout;
54516         layout = config.layout;
54517         delete config.layout;
54518     }
54519     if (layout.xtype && !layout.getEl) {
54520         // then layout needs constructing..
54521         layout = Roo.factory(layout, Roo);
54522     }
54523     */
54524     
54525     
54526     Roo.NestedLayoutPanel.superclass.constructor.call(this, layout.getEl(), config);
54527     
54528     layout.monitorWindowResize = false; // turn off autosizing
54529     this.layout = layout;
54530     this.layout.getEl().addClass("x-layout-nested-layout");
54531     
54532     
54533     
54534     
54535 };
54536
54537 Roo.extend(Roo.NestedLayoutPanel, Roo.ContentPanel, {
54538
54539     setSize : function(width, height){
54540         if(!this.ignoreResize(width, height)){
54541             var size = this.adjustForComponents(width, height);
54542             var el = this.layout.getEl();
54543             el.setSize(size.width, size.height);
54544             var touch = el.dom.offsetWidth;
54545             this.layout.layout();
54546             // ie requires a double layout on the first pass
54547             if(Roo.isIE && !this.initialized){
54548                 this.initialized = true;
54549                 this.layout.layout();
54550             }
54551         }
54552     },
54553     
54554     // activate all subpanels if not currently active..
54555     
54556     setActiveState : function(active){
54557         this.active = active;
54558         if(!active){
54559             this.fireEvent("deactivate", this);
54560             return;
54561         }
54562         
54563         this.fireEvent("activate", this);
54564         // not sure if this should happen before or after..
54565         if (!this.layout) {
54566             return; // should not happen..
54567         }
54568         var reg = false;
54569         for (var r in this.layout.regions) {
54570             reg = this.layout.getRegion(r);
54571             if (reg.getActivePanel()) {
54572                 //reg.showPanel(reg.getActivePanel()); // force it to activate.. 
54573                 reg.setActivePanel(reg.getActivePanel());
54574                 continue;
54575             }
54576             if (!reg.panels.length) {
54577                 continue;
54578             }
54579             reg.showPanel(reg.getPanel(0));
54580         }
54581         
54582         
54583         
54584         
54585     },
54586     
54587     /**
54588      * Returns the nested BorderLayout for this panel
54589      * @return {Roo.BorderLayout} 
54590      */
54591     getLayout : function(){
54592         return this.layout;
54593     },
54594     
54595      /**
54596      * Adds a xtype elements to the layout of the nested panel
54597      * <pre><code>
54598
54599 panel.addxtype({
54600        xtype : 'ContentPanel',
54601        region: 'west',
54602        items: [ .... ]
54603    }
54604 );
54605
54606 panel.addxtype({
54607         xtype : 'NestedLayoutPanel',
54608         region: 'west',
54609         layout: {
54610            center: { },
54611            west: { }   
54612         },
54613         items : [ ... list of content panels or nested layout panels.. ]
54614    }
54615 );
54616 </code></pre>
54617      * @param {Object} cfg Xtype definition of item to add.
54618      */
54619     addxtype : function(cfg) {
54620         return this.layout.addxtype(cfg);
54621     
54622     }
54623 });
54624
54625 Roo.ScrollPanel = function(el, config, content){
54626     config = config || {};
54627     config.fitToFrame = true;
54628     Roo.ScrollPanel.superclass.constructor.call(this, el, config, content);
54629     
54630     this.el.dom.style.overflow = "hidden";
54631     var wrap = this.el.wrap({cls: "x-scroller x-layout-inactive-content"});
54632     this.el.removeClass("x-layout-inactive-content");
54633     this.el.on("mousewheel", this.onWheel, this);
54634
54635     var up = wrap.createChild({cls: "x-scroller-up", html: "&#160;"}, this.el.dom);
54636     var down = wrap.createChild({cls: "x-scroller-down", html: "&#160;"});
54637     up.unselectable(); down.unselectable();
54638     up.on("click", this.scrollUp, this);
54639     down.on("click", this.scrollDown, this);
54640     up.addClassOnOver("x-scroller-btn-over");
54641     down.addClassOnOver("x-scroller-btn-over");
54642     up.addClassOnClick("x-scroller-btn-click");
54643     down.addClassOnClick("x-scroller-btn-click");
54644     this.adjustments = [0, -(up.getHeight() + down.getHeight())];
54645
54646     this.resizeEl = this.el;
54647     this.el = wrap; this.up = up; this.down = down;
54648 };
54649
54650 Roo.extend(Roo.ScrollPanel, Roo.ContentPanel, {
54651     increment : 100,
54652     wheelIncrement : 5,
54653     scrollUp : function(){
54654         this.resizeEl.scroll("up", this.increment, {callback: this.afterScroll, scope: this});
54655     },
54656
54657     scrollDown : function(){
54658         this.resizeEl.scroll("down", this.increment, {callback: this.afterScroll, scope: this});
54659     },
54660
54661     afterScroll : function(){
54662         var el = this.resizeEl;
54663         var t = el.dom.scrollTop, h = el.dom.scrollHeight, ch = el.dom.clientHeight;
54664         this.up[t == 0 ? "addClass" : "removeClass"]("x-scroller-btn-disabled");
54665         this.down[h - t <= ch ? "addClass" : "removeClass"]("x-scroller-btn-disabled");
54666     },
54667
54668     setSize : function(){
54669         Roo.ScrollPanel.superclass.setSize.apply(this, arguments);
54670         this.afterScroll();
54671     },
54672
54673     onWheel : function(e){
54674         var d = e.getWheelDelta();
54675         this.resizeEl.dom.scrollTop -= (d*this.wheelIncrement);
54676         this.afterScroll();
54677         e.stopEvent();
54678     },
54679
54680     setContent : function(content, loadScripts){
54681         this.resizeEl.update(content, loadScripts);
54682     }
54683
54684 });
54685
54686
54687
54688
54689
54690
54691
54692
54693
54694 /**
54695  * @class Roo.TreePanel
54696  * @extends Roo.ContentPanel
54697  * @constructor
54698  * Create a new TreePanel. - defaults to fit/scoll contents.
54699  * @param {String/Object} config A string to set only the panel's title, or a config object
54700  * @cfg {Roo.tree.TreePanel} tree The tree TreePanel, with config etc.
54701  */
54702 Roo.TreePanel = function(config){
54703     var el = config.el;
54704     var tree = config.tree;
54705     delete config.tree; 
54706     delete config.el; // hopefull!
54707     
54708     // wrapper for IE7 strict & safari scroll issue
54709     
54710     var treeEl = el.createChild();
54711     config.resizeEl = treeEl;
54712     
54713     
54714     
54715     Roo.TreePanel.superclass.constructor.call(this, el, config);
54716  
54717  
54718     this.tree = new Roo.tree.TreePanel(treeEl , tree);
54719     //console.log(tree);
54720     this.on('activate', function()
54721     {
54722         if (this.tree.rendered) {
54723             return;
54724         }
54725         //console.log('render tree');
54726         this.tree.render();
54727     });
54728     // this should not be needed.. - it's actually the 'el' that resizes?
54729     // actuall it breaks the containerScroll - dragging nodes auto scroll at top
54730     
54731     //this.on('resize',  function (cp, w, h) {
54732     //        this.tree.innerCt.setWidth(w);
54733     //        this.tree.innerCt.setHeight(h);
54734     //        //this.tree.innerCt.setStyle('overflow-y', 'auto');
54735     //});
54736
54737         
54738     
54739 };
54740
54741 Roo.extend(Roo.TreePanel, Roo.ContentPanel, {   
54742     fitToFrame : true,
54743     autoScroll : true
54744 });
54745
54746
54747
54748
54749
54750
54751
54752
54753
54754
54755
54756 /*
54757  * Based on:
54758  * Ext JS Library 1.1.1
54759  * Copyright(c) 2006-2007, Ext JS, LLC.
54760  *
54761  * Originally Released Under LGPL - original licence link has changed is not relivant.
54762  *
54763  * Fork - LGPL
54764  * <script type="text/javascript">
54765  */
54766  
54767
54768 /**
54769  * @class Roo.ReaderLayout
54770  * @extends Roo.BorderLayout
54771  * This is a pre-built layout that represents a classic, 5-pane application.  It consists of a header, a primary
54772  * center region containing two nested regions (a top one for a list view and one for item preview below),
54773  * and regions on either side that can be used for navigation, application commands, informational displays, etc.
54774  * The setup and configuration work exactly the same as it does for a {@link Roo.BorderLayout} - this class simply
54775  * expedites the setup of the overall layout and regions for this common application style.
54776  * Example:
54777  <pre><code>
54778 var reader = new Roo.ReaderLayout();
54779 var CP = Roo.ContentPanel;  // shortcut for adding
54780
54781 reader.beginUpdate();
54782 reader.add("north", new CP("north", "North"));
54783 reader.add("west", new CP("west", {title: "West"}));
54784 reader.add("east", new CP("east", {title: "East"}));
54785
54786 reader.regions.listView.add(new CP("listView", "List"));
54787 reader.regions.preview.add(new CP("preview", "Preview"));
54788 reader.endUpdate();
54789 </code></pre>
54790 * @constructor
54791 * Create a new ReaderLayout
54792 * @param {Object} config Configuration options
54793 * @param {String/HTMLElement/Element} container (optional) The container this layout is bound to (defaults to
54794 * document.body if omitted)
54795 */
54796 Roo.ReaderLayout = function(config, renderTo){
54797     var c = config || {size:{}};
54798     Roo.ReaderLayout.superclass.constructor.call(this, renderTo || document.body, {
54799         north: c.north !== false ? Roo.apply({
54800             split:false,
54801             initialSize: 32,
54802             titlebar: false
54803         }, c.north) : false,
54804         west: c.west !== false ? Roo.apply({
54805             split:true,
54806             initialSize: 200,
54807             minSize: 175,
54808             maxSize: 400,
54809             titlebar: true,
54810             collapsible: true,
54811             animate: true,
54812             margins:{left:5,right:0,bottom:5,top:5},
54813             cmargins:{left:5,right:5,bottom:5,top:5}
54814         }, c.west) : false,
54815         east: c.east !== false ? Roo.apply({
54816             split:true,
54817             initialSize: 200,
54818             minSize: 175,
54819             maxSize: 400,
54820             titlebar: true,
54821             collapsible: true,
54822             animate: true,
54823             margins:{left:0,right:5,bottom:5,top:5},
54824             cmargins:{left:5,right:5,bottom:5,top:5}
54825         }, c.east) : false,
54826         center: Roo.apply({
54827             tabPosition: 'top',
54828             autoScroll:false,
54829             closeOnTab: true,
54830             titlebar:false,
54831             margins:{left:c.west!==false ? 0 : 5,right:c.east!==false ? 0 : 5,bottom:5,top:2}
54832         }, c.center)
54833     });
54834
54835     this.el.addClass('x-reader');
54836
54837     this.beginUpdate();
54838
54839     var inner = new Roo.BorderLayout(Roo.get(document.body).createChild(), {
54840         south: c.preview !== false ? Roo.apply({
54841             split:true,
54842             initialSize: 200,
54843             minSize: 100,
54844             autoScroll:true,
54845             collapsible:true,
54846             titlebar: true,
54847             cmargins:{top:5,left:0, right:0, bottom:0}
54848         }, c.preview) : false,
54849         center: Roo.apply({
54850             autoScroll:false,
54851             titlebar:false,
54852             minHeight:200
54853         }, c.listView)
54854     });
54855     this.add('center', new Roo.NestedLayoutPanel(inner,
54856             Roo.apply({title: c.mainTitle || '',tabTip:''},c.innerPanelCfg)));
54857
54858     this.endUpdate();
54859
54860     this.regions.preview = inner.getRegion('south');
54861     this.regions.listView = inner.getRegion('center');
54862 };
54863
54864 Roo.extend(Roo.ReaderLayout, Roo.BorderLayout);/*
54865  * Based on:
54866  * Ext JS Library 1.1.1
54867  * Copyright(c) 2006-2007, Ext JS, LLC.
54868  *
54869  * Originally Released Under LGPL - original licence link has changed is not relivant.
54870  *
54871  * Fork - LGPL
54872  * <script type="text/javascript">
54873  */
54874  
54875 /**
54876  * @class Roo.grid.Grid
54877  * @extends Roo.util.Observable
54878  * This class represents the primary interface of a component based grid control.
54879  * <br><br>Usage:<pre><code>
54880  var grid = new Roo.grid.Grid("my-container-id", {
54881      ds: myDataStore,
54882      cm: myColModel,
54883      selModel: mySelectionModel,
54884      autoSizeColumns: true,
54885      monitorWindowResize: false,
54886      trackMouseOver: true
54887  });
54888  // set any options
54889  grid.render();
54890  * </code></pre>
54891  * <b>Common Problems:</b><br/>
54892  * - Grid does not resize properly when going smaller: Setting overflow hidden on the container
54893  * element will correct this<br/>
54894  * - If you get el.style[camel]= NaNpx or -2px or something related, be certain you have given your container element
54895  * dimensions. The grid adapts to your container's size, if your container has no size defined then the results
54896  * are unpredictable.<br/>
54897  * - Do not render the grid into an element with display:none. Try using visibility:hidden. Otherwise there is no way for the
54898  * grid to calculate dimensions/offsets.<br/>
54899   * @constructor
54900  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
54901  * The container MUST have some type of size defined for the grid to fill. The container will be
54902  * automatically set to position relative if it isn't already.
54903  * @param {Object} config A config object that sets properties on this grid.
54904  */
54905 Roo.grid.Grid = function(container, config){
54906         // initialize the container
54907         this.container = Roo.get(container);
54908         this.container.update("");
54909         this.container.setStyle("overflow", "hidden");
54910     this.container.addClass('x-grid-container');
54911
54912     this.id = this.container.id;
54913
54914     Roo.apply(this, config);
54915     // check and correct shorthanded configs
54916     if(this.ds){
54917         this.dataSource = this.ds;
54918         delete this.ds;
54919     }
54920     if(this.cm){
54921         this.colModel = this.cm;
54922         delete this.cm;
54923     }
54924     if(this.sm){
54925         this.selModel = this.sm;
54926         delete this.sm;
54927     }
54928
54929     if (this.selModel) {
54930         this.selModel = Roo.factory(this.selModel, Roo.grid);
54931         this.sm = this.selModel;
54932         this.sm.xmodule = this.xmodule || false;
54933     }
54934     if (typeof(this.colModel.config) == 'undefined') {
54935         this.colModel = new Roo.grid.ColumnModel(this.colModel);
54936         this.cm = this.colModel;
54937         this.cm.xmodule = this.xmodule || false;
54938     }
54939     if (this.dataSource) {
54940         this.dataSource= Roo.factory(this.dataSource, Roo.data);
54941         this.ds = this.dataSource;
54942         this.ds.xmodule = this.xmodule || false;
54943          
54944     }
54945     
54946     
54947     
54948     if(this.width){
54949         this.container.setWidth(this.width);
54950     }
54951
54952     if(this.height){
54953         this.container.setHeight(this.height);
54954     }
54955     /** @private */
54956         this.addEvents({
54957         // raw events
54958         /**
54959          * @event click
54960          * The raw click event for the entire grid.
54961          * @param {Roo.EventObject} e
54962          */
54963         "click" : true,
54964         /**
54965          * @event dblclick
54966          * The raw dblclick event for the entire grid.
54967          * @param {Roo.EventObject} e
54968          */
54969         "dblclick" : true,
54970         /**
54971          * @event contextmenu
54972          * The raw contextmenu event for the entire grid.
54973          * @param {Roo.EventObject} e
54974          */
54975         "contextmenu" : true,
54976         /**
54977          * @event mousedown
54978          * The raw mousedown event for the entire grid.
54979          * @param {Roo.EventObject} e
54980          */
54981         "mousedown" : true,
54982         /**
54983          * @event mouseup
54984          * The raw mouseup event for the entire grid.
54985          * @param {Roo.EventObject} e
54986          */
54987         "mouseup" : true,
54988         /**
54989          * @event mouseover
54990          * The raw mouseover event for the entire grid.
54991          * @param {Roo.EventObject} e
54992          */
54993         "mouseover" : true,
54994         /**
54995          * @event mouseout
54996          * The raw mouseout event for the entire grid.
54997          * @param {Roo.EventObject} e
54998          */
54999         "mouseout" : true,
55000         /**
55001          * @event keypress
55002          * The raw keypress event for the entire grid.
55003          * @param {Roo.EventObject} e
55004          */
55005         "keypress" : true,
55006         /**
55007          * @event keydown
55008          * The raw keydown event for the entire grid.
55009          * @param {Roo.EventObject} e
55010          */
55011         "keydown" : true,
55012
55013         // custom events
55014
55015         /**
55016          * @event cellclick
55017          * Fires when a cell is clicked
55018          * @param {Grid} this
55019          * @param {Number} rowIndex
55020          * @param {Number} columnIndex
55021          * @param {Roo.EventObject} e
55022          */
55023         "cellclick" : true,
55024         /**
55025          * @event celldblclick
55026          * Fires when a cell is double clicked
55027          * @param {Grid} this
55028          * @param {Number} rowIndex
55029          * @param {Number} columnIndex
55030          * @param {Roo.EventObject} e
55031          */
55032         "celldblclick" : true,
55033         /**
55034          * @event rowclick
55035          * Fires when a row is clicked
55036          * @param {Grid} this
55037          * @param {Number} rowIndex
55038          * @param {Roo.EventObject} e
55039          */
55040         "rowclick" : true,
55041         /**
55042          * @event rowdblclick
55043          * Fires when a row is double clicked
55044          * @param {Grid} this
55045          * @param {Number} rowIndex
55046          * @param {Roo.EventObject} e
55047          */
55048         "rowdblclick" : true,
55049         /**
55050          * @event headerclick
55051          * Fires when a header is clicked
55052          * @param {Grid} this
55053          * @param {Number} columnIndex
55054          * @param {Roo.EventObject} e
55055          */
55056         "headerclick" : true,
55057         /**
55058          * @event headerdblclick
55059          * Fires when a header cell is double clicked
55060          * @param {Grid} this
55061          * @param {Number} columnIndex
55062          * @param {Roo.EventObject} e
55063          */
55064         "headerdblclick" : true,
55065         /**
55066          * @event rowcontextmenu
55067          * Fires when a row is right clicked
55068          * @param {Grid} this
55069          * @param {Number} rowIndex
55070          * @param {Roo.EventObject} e
55071          */
55072         "rowcontextmenu" : true,
55073         /**
55074          * @event cellcontextmenu
55075          * Fires when a cell is right clicked
55076          * @param {Grid} this
55077          * @param {Number} rowIndex
55078          * @param {Number} cellIndex
55079          * @param {Roo.EventObject} e
55080          */
55081          "cellcontextmenu" : true,
55082         /**
55083          * @event headercontextmenu
55084          * Fires when a header is right clicked
55085          * @param {Grid} this
55086          * @param {Number} columnIndex
55087          * @param {Roo.EventObject} e
55088          */
55089         "headercontextmenu" : true,
55090         /**
55091          * @event bodyscroll
55092          * Fires when the body element is scrolled
55093          * @param {Number} scrollLeft
55094          * @param {Number} scrollTop
55095          */
55096         "bodyscroll" : true,
55097         /**
55098          * @event columnresize
55099          * Fires when the user resizes a column
55100          * @param {Number} columnIndex
55101          * @param {Number} newSize
55102          */
55103         "columnresize" : true,
55104         /**
55105          * @event columnmove
55106          * Fires when the user moves a column
55107          * @param {Number} oldIndex
55108          * @param {Number} newIndex
55109          */
55110         "columnmove" : true,
55111         /**
55112          * @event startdrag
55113          * Fires when row(s) start being dragged
55114          * @param {Grid} this
55115          * @param {Roo.GridDD} dd The drag drop object
55116          * @param {event} e The raw browser event
55117          */
55118         "startdrag" : true,
55119         /**
55120          * @event enddrag
55121          * Fires when a drag operation is complete
55122          * @param {Grid} this
55123          * @param {Roo.GridDD} dd The drag drop object
55124          * @param {event} e The raw browser event
55125          */
55126         "enddrag" : true,
55127         /**
55128          * @event dragdrop
55129          * Fires when dragged row(s) are dropped on a valid DD target
55130          * @param {Grid} this
55131          * @param {Roo.GridDD} dd The drag drop object
55132          * @param {String} targetId The target drag drop object
55133          * @param {event} e The raw browser event
55134          */
55135         "dragdrop" : true,
55136         /**
55137          * @event dragover
55138          * Fires while row(s) are being dragged. "targetId" is the id of the Yahoo.util.DD object the selected rows are being dragged over.
55139          * @param {Grid} this
55140          * @param {Roo.GridDD} dd The drag drop object
55141          * @param {String} targetId The target drag drop object
55142          * @param {event} e The raw browser event
55143          */
55144         "dragover" : true,
55145         /**
55146          * @event dragenter
55147          *  Fires when the dragged row(s) first cross another DD target while being dragged
55148          * @param {Grid} this
55149          * @param {Roo.GridDD} dd The drag drop object
55150          * @param {String} targetId The target drag drop object
55151          * @param {event} e The raw browser event
55152          */
55153         "dragenter" : true,
55154         /**
55155          * @event dragout
55156          * Fires when the dragged row(s) leave another DD target while being dragged
55157          * @param {Grid} this
55158          * @param {Roo.GridDD} dd The drag drop object
55159          * @param {String} targetId The target drag drop object
55160          * @param {event} e The raw browser event
55161          */
55162         "dragout" : true,
55163         /**
55164          * @event rowclass
55165          * Fires when a row is rendered, so you can change add a style to it.
55166          * @param {GridView} gridview   The grid view
55167          * @param {Object} rowcfg   contains record  rowIndex and rowClass - set rowClass to add a style.
55168          */
55169         'rowclass' : true,
55170
55171         /**
55172          * @event render
55173          * Fires when the grid is rendered
55174          * @param {Grid} grid
55175          */
55176         'render' : true
55177     });
55178
55179     Roo.grid.Grid.superclass.constructor.call(this);
55180 };
55181 Roo.extend(Roo.grid.Grid, Roo.util.Observable, {
55182     
55183     /**
55184      * @cfg {String} ddGroup - drag drop group.
55185      */
55186
55187     /**
55188      * @cfg {Number} minColumnWidth The minimum width a column can be resized to. Default is 25.
55189      */
55190     minColumnWidth : 25,
55191
55192     /**
55193      * @cfg {Boolean} autoSizeColumns True to automatically resize the columns to fit their content
55194      * <b>on initial render.</b> It is more efficient to explicitly size the columns
55195      * through the ColumnModel's {@link Roo.grid.ColumnModel#width} config option.  Default is false.
55196      */
55197     autoSizeColumns : false,
55198
55199     /**
55200      * @cfg {Boolean} autoSizeHeaders True to measure headers with column data when auto sizing columns. Default is true.
55201      */
55202     autoSizeHeaders : true,
55203
55204     /**
55205      * @cfg {Boolean} monitorWindowResize True to autoSize the grid when the window resizes. Default is true.
55206      */
55207     monitorWindowResize : true,
55208
55209     /**
55210      * @cfg {Boolean} maxRowsToMeasure If autoSizeColumns is on, maxRowsToMeasure can be used to limit the number of
55211      * rows measured to get a columns size. Default is 0 (all rows).
55212      */
55213     maxRowsToMeasure : 0,
55214
55215     /**
55216      * @cfg {Boolean} trackMouseOver True to highlight rows when the mouse is over. Default is true.
55217      */
55218     trackMouseOver : true,
55219
55220     /**
55221     * @cfg {Boolean} enableDrag  True to enable drag of rows. Default is false. (double check if this is needed?)
55222     */
55223     
55224     /**
55225     * @cfg {Boolean} enableDragDrop True to enable drag and drop of rows. Default is false.
55226     */
55227     enableDragDrop : false,
55228     
55229     /**
55230     * @cfg {Boolean} enableColumnMove True to enable drag and drop reorder of columns. Default is true.
55231     */
55232     enableColumnMove : true,
55233     
55234     /**
55235     * @cfg {Boolean} enableColumnHide True to enable hiding of columns with the header context menu. Default is true.
55236     */
55237     enableColumnHide : true,
55238     
55239     /**
55240     * @cfg {Boolean} enableRowHeightSync True to manually sync row heights across locked and not locked rows. Default is false.
55241     */
55242     enableRowHeightSync : false,
55243     
55244     /**
55245     * @cfg {Boolean} stripeRows True to stripe the rows.  Default is true.
55246     */
55247     stripeRows : true,
55248     
55249     /**
55250     * @cfg {Boolean} autoHeight True to fit the height of the grid container to the height of the data. Default is false.
55251     */
55252     autoHeight : false,
55253
55254     /**
55255      * @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.
55256      */
55257     autoExpandColumn : false,
55258
55259     /**
55260     * @cfg {Number} autoExpandMin The minimum width the autoExpandColumn can have (if enabled).
55261     * Default is 50.
55262     */
55263     autoExpandMin : 50,
55264
55265     /**
55266     * @cfg {Number} autoExpandMax The maximum width the autoExpandColumn can have (if enabled). Default is 1000.
55267     */
55268     autoExpandMax : 1000,
55269
55270     /**
55271     * @cfg {Object} view The {@link Roo.grid.GridView} used by the grid. This can be set before a call to render().
55272     */
55273     view : null,
55274
55275     /**
55276     * @cfg {Object} loadMask An {@link Roo.LoadMask} config or true to mask the grid while loading. Default is false.
55277     */
55278     loadMask : false,
55279     /**
55280     * @cfg {Roo.dd.DropTarget} dropTarget An {@link Roo.dd.DropTarget} config
55281     */
55282     dropTarget: false,
55283     
55284    
55285     
55286     // private
55287     rendered : false,
55288
55289     /**
55290     * @cfg {Boolean} autoWidth True to set the grid's width to the default total width of the grid's columns instead
55291     * of a fixed width. Default is false.
55292     */
55293     /**
55294     * @cfg {Number} maxHeight Sets the maximum height of the grid - ignored if autoHeight is not on.
55295     */
55296     /**
55297      * Called once after all setup has been completed and the grid is ready to be rendered.
55298      * @return {Roo.grid.Grid} this
55299      */
55300     render : function()
55301     {
55302         var c = this.container;
55303         // try to detect autoHeight/width mode
55304         if((!c.dom.offsetHeight || c.dom.offsetHeight < 20) || c.getStyle("height") == "auto"){
55305             this.autoHeight = true;
55306         }
55307         var view = this.getView();
55308         view.init(this);
55309
55310         c.on("click", this.onClick, this);
55311         c.on("dblclick", this.onDblClick, this);
55312         c.on("contextmenu", this.onContextMenu, this);
55313         c.on("keydown", this.onKeyDown, this);
55314         if (Roo.isTouch) {
55315             c.on("touchstart", this.onTouchStart, this);
55316         }
55317
55318         this.relayEvents(c, ["mousedown","mouseup","mouseover","mouseout","keypress"]);
55319
55320         this.getSelectionModel().init(this);
55321
55322         view.render();
55323
55324         if(this.loadMask){
55325             this.loadMask = new Roo.LoadMask(this.container,
55326                     Roo.apply({store:this.dataSource}, this.loadMask));
55327         }
55328         
55329         
55330         if (this.toolbar && this.toolbar.xtype) {
55331             this.toolbar.container = this.getView().getHeaderPanel(true);
55332             this.toolbar = new Roo.Toolbar(this.toolbar);
55333         }
55334         if (this.footer && this.footer.xtype) {
55335             this.footer.dataSource = this.getDataSource();
55336             this.footer.container = this.getView().getFooterPanel(true);
55337             this.footer = Roo.factory(this.footer, Roo);
55338         }
55339         if (this.dropTarget && this.dropTarget.xtype) {
55340             delete this.dropTarget.xtype;
55341             this.dropTarget =  new Roo.dd.DropTarget(this.getView().mainBody, this.dropTarget);
55342         }
55343         
55344         
55345         this.rendered = true;
55346         this.fireEvent('render', this);
55347         return this;
55348     },
55349
55350         /**
55351          * Reconfigures the grid to use a different Store and Column Model.
55352          * The View will be bound to the new objects and refreshed.
55353          * @param {Roo.data.Store} dataSource The new {@link Roo.data.Store} object
55354          * @param {Roo.grid.ColumnModel} The new {@link Roo.grid.ColumnModel} object
55355          */
55356     reconfigure : function(dataSource, colModel){
55357         if(this.loadMask){
55358             this.loadMask.destroy();
55359             this.loadMask = new Roo.LoadMask(this.container,
55360                     Roo.apply({store:dataSource}, this.loadMask));
55361         }
55362         this.view.bind(dataSource, colModel);
55363         this.dataSource = dataSource;
55364         this.colModel = colModel;
55365         this.view.refresh(true);
55366     },
55367
55368     // private
55369     onKeyDown : function(e){
55370         this.fireEvent("keydown", e);
55371     },
55372
55373     /**
55374      * Destroy this grid.
55375      * @param {Boolean} removeEl True to remove the element
55376      */
55377     destroy : function(removeEl, keepListeners){
55378         if(this.loadMask){
55379             this.loadMask.destroy();
55380         }
55381         var c = this.container;
55382         c.removeAllListeners();
55383         this.view.destroy();
55384         this.colModel.purgeListeners();
55385         if(!keepListeners){
55386             this.purgeListeners();
55387         }
55388         c.update("");
55389         if(removeEl === true){
55390             c.remove();
55391         }
55392     },
55393
55394     // private
55395     processEvent : function(name, e){
55396         // does this fire select???
55397         //Roo.log('grid:processEvent '  + name);
55398         
55399         if (name != 'touchstart' ) {
55400             this.fireEvent(name, e);    
55401         }
55402         
55403         var t = e.getTarget();
55404         var v = this.view;
55405         var header = v.findHeaderIndex(t);
55406         if(header !== false){
55407             var ename = name == 'touchstart' ? 'click' : name;
55408              
55409             this.fireEvent("header" + ename, this, header, e);
55410         }else{
55411             var row = v.findRowIndex(t);
55412             var cell = v.findCellIndex(t);
55413             if (name == 'touchstart') {
55414                 // first touch is always a click.
55415                 // hopefull this happens after selection is updated.?
55416                 name = false;
55417                 
55418                 if (typeof(this.selModel.getSelectedCell) != 'undefined') {
55419                     var cs = this.selModel.getSelectedCell();
55420                     if (row == cs[0] && cell == cs[1]){
55421                         name = 'dblclick';
55422                     }
55423                 }
55424                 if (typeof(this.selModel.getSelections) != 'undefined') {
55425                     var cs = this.selModel.getSelections();
55426                     var ds = this.dataSource;
55427                     if (cs.length == 1 && ds.getAt(row) == cs[0]){
55428                         name = 'dblclick';
55429                     }
55430                 }
55431                 if (!name) {
55432                     return;
55433                 }
55434             }
55435             
55436             
55437             if(row !== false){
55438                 this.fireEvent("row" + name, this, row, e);
55439                 if(cell !== false){
55440                     this.fireEvent("cell" + name, this, row, cell, e);
55441                 }
55442             }
55443         }
55444     },
55445
55446     // private
55447     onClick : function(e){
55448         this.processEvent("click", e);
55449     },
55450    // private
55451     onTouchStart : function(e){
55452         this.processEvent("touchstart", e);
55453     },
55454
55455     // private
55456     onContextMenu : function(e, t){
55457         this.processEvent("contextmenu", e);
55458     },
55459
55460     // private
55461     onDblClick : function(e){
55462         this.processEvent("dblclick", e);
55463     },
55464
55465     // private
55466     walkCells : function(row, col, step, fn, scope){
55467         var cm = this.colModel, clen = cm.getColumnCount();
55468         var ds = this.dataSource, rlen = ds.getCount(), first = true;
55469         if(step < 0){
55470             if(col < 0){
55471                 row--;
55472                 first = false;
55473             }
55474             while(row >= 0){
55475                 if(!first){
55476                     col = clen-1;
55477                 }
55478                 first = false;
55479                 while(col >= 0){
55480                     if(fn.call(scope || this, row, col, cm) === true){
55481                         return [row, col];
55482                     }
55483                     col--;
55484                 }
55485                 row--;
55486             }
55487         } else {
55488             if(col >= clen){
55489                 row++;
55490                 first = false;
55491             }
55492             while(row < rlen){
55493                 if(!first){
55494                     col = 0;
55495                 }
55496                 first = false;
55497                 while(col < clen){
55498                     if(fn.call(scope || this, row, col, cm) === true){
55499                         return [row, col];
55500                     }
55501                     col++;
55502                 }
55503                 row++;
55504             }
55505         }
55506         return null;
55507     },
55508
55509     // private
55510     getSelections : function(){
55511         return this.selModel.getSelections();
55512     },
55513
55514     /**
55515      * Causes the grid to manually recalculate its dimensions. Generally this is done automatically,
55516      * but if manual update is required this method will initiate it.
55517      */
55518     autoSize : function(){
55519         if(this.rendered){
55520             this.view.layout();
55521             if(this.view.adjustForScroll){
55522                 this.view.adjustForScroll();
55523             }
55524         }
55525     },
55526
55527     /**
55528      * Returns the grid's underlying element.
55529      * @return {Element} The element
55530      */
55531     getGridEl : function(){
55532         return this.container;
55533     },
55534
55535     // private for compatibility, overridden by editor grid
55536     stopEditing : function(){},
55537
55538     /**
55539      * Returns the grid's SelectionModel.
55540      * @return {SelectionModel}
55541      */
55542     getSelectionModel : function(){
55543         if(!this.selModel){
55544             this.selModel = new Roo.grid.RowSelectionModel();
55545         }
55546         return this.selModel;
55547     },
55548
55549     /**
55550      * Returns the grid's DataSource.
55551      * @return {DataSource}
55552      */
55553     getDataSource : function(){
55554         return this.dataSource;
55555     },
55556
55557     /**
55558      * Returns the grid's ColumnModel.
55559      * @return {ColumnModel}
55560      */
55561     getColumnModel : function(){
55562         return this.colModel;
55563     },
55564
55565     /**
55566      * Returns the grid's GridView object.
55567      * @return {GridView}
55568      */
55569     getView : function(){
55570         if(!this.view){
55571             this.view = new Roo.grid.GridView(this.viewConfig);
55572         }
55573         return this.view;
55574     },
55575     /**
55576      * Called to get grid's drag proxy text, by default returns this.ddText.
55577      * @return {String}
55578      */
55579     getDragDropText : function(){
55580         var count = this.selModel.getCount();
55581         return String.format(this.ddText, count, count == 1 ? '' : 's');
55582     }
55583 });
55584 /**
55585  * Configures the text is the drag proxy (defaults to "%0 selected row(s)").
55586  * %0 is replaced with the number of selected rows.
55587  * @type String
55588  */
55589 Roo.grid.Grid.prototype.ddText = "{0} selected row{1}";/*
55590  * Based on:
55591  * Ext JS Library 1.1.1
55592  * Copyright(c) 2006-2007, Ext JS, LLC.
55593  *
55594  * Originally Released Under LGPL - original licence link has changed is not relivant.
55595  *
55596  * Fork - LGPL
55597  * <script type="text/javascript">
55598  */
55599  
55600 Roo.grid.AbstractGridView = function(){
55601         this.grid = null;
55602         
55603         this.events = {
55604             "beforerowremoved" : true,
55605             "beforerowsinserted" : true,
55606             "beforerefresh" : true,
55607             "rowremoved" : true,
55608             "rowsinserted" : true,
55609             "rowupdated" : true,
55610             "refresh" : true
55611         };
55612     Roo.grid.AbstractGridView.superclass.constructor.call(this);
55613 };
55614
55615 Roo.extend(Roo.grid.AbstractGridView, Roo.util.Observable, {
55616     rowClass : "x-grid-row",
55617     cellClass : "x-grid-cell",
55618     tdClass : "x-grid-td",
55619     hdClass : "x-grid-hd",
55620     splitClass : "x-grid-hd-split",
55621     
55622     init: function(grid){
55623         this.grid = grid;
55624                 var cid = this.grid.getGridEl().id;
55625         this.colSelector = "#" + cid + " ." + this.cellClass + "-";
55626         this.tdSelector = "#" + cid + " ." + this.tdClass + "-";
55627         this.hdSelector = "#" + cid + " ." + this.hdClass + "-";
55628         this.splitSelector = "#" + cid + " ." + this.splitClass + "-";
55629         },
55630         
55631     getColumnRenderers : function(){
55632         var renderers = [];
55633         var cm = this.grid.colModel;
55634         var colCount = cm.getColumnCount();
55635         for(var i = 0; i < colCount; i++){
55636             renderers[i] = cm.getRenderer(i);
55637         }
55638         return renderers;
55639     },
55640     
55641     getColumnIds : function(){
55642         var ids = [];
55643         var cm = this.grid.colModel;
55644         var colCount = cm.getColumnCount();
55645         for(var i = 0; i < colCount; i++){
55646             ids[i] = cm.getColumnId(i);
55647         }
55648         return ids;
55649     },
55650     
55651     getDataIndexes : function(){
55652         if(!this.indexMap){
55653             this.indexMap = this.buildIndexMap();
55654         }
55655         return this.indexMap.colToData;
55656     },
55657     
55658     getColumnIndexByDataIndex : function(dataIndex){
55659         if(!this.indexMap){
55660             this.indexMap = this.buildIndexMap();
55661         }
55662         return this.indexMap.dataToCol[dataIndex];
55663     },
55664     
55665     /**
55666      * Set a css style for a column dynamically. 
55667      * @param {Number} colIndex The index of the column
55668      * @param {String} name The css property name
55669      * @param {String} value The css value
55670      */
55671     setCSSStyle : function(colIndex, name, value){
55672         var selector = "#" + this.grid.id + " .x-grid-col-" + colIndex;
55673         Roo.util.CSS.updateRule(selector, name, value);
55674     },
55675     
55676     generateRules : function(cm){
55677         var ruleBuf = [], rulesId = this.grid.id + '-cssrules';
55678         Roo.util.CSS.removeStyleSheet(rulesId);
55679         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
55680             var cid = cm.getColumnId(i);
55681             ruleBuf.push(this.colSelector, cid, " {\n", cm.config[i].css, "}\n",
55682                          this.tdSelector, cid, " {\n}\n",
55683                          this.hdSelector, cid, " {\n}\n",
55684                          this.splitSelector, cid, " {\n}\n");
55685         }
55686         return Roo.util.CSS.createStyleSheet(ruleBuf.join(""), rulesId);
55687     }
55688 });/*
55689  * Based on:
55690  * Ext JS Library 1.1.1
55691  * Copyright(c) 2006-2007, Ext JS, LLC.
55692  *
55693  * Originally Released Under LGPL - original licence link has changed is not relivant.
55694  *
55695  * Fork - LGPL
55696  * <script type="text/javascript">
55697  */
55698
55699 // private
55700 // This is a support class used internally by the Grid components
55701 Roo.grid.HeaderDragZone = function(grid, hd, hd2){
55702     this.grid = grid;
55703     this.view = grid.getView();
55704     this.ddGroup = "gridHeader" + this.grid.getGridEl().id;
55705     Roo.grid.HeaderDragZone.superclass.constructor.call(this, hd);
55706     if(hd2){
55707         this.setHandleElId(Roo.id(hd));
55708         this.setOuterHandleElId(Roo.id(hd2));
55709     }
55710     this.scroll = false;
55711 };
55712 Roo.extend(Roo.grid.HeaderDragZone, Roo.dd.DragZone, {
55713     maxDragWidth: 120,
55714     getDragData : function(e){
55715         var t = Roo.lib.Event.getTarget(e);
55716         var h = this.view.findHeaderCell(t);
55717         if(h){
55718             return {ddel: h.firstChild, header:h};
55719         }
55720         return false;
55721     },
55722
55723     onInitDrag : function(e){
55724         this.view.headersDisabled = true;
55725         var clone = this.dragData.ddel.cloneNode(true);
55726         clone.id = Roo.id();
55727         clone.style.width = Math.min(this.dragData.header.offsetWidth,this.maxDragWidth) + "px";
55728         this.proxy.update(clone);
55729         return true;
55730     },
55731
55732     afterValidDrop : function(){
55733         var v = this.view;
55734         setTimeout(function(){
55735             v.headersDisabled = false;
55736         }, 50);
55737     },
55738
55739     afterInvalidDrop : function(){
55740         var v = this.view;
55741         setTimeout(function(){
55742             v.headersDisabled = false;
55743         }, 50);
55744     }
55745 });
55746 /*
55747  * Based on:
55748  * Ext JS Library 1.1.1
55749  * Copyright(c) 2006-2007, Ext JS, LLC.
55750  *
55751  * Originally Released Under LGPL - original licence link has changed is not relivant.
55752  *
55753  * Fork - LGPL
55754  * <script type="text/javascript">
55755  */
55756 // private
55757 // This is a support class used internally by the Grid components
55758 Roo.grid.HeaderDropZone = function(grid, hd, hd2){
55759     this.grid = grid;
55760     this.view = grid.getView();
55761     // split the proxies so they don't interfere with mouse events
55762     this.proxyTop = Roo.DomHelper.append(document.body, {
55763         cls:"col-move-top", html:"&#160;"
55764     }, true);
55765     this.proxyBottom = Roo.DomHelper.append(document.body, {
55766         cls:"col-move-bottom", html:"&#160;"
55767     }, true);
55768     this.proxyTop.hide = this.proxyBottom.hide = function(){
55769         this.setLeftTop(-100,-100);
55770         this.setStyle("visibility", "hidden");
55771     };
55772     this.ddGroup = "gridHeader" + this.grid.getGridEl().id;
55773     // temporarily disabled
55774     //Roo.dd.ScrollManager.register(this.view.scroller.dom);
55775     Roo.grid.HeaderDropZone.superclass.constructor.call(this, grid.getGridEl().dom);
55776 };
55777 Roo.extend(Roo.grid.HeaderDropZone, Roo.dd.DropZone, {
55778     proxyOffsets : [-4, -9],
55779     fly: Roo.Element.fly,
55780
55781     getTargetFromEvent : function(e){
55782         var t = Roo.lib.Event.getTarget(e);
55783         var cindex = this.view.findCellIndex(t);
55784         if(cindex !== false){
55785             return this.view.getHeaderCell(cindex);
55786         }
55787         return null;
55788     },
55789
55790     nextVisible : function(h){
55791         var v = this.view, cm = this.grid.colModel;
55792         h = h.nextSibling;
55793         while(h){
55794             if(!cm.isHidden(v.getCellIndex(h))){
55795                 return h;
55796             }
55797             h = h.nextSibling;
55798         }
55799         return null;
55800     },
55801
55802     prevVisible : function(h){
55803         var v = this.view, cm = this.grid.colModel;
55804         h = h.prevSibling;
55805         while(h){
55806             if(!cm.isHidden(v.getCellIndex(h))){
55807                 return h;
55808             }
55809             h = h.prevSibling;
55810         }
55811         return null;
55812     },
55813
55814     positionIndicator : function(h, n, e){
55815         var x = Roo.lib.Event.getPageX(e);
55816         var r = Roo.lib.Dom.getRegion(n.firstChild);
55817         var px, pt, py = r.top + this.proxyOffsets[1];
55818         if((r.right - x) <= (r.right-r.left)/2){
55819             px = r.right+this.view.borderWidth;
55820             pt = "after";
55821         }else{
55822             px = r.left;
55823             pt = "before";
55824         }
55825         var oldIndex = this.view.getCellIndex(h);
55826         var newIndex = this.view.getCellIndex(n);
55827
55828         if(this.grid.colModel.isFixed(newIndex)){
55829             return false;
55830         }
55831
55832         var locked = this.grid.colModel.isLocked(newIndex);
55833
55834         if(pt == "after"){
55835             newIndex++;
55836         }
55837         if(oldIndex < newIndex){
55838             newIndex--;
55839         }
55840         if(oldIndex == newIndex && (locked == this.grid.colModel.isLocked(oldIndex))){
55841             return false;
55842         }
55843         px +=  this.proxyOffsets[0];
55844         this.proxyTop.setLeftTop(px, py);
55845         this.proxyTop.show();
55846         if(!this.bottomOffset){
55847             this.bottomOffset = this.view.mainHd.getHeight();
55848         }
55849         this.proxyBottom.setLeftTop(px, py+this.proxyTop.dom.offsetHeight+this.bottomOffset);
55850         this.proxyBottom.show();
55851         return pt;
55852     },
55853
55854     onNodeEnter : function(n, dd, e, data){
55855         if(data.header != n){
55856             this.positionIndicator(data.header, n, e);
55857         }
55858     },
55859
55860     onNodeOver : function(n, dd, e, data){
55861         var result = false;
55862         if(data.header != n){
55863             result = this.positionIndicator(data.header, n, e);
55864         }
55865         if(!result){
55866             this.proxyTop.hide();
55867             this.proxyBottom.hide();
55868         }
55869         return result ? this.dropAllowed : this.dropNotAllowed;
55870     },
55871
55872     onNodeOut : function(n, dd, e, data){
55873         this.proxyTop.hide();
55874         this.proxyBottom.hide();
55875     },
55876
55877     onNodeDrop : function(n, dd, e, data){
55878         var h = data.header;
55879         if(h != n){
55880             var cm = this.grid.colModel;
55881             var x = Roo.lib.Event.getPageX(e);
55882             var r = Roo.lib.Dom.getRegion(n.firstChild);
55883             var pt = (r.right - x) <= ((r.right-r.left)/2) ? "after" : "before";
55884             var oldIndex = this.view.getCellIndex(h);
55885             var newIndex = this.view.getCellIndex(n);
55886             var locked = cm.isLocked(newIndex);
55887             if(pt == "after"){
55888                 newIndex++;
55889             }
55890             if(oldIndex < newIndex){
55891                 newIndex--;
55892             }
55893             if(oldIndex == newIndex && (locked == cm.isLocked(oldIndex))){
55894                 return false;
55895             }
55896             cm.setLocked(oldIndex, locked, true);
55897             cm.moveColumn(oldIndex, newIndex);
55898             this.grid.fireEvent("columnmove", oldIndex, newIndex);
55899             return true;
55900         }
55901         return false;
55902     }
55903 });
55904 /*
55905  * Based on:
55906  * Ext JS Library 1.1.1
55907  * Copyright(c) 2006-2007, Ext JS, LLC.
55908  *
55909  * Originally Released Under LGPL - original licence link has changed is not relivant.
55910  *
55911  * Fork - LGPL
55912  * <script type="text/javascript">
55913  */
55914   
55915 /**
55916  * @class Roo.grid.GridView
55917  * @extends Roo.util.Observable
55918  *
55919  * @constructor
55920  * @param {Object} config
55921  */
55922 Roo.grid.GridView = function(config){
55923     Roo.grid.GridView.superclass.constructor.call(this);
55924     this.el = null;
55925
55926     Roo.apply(this, config);
55927 };
55928
55929 Roo.extend(Roo.grid.GridView, Roo.grid.AbstractGridView, {
55930
55931     unselectable :  'unselectable="on"',
55932     unselectableCls :  'x-unselectable',
55933     
55934     
55935     rowClass : "x-grid-row",
55936
55937     cellClass : "x-grid-col",
55938
55939     tdClass : "x-grid-td",
55940
55941     hdClass : "x-grid-hd",
55942
55943     splitClass : "x-grid-split",
55944
55945     sortClasses : ["sort-asc", "sort-desc"],
55946
55947     enableMoveAnim : false,
55948
55949     hlColor: "C3DAF9",
55950
55951     dh : Roo.DomHelper,
55952
55953     fly : Roo.Element.fly,
55954
55955     css : Roo.util.CSS,
55956
55957     borderWidth: 1,
55958
55959     splitOffset: 3,
55960
55961     scrollIncrement : 22,
55962
55963     cellRE: /(?:.*?)x-grid-(?:hd|cell|csplit)-(?:[\d]+)-([\d]+)(?:.*?)/,
55964
55965     findRE: /\s?(?:x-grid-hd|x-grid-col|x-grid-csplit)\s/,
55966
55967     bind : function(ds, cm){
55968         if(this.ds){
55969             this.ds.un("load", this.onLoad, this);
55970             this.ds.un("datachanged", this.onDataChange, this);
55971             this.ds.un("add", this.onAdd, this);
55972             this.ds.un("remove", this.onRemove, this);
55973             this.ds.un("update", this.onUpdate, this);
55974             this.ds.un("clear", this.onClear, this);
55975         }
55976         if(ds){
55977             ds.on("load", this.onLoad, this);
55978             ds.on("datachanged", this.onDataChange, this);
55979             ds.on("add", this.onAdd, this);
55980             ds.on("remove", this.onRemove, this);
55981             ds.on("update", this.onUpdate, this);
55982             ds.on("clear", this.onClear, this);
55983         }
55984         this.ds = ds;
55985
55986         if(this.cm){
55987             this.cm.un("widthchange", this.onColWidthChange, this);
55988             this.cm.un("headerchange", this.onHeaderChange, this);
55989             this.cm.un("hiddenchange", this.onHiddenChange, this);
55990             this.cm.un("columnmoved", this.onColumnMove, this);
55991             this.cm.un("columnlockchange", this.onColumnLock, this);
55992         }
55993         if(cm){
55994             this.generateRules(cm);
55995             cm.on("widthchange", this.onColWidthChange, this);
55996             cm.on("headerchange", this.onHeaderChange, this);
55997             cm.on("hiddenchange", this.onHiddenChange, this);
55998             cm.on("columnmoved", this.onColumnMove, this);
55999             cm.on("columnlockchange", this.onColumnLock, this);
56000         }
56001         this.cm = cm;
56002     },
56003
56004     init: function(grid){
56005         Roo.grid.GridView.superclass.init.call(this, grid);
56006
56007         this.bind(grid.dataSource, grid.colModel);
56008
56009         grid.on("headerclick", this.handleHeaderClick, this);
56010
56011         if(grid.trackMouseOver){
56012             grid.on("mouseover", this.onRowOver, this);
56013             grid.on("mouseout", this.onRowOut, this);
56014         }
56015         grid.cancelTextSelection = function(){};
56016         this.gridId = grid.id;
56017
56018         var tpls = this.templates || {};
56019
56020         if(!tpls.master){
56021             tpls.master = new Roo.Template(
56022                '<div class="x-grid" hidefocus="true">',
56023                 '<a href="#" class="x-grid-focus" tabIndex="-1"></a>',
56024                   '<div class="x-grid-topbar"></div>',
56025                   '<div class="x-grid-scroller"><div></div></div>',
56026                   '<div class="x-grid-locked">',
56027                       '<div class="x-grid-header">{lockedHeader}</div>',
56028                       '<div class="x-grid-body">{lockedBody}</div>',
56029                   "</div>",
56030                   '<div class="x-grid-viewport">',
56031                       '<div class="x-grid-header">{header}</div>',
56032                       '<div class="x-grid-body">{body}</div>',
56033                   "</div>",
56034                   '<div class="x-grid-bottombar"></div>',
56035                  
56036                   '<div class="x-grid-resize-proxy">&#160;</div>',
56037                "</div>"
56038             );
56039             tpls.master.disableformats = true;
56040         }
56041
56042         if(!tpls.header){
56043             tpls.header = new Roo.Template(
56044                '<table border="0" cellspacing="0" cellpadding="0">',
56045                '<tbody><tr class="x-grid-hd-row">{cells}</tr></tbody>',
56046                "</table>{splits}"
56047             );
56048             tpls.header.disableformats = true;
56049         }
56050         tpls.header.compile();
56051
56052         if(!tpls.hcell){
56053             tpls.hcell = new Roo.Template(
56054                 '<td class="x-grid-hd x-grid-td-{id} {cellId}"><div title="{title}" class="x-grid-hd-inner x-grid-hd-{id}">',
56055                 '<div class="x-grid-hd-text ' + this.unselectableCls +  '" ' + this.unselectable +'>{value}<img class="x-grid-sort-icon" src="', Roo.BLANK_IMAGE_URL, '" /></div>',
56056                 "</div></td>"
56057              );
56058              tpls.hcell.disableFormats = true;
56059         }
56060         tpls.hcell.compile();
56061
56062         if(!tpls.hsplit){
56063             tpls.hsplit = new Roo.Template('<div class="x-grid-split {splitId} x-grid-split-{id}" style="{style} ' +
56064                                             this.unselectableCls +  '" ' + this.unselectable +'>&#160;</div>');
56065             tpls.hsplit.disableFormats = true;
56066         }
56067         tpls.hsplit.compile();
56068
56069         if(!tpls.body){
56070             tpls.body = new Roo.Template(
56071                '<table border="0" cellspacing="0" cellpadding="0">',
56072                "<tbody>{rows}</tbody>",
56073                "</table>"
56074             );
56075             tpls.body.disableFormats = true;
56076         }
56077         tpls.body.compile();
56078
56079         if(!tpls.row){
56080             tpls.row = new Roo.Template('<tr class="x-grid-row {alt}">{cells}</tr>');
56081             tpls.row.disableFormats = true;
56082         }
56083         tpls.row.compile();
56084
56085         if(!tpls.cell){
56086             tpls.cell = new Roo.Template(
56087                 '<td class="x-grid-col x-grid-td-{id} {cellId} {css}" tabIndex="0">',
56088                 '<div class="x-grid-col-{id} x-grid-cell-inner"><div class="x-grid-cell-text ' +
56089                     this.unselectableCls +  '" ' + this.unselectable +'" {attr}>{value}</div></div>',
56090                 "</td>"
56091             );
56092             tpls.cell.disableFormats = true;
56093         }
56094         tpls.cell.compile();
56095
56096         this.templates = tpls;
56097     },
56098
56099     // remap these for backwards compat
56100     onColWidthChange : function(){
56101         this.updateColumns.apply(this, arguments);
56102     },
56103     onHeaderChange : function(){
56104         this.updateHeaders.apply(this, arguments);
56105     }, 
56106     onHiddenChange : function(){
56107         this.handleHiddenChange.apply(this, arguments);
56108     },
56109     onColumnMove : function(){
56110         this.handleColumnMove.apply(this, arguments);
56111     },
56112     onColumnLock : function(){
56113         this.handleLockChange.apply(this, arguments);
56114     },
56115
56116     onDataChange : function(){
56117         this.refresh();
56118         this.updateHeaderSortState();
56119     },
56120
56121     onClear : function(){
56122         this.refresh();
56123     },
56124
56125     onUpdate : function(ds, record){
56126         this.refreshRow(record);
56127     },
56128
56129     refreshRow : function(record){
56130         var ds = this.ds, index;
56131         if(typeof record == 'number'){
56132             index = record;
56133             record = ds.getAt(index);
56134         }else{
56135             index = ds.indexOf(record);
56136         }
56137         this.insertRows(ds, index, index, true);
56138         this.onRemove(ds, record, index+1, true);
56139         this.syncRowHeights(index, index);
56140         this.layout();
56141         this.fireEvent("rowupdated", this, index, record);
56142     },
56143
56144     onAdd : function(ds, records, index){
56145         this.insertRows(ds, index, index + (records.length-1));
56146     },
56147
56148     onRemove : function(ds, record, index, isUpdate){
56149         if(isUpdate !== true){
56150             this.fireEvent("beforerowremoved", this, index, record);
56151         }
56152         var bt = this.getBodyTable(), lt = this.getLockedTable();
56153         if(bt.rows[index]){
56154             bt.firstChild.removeChild(bt.rows[index]);
56155         }
56156         if(lt.rows[index]){
56157             lt.firstChild.removeChild(lt.rows[index]);
56158         }
56159         if(isUpdate !== true){
56160             this.stripeRows(index);
56161             this.syncRowHeights(index, index);
56162             this.layout();
56163             this.fireEvent("rowremoved", this, index, record);
56164         }
56165     },
56166
56167     onLoad : function(){
56168         this.scrollToTop();
56169     },
56170
56171     /**
56172      * Scrolls the grid to the top
56173      */
56174     scrollToTop : function(){
56175         if(this.scroller){
56176             this.scroller.dom.scrollTop = 0;
56177             this.syncScroll();
56178         }
56179     },
56180
56181     /**
56182      * Gets a panel in the header of the grid that can be used for toolbars etc.
56183      * After modifying the contents of this panel a call to grid.autoSize() may be
56184      * required to register any changes in size.
56185      * @param {Boolean} doShow By default the header is hidden. Pass true to show the panel
56186      * @return Roo.Element
56187      */
56188     getHeaderPanel : function(doShow){
56189         if(doShow){
56190             this.headerPanel.show();
56191         }
56192         return this.headerPanel;
56193     },
56194
56195     /**
56196      * Gets a panel in the footer of the grid that can be used for toolbars etc.
56197      * After modifying the contents of this panel a call to grid.autoSize() may be
56198      * required to register any changes in size.
56199      * @param {Boolean} doShow By default the footer is hidden. Pass true to show the panel
56200      * @return Roo.Element
56201      */
56202     getFooterPanel : function(doShow){
56203         if(doShow){
56204             this.footerPanel.show();
56205         }
56206         return this.footerPanel;
56207     },
56208
56209     initElements : function(){
56210         var E = Roo.Element;
56211         var el = this.grid.getGridEl().dom.firstChild;
56212         var cs = el.childNodes;
56213
56214         this.el = new E(el);
56215         
56216          this.focusEl = new E(el.firstChild);
56217         this.focusEl.swallowEvent("click", true);
56218         
56219         this.headerPanel = new E(cs[1]);
56220         this.headerPanel.enableDisplayMode("block");
56221
56222         this.scroller = new E(cs[2]);
56223         this.scrollSizer = new E(this.scroller.dom.firstChild);
56224
56225         this.lockedWrap = new E(cs[3]);
56226         this.lockedHd = new E(this.lockedWrap.dom.firstChild);
56227         this.lockedBody = new E(this.lockedWrap.dom.childNodes[1]);
56228
56229         this.mainWrap = new E(cs[4]);
56230         this.mainHd = new E(this.mainWrap.dom.firstChild);
56231         this.mainBody = new E(this.mainWrap.dom.childNodes[1]);
56232
56233         this.footerPanel = new E(cs[5]);
56234         this.footerPanel.enableDisplayMode("block");
56235
56236         this.resizeProxy = new E(cs[6]);
56237
56238         this.headerSelector = String.format(
56239            '#{0} td.x-grid-hd, #{1} td.x-grid-hd',
56240            this.lockedHd.id, this.mainHd.id
56241         );
56242
56243         this.splitterSelector = String.format(
56244            '#{0} div.x-grid-split, #{1} div.x-grid-split',
56245            this.idToCssName(this.lockedHd.id), this.idToCssName(this.mainHd.id)
56246         );
56247     },
56248     idToCssName : function(s)
56249     {
56250         return s.replace(/[^a-z0-9]+/ig, '-');
56251     },
56252
56253     getHeaderCell : function(index){
56254         return Roo.DomQuery.select(this.headerSelector)[index];
56255     },
56256
56257     getHeaderCellMeasure : function(index){
56258         return this.getHeaderCell(index).firstChild;
56259     },
56260
56261     getHeaderCellText : function(index){
56262         return this.getHeaderCell(index).firstChild.firstChild;
56263     },
56264
56265     getLockedTable : function(){
56266         return this.lockedBody.dom.firstChild;
56267     },
56268
56269     getBodyTable : function(){
56270         return this.mainBody.dom.firstChild;
56271     },
56272
56273     getLockedRow : function(index){
56274         return this.getLockedTable().rows[index];
56275     },
56276
56277     getRow : function(index){
56278         return this.getBodyTable().rows[index];
56279     },
56280
56281     getRowComposite : function(index){
56282         if(!this.rowEl){
56283             this.rowEl = new Roo.CompositeElementLite();
56284         }
56285         var els = [], lrow, mrow;
56286         if(lrow = this.getLockedRow(index)){
56287             els.push(lrow);
56288         }
56289         if(mrow = this.getRow(index)){
56290             els.push(mrow);
56291         }
56292         this.rowEl.elements = els;
56293         return this.rowEl;
56294     },
56295     /**
56296      * Gets the 'td' of the cell
56297      * 
56298      * @param {Integer} rowIndex row to select
56299      * @param {Integer} colIndex column to select
56300      * 
56301      * @return {Object} 
56302      */
56303     getCell : function(rowIndex, colIndex){
56304         var locked = this.cm.getLockedCount();
56305         var source;
56306         if(colIndex < locked){
56307             source = this.lockedBody.dom.firstChild;
56308         }else{
56309             source = this.mainBody.dom.firstChild;
56310             colIndex -= locked;
56311         }
56312         return source.rows[rowIndex].childNodes[colIndex];
56313     },
56314
56315     getCellText : function(rowIndex, colIndex){
56316         return this.getCell(rowIndex, colIndex).firstChild.firstChild;
56317     },
56318
56319     getCellBox : function(cell){
56320         var b = this.fly(cell).getBox();
56321         if(Roo.isOpera){ // opera fails to report the Y
56322             b.y = cell.offsetTop + this.mainBody.getY();
56323         }
56324         return b;
56325     },
56326
56327     getCellIndex : function(cell){
56328         var id = String(cell.className).match(this.cellRE);
56329         if(id){
56330             return parseInt(id[1], 10);
56331         }
56332         return 0;
56333     },
56334
56335     findHeaderIndex : function(n){
56336         var r = Roo.fly(n).findParent("td." + this.hdClass, 6);
56337         return r ? this.getCellIndex(r) : false;
56338     },
56339
56340     findHeaderCell : function(n){
56341         var r = Roo.fly(n).findParent("td." + this.hdClass, 6);
56342         return r ? r : false;
56343     },
56344
56345     findRowIndex : function(n){
56346         if(!n){
56347             return false;
56348         }
56349         var r = Roo.fly(n).findParent("tr." + this.rowClass, 6);
56350         return r ? r.rowIndex : false;
56351     },
56352
56353     findCellIndex : function(node){
56354         var stop = this.el.dom;
56355         while(node && node != stop){
56356             if(this.findRE.test(node.className)){
56357                 return this.getCellIndex(node);
56358             }
56359             node = node.parentNode;
56360         }
56361         return false;
56362     },
56363
56364     getColumnId : function(index){
56365         return this.cm.getColumnId(index);
56366     },
56367
56368     getSplitters : function()
56369     {
56370         if(this.splitterSelector){
56371            return Roo.DomQuery.select(this.splitterSelector);
56372         }else{
56373             return null;
56374       }
56375     },
56376
56377     getSplitter : function(index){
56378         return this.getSplitters()[index];
56379     },
56380
56381     onRowOver : function(e, t){
56382         var row;
56383         if((row = this.findRowIndex(t)) !== false){
56384             this.getRowComposite(row).addClass("x-grid-row-over");
56385         }
56386     },
56387
56388     onRowOut : function(e, t){
56389         var row;
56390         if((row = this.findRowIndex(t)) !== false && row !== this.findRowIndex(e.getRelatedTarget())){
56391             this.getRowComposite(row).removeClass("x-grid-row-over");
56392         }
56393     },
56394
56395     renderHeaders : function(){
56396         var cm = this.cm;
56397         var ct = this.templates.hcell, ht = this.templates.header, st = this.templates.hsplit;
56398         var cb = [], lb = [], sb = [], lsb = [], p = {};
56399         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56400             p.cellId = "x-grid-hd-0-" + i;
56401             p.splitId = "x-grid-csplit-0-" + i;
56402             p.id = cm.getColumnId(i);
56403             p.value = cm.getColumnHeader(i) || "";
56404             p.title = cm.getColumnTooltip(i) || (''+p.value).match(/\</)  ? '' :  p.value  || "";
56405             p.style = (this.grid.enableColumnResize === false || !cm.isResizable(i) || cm.isFixed(i)) ? 'cursor:default' : '';
56406             if(!cm.isLocked(i)){
56407                 cb[cb.length] = ct.apply(p);
56408                 sb[sb.length] = st.apply(p);
56409             }else{
56410                 lb[lb.length] = ct.apply(p);
56411                 lsb[lsb.length] = st.apply(p);
56412             }
56413         }
56414         return [ht.apply({cells: lb.join(""), splits:lsb.join("")}),
56415                 ht.apply({cells: cb.join(""), splits:sb.join("")})];
56416     },
56417
56418     updateHeaders : function(){
56419         var html = this.renderHeaders();
56420         this.lockedHd.update(html[0]);
56421         this.mainHd.update(html[1]);
56422     },
56423
56424     /**
56425      * Focuses the specified row.
56426      * @param {Number} row The row index
56427      */
56428     focusRow : function(row)
56429     {
56430         //Roo.log('GridView.focusRow');
56431         var x = this.scroller.dom.scrollLeft;
56432         this.focusCell(row, 0, false);
56433         this.scroller.dom.scrollLeft = x;
56434     },
56435
56436     /**
56437      * Focuses the specified cell.
56438      * @param {Number} row The row index
56439      * @param {Number} col The column index
56440      * @param {Boolean} hscroll false to disable horizontal scrolling
56441      */
56442     focusCell : function(row, col, hscroll)
56443     {
56444         //Roo.log('GridView.focusCell');
56445         var el = this.ensureVisible(row, col, hscroll);
56446         this.focusEl.alignTo(el, "tl-tl");
56447         if(Roo.isGecko){
56448             this.focusEl.focus();
56449         }else{
56450             this.focusEl.focus.defer(1, this.focusEl);
56451         }
56452     },
56453
56454     /**
56455      * Scrolls the specified cell into view
56456      * @param {Number} row The row index
56457      * @param {Number} col The column index
56458      * @param {Boolean} hscroll false to disable horizontal scrolling
56459      */
56460     ensureVisible : function(row, col, hscroll)
56461     {
56462         //Roo.log('GridView.ensureVisible,' + row + ',' + col);
56463         //return null; //disable for testing.
56464         if(typeof row != "number"){
56465             row = row.rowIndex;
56466         }
56467         if(row < 0 && row >= this.ds.getCount()){
56468             return  null;
56469         }
56470         col = (col !== undefined ? col : 0);
56471         var cm = this.grid.colModel;
56472         while(cm.isHidden(col)){
56473             col++;
56474         }
56475
56476         var el = this.getCell(row, col);
56477         if(!el){
56478             return null;
56479         }
56480         var c = this.scroller.dom;
56481
56482         var ctop = parseInt(el.offsetTop, 10);
56483         var cleft = parseInt(el.offsetLeft, 10);
56484         var cbot = ctop + el.offsetHeight;
56485         var cright = cleft + el.offsetWidth;
56486         
56487         var ch = c.clientHeight - this.mainHd.dom.offsetHeight;
56488         var stop = parseInt(c.scrollTop, 10);
56489         var sleft = parseInt(c.scrollLeft, 10);
56490         var sbot = stop + ch;
56491         var sright = sleft + c.clientWidth;
56492         /*
56493         Roo.log('GridView.ensureVisible:' +
56494                 ' ctop:' + ctop +
56495                 ' c.clientHeight:' + c.clientHeight +
56496                 ' this.mainHd.dom.offsetHeight:' + this.mainHd.dom.offsetHeight +
56497                 ' stop:' + stop +
56498                 ' cbot:' + cbot +
56499                 ' sbot:' + sbot +
56500                 ' ch:' + ch  
56501                 );
56502         */
56503         if(ctop < stop){
56504              c.scrollTop = ctop;
56505             //Roo.log("set scrolltop to ctop DISABLE?");
56506         }else if(cbot > sbot){
56507             //Roo.log("set scrolltop to cbot-ch");
56508             c.scrollTop = cbot-ch;
56509         }
56510         
56511         if(hscroll !== false){
56512             if(cleft < sleft){
56513                 c.scrollLeft = cleft;
56514             }else if(cright > sright){
56515                 c.scrollLeft = cright-c.clientWidth;
56516             }
56517         }
56518          
56519         return el;
56520     },
56521
56522     updateColumns : function(){
56523         this.grid.stopEditing();
56524         var cm = this.grid.colModel, colIds = this.getColumnIds();
56525         //var totalWidth = cm.getTotalWidth();
56526         var pos = 0;
56527         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56528             //if(cm.isHidden(i)) continue;
56529             var w = cm.getColumnWidth(i);
56530             this.css.updateRule(this.colSelector+this.idToCssName(colIds[i]), "width", (w - this.borderWidth) + "px");
56531             this.css.updateRule(this.hdSelector+this.idToCssName(colIds[i]), "width", (w - this.borderWidth) + "px");
56532         }
56533         this.updateSplitters();
56534     },
56535
56536     generateRules : function(cm){
56537         var ruleBuf = [], rulesId = this.idToCssName(this.grid.id)+ '-cssrules';
56538         Roo.util.CSS.removeStyleSheet(rulesId);
56539         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56540             var cid = cm.getColumnId(i);
56541             var align = '';
56542             if(cm.config[i].align){
56543                 align = 'text-align:'+cm.config[i].align+';';
56544             }
56545             var hidden = '';
56546             if(cm.isHidden(i)){
56547                 hidden = 'display:none;';
56548             }
56549             var width = "width:" + (cm.getColumnWidth(i) - this.borderWidth) + "px;";
56550             ruleBuf.push(
56551                     this.colSelector, cid, " {\n", cm.config[i].css, align, width, "\n}\n",
56552                     this.hdSelector, cid, " {\n", align, width, "}\n",
56553                     this.tdSelector, cid, " {\n",hidden,"\n}\n",
56554                     this.splitSelector, cid, " {\n", hidden , "\n}\n");
56555         }
56556         return Roo.util.CSS.createStyleSheet(ruleBuf.join(""), rulesId);
56557     },
56558
56559     updateSplitters : function(){
56560         var cm = this.cm, s = this.getSplitters();
56561         if(s){ // splitters not created yet
56562             var pos = 0, locked = true;
56563             for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56564                 if(cm.isHidden(i)) {
56565                     continue;
56566                 }
56567                 var w = cm.getColumnWidth(i); // make sure it's a number
56568                 if(!cm.isLocked(i) && locked){
56569                     pos = 0;
56570                     locked = false;
56571                 }
56572                 pos += w;
56573                 s[i].style.left = (pos-this.splitOffset) + "px";
56574             }
56575         }
56576     },
56577
56578     handleHiddenChange : function(colModel, colIndex, hidden){
56579         if(hidden){
56580             this.hideColumn(colIndex);
56581         }else{
56582             this.unhideColumn(colIndex);
56583         }
56584     },
56585
56586     hideColumn : function(colIndex){
56587         var cid = this.getColumnId(colIndex);
56588         this.css.updateRule(this.tdSelector+this.idToCssName(cid), "display", "none");
56589         this.css.updateRule(this.splitSelector+this.idToCssName(cid), "display", "none");
56590         if(Roo.isSafari){
56591             this.updateHeaders();
56592         }
56593         this.updateSplitters();
56594         this.layout();
56595     },
56596
56597     unhideColumn : function(colIndex){
56598         var cid = this.getColumnId(colIndex);
56599         this.css.updateRule(this.tdSelector+this.idToCssName(cid), "display", "");
56600         this.css.updateRule(this.splitSelector+this.idToCssName(cid), "display", "");
56601
56602         if(Roo.isSafari){
56603             this.updateHeaders();
56604         }
56605         this.updateSplitters();
56606         this.layout();
56607     },
56608
56609     insertRows : function(dm, firstRow, lastRow, isUpdate){
56610         if(firstRow == 0 && lastRow == dm.getCount()-1){
56611             this.refresh();
56612         }else{
56613             if(!isUpdate){
56614                 this.fireEvent("beforerowsinserted", this, firstRow, lastRow);
56615             }
56616             var s = this.getScrollState();
56617             var markup = this.renderRows(firstRow, lastRow);
56618             this.bufferRows(markup[0], this.getLockedTable(), firstRow);
56619             this.bufferRows(markup[1], this.getBodyTable(), firstRow);
56620             this.restoreScroll(s);
56621             if(!isUpdate){
56622                 this.fireEvent("rowsinserted", this, firstRow, lastRow);
56623                 this.syncRowHeights(firstRow, lastRow);
56624                 this.stripeRows(firstRow);
56625                 this.layout();
56626             }
56627         }
56628     },
56629
56630     bufferRows : function(markup, target, index){
56631         var before = null, trows = target.rows, tbody = target.tBodies[0];
56632         if(index < trows.length){
56633             before = trows[index];
56634         }
56635         var b = document.createElement("div");
56636         b.innerHTML = "<table><tbody>"+markup+"</tbody></table>";
56637         var rows = b.firstChild.rows;
56638         for(var i = 0, len = rows.length; i < len; i++){
56639             if(before){
56640                 tbody.insertBefore(rows[0], before);
56641             }else{
56642                 tbody.appendChild(rows[0]);
56643             }
56644         }
56645         b.innerHTML = "";
56646         b = null;
56647     },
56648
56649     deleteRows : function(dm, firstRow, lastRow){
56650         if(dm.getRowCount()<1){
56651             this.fireEvent("beforerefresh", this);
56652             this.mainBody.update("");
56653             this.lockedBody.update("");
56654             this.fireEvent("refresh", this);
56655         }else{
56656             this.fireEvent("beforerowsdeleted", this, firstRow, lastRow);
56657             var bt = this.getBodyTable();
56658             var tbody = bt.firstChild;
56659             var rows = bt.rows;
56660             for(var rowIndex = firstRow; rowIndex <= lastRow; rowIndex++){
56661                 tbody.removeChild(rows[firstRow]);
56662             }
56663             this.stripeRows(firstRow);
56664             this.fireEvent("rowsdeleted", this, firstRow, lastRow);
56665         }
56666     },
56667
56668     updateRows : function(dataSource, firstRow, lastRow){
56669         var s = this.getScrollState();
56670         this.refresh();
56671         this.restoreScroll(s);
56672     },
56673
56674     handleSort : function(dataSource, sortColumnIndex, sortDir, noRefresh){
56675         if(!noRefresh){
56676            this.refresh();
56677         }
56678         this.updateHeaderSortState();
56679     },
56680
56681     getScrollState : function(){
56682         
56683         var sb = this.scroller.dom;
56684         return {left: sb.scrollLeft, top: sb.scrollTop};
56685     },
56686
56687     stripeRows : function(startRow){
56688         if(!this.grid.stripeRows || this.ds.getCount() < 1){
56689             return;
56690         }
56691         startRow = startRow || 0;
56692         var rows = this.getBodyTable().rows;
56693         var lrows = this.getLockedTable().rows;
56694         var cls = ' x-grid-row-alt ';
56695         for(var i = startRow, len = rows.length; i < len; i++){
56696             var row = rows[i], lrow = lrows[i];
56697             var isAlt = ((i+1) % 2 == 0);
56698             var hasAlt = (' '+row.className + ' ').indexOf(cls) != -1;
56699             if(isAlt == hasAlt){
56700                 continue;
56701             }
56702             if(isAlt){
56703                 row.className += " x-grid-row-alt";
56704             }else{
56705                 row.className = row.className.replace("x-grid-row-alt", "");
56706             }
56707             if(lrow){
56708                 lrow.className = row.className;
56709             }
56710         }
56711     },
56712
56713     restoreScroll : function(state){
56714         //Roo.log('GridView.restoreScroll');
56715         var sb = this.scroller.dom;
56716         sb.scrollLeft = state.left;
56717         sb.scrollTop = state.top;
56718         this.syncScroll();
56719     },
56720
56721     syncScroll : function(){
56722         //Roo.log('GridView.syncScroll');
56723         var sb = this.scroller.dom;
56724         var sh = this.mainHd.dom;
56725         var bs = this.mainBody.dom;
56726         var lv = this.lockedBody.dom;
56727         sh.scrollLeft = bs.scrollLeft = sb.scrollLeft;
56728         lv.scrollTop = bs.scrollTop = sb.scrollTop;
56729     },
56730
56731     handleScroll : function(e){
56732         this.syncScroll();
56733         var sb = this.scroller.dom;
56734         this.grid.fireEvent("bodyscroll", sb.scrollLeft, sb.scrollTop);
56735         e.stopEvent();
56736     },
56737
56738     handleWheel : function(e){
56739         var d = e.getWheelDelta();
56740         this.scroller.dom.scrollTop -= d*22;
56741         // set this here to prevent jumpy scrolling on large tables
56742         this.lockedBody.dom.scrollTop = this.mainBody.dom.scrollTop = this.scroller.dom.scrollTop;
56743         e.stopEvent();
56744     },
56745
56746     renderRows : function(startRow, endRow){
56747         // pull in all the crap needed to render rows
56748         var g = this.grid, cm = g.colModel, ds = g.dataSource, stripe = g.stripeRows;
56749         var colCount = cm.getColumnCount();
56750
56751         if(ds.getCount() < 1){
56752             return ["", ""];
56753         }
56754
56755         // build a map for all the columns
56756         var cs = [];
56757         for(var i = 0; i < colCount; i++){
56758             var name = cm.getDataIndex(i);
56759             cs[i] = {
56760                 name : typeof name == 'undefined' ? ds.fields.get(i).name : name,
56761                 renderer : cm.getRenderer(i),
56762                 id : cm.getColumnId(i),
56763                 locked : cm.isLocked(i),
56764                 has_editor : cm.isCellEditable(i)
56765             };
56766         }
56767
56768         startRow = startRow || 0;
56769         endRow = typeof endRow == "undefined"? ds.getCount()-1 : endRow;
56770
56771         // records to render
56772         var rs = ds.getRange(startRow, endRow);
56773
56774         return this.doRender(cs, rs, ds, startRow, colCount, stripe);
56775     },
56776
56777     // As much as I hate to duplicate code, this was branched because FireFox really hates
56778     // [].join("") on strings. The performance difference was substantial enough to
56779     // branch this function
56780     doRender : Roo.isGecko ?
56781             function(cs, rs, ds, startRow, colCount, stripe){
56782                 var ts = this.templates, ct = ts.cell, rt = ts.row;
56783                 // buffers
56784                 var buf = "", lbuf = "", cb, lcb, c, p = {}, rp = {}, r, rowIndex;
56785                 
56786                 var hasListener = this.grid.hasListener('rowclass');
56787                 var rowcfg = {};
56788                 for(var j = 0, len = rs.length; j < len; j++){
56789                     r = rs[j]; cb = ""; lcb = ""; rowIndex = (j+startRow);
56790                     for(var i = 0; i < colCount; i++){
56791                         c = cs[i];
56792                         p.cellId = "x-grid-cell-" + rowIndex + "-" + i;
56793                         p.id = c.id;
56794                         p.css = p.attr = "";
56795                         p.value = c.renderer(r.data[c.name], p, r, rowIndex, i, ds);
56796                         if(p.value == undefined || p.value === "") {
56797                             p.value = "&#160;";
56798                         }
56799                         if(c.has_editor){
56800                             p.css += ' x-grid-editable-cell';
56801                         }
56802                         if(c.dirty && typeof r.modified[c.name] !== 'undefined'){
56803                             p.css +=  ' x-grid-dirty-cell';
56804                         }
56805                         var markup = ct.apply(p);
56806                         if(!c.locked){
56807                             cb+= markup;
56808                         }else{
56809                             lcb+= markup;
56810                         }
56811                     }
56812                     var alt = [];
56813                     if(stripe && ((rowIndex+1) % 2 == 0)){
56814                         alt.push("x-grid-row-alt")
56815                     }
56816                     if(r.dirty){
56817                         alt.push(  " x-grid-dirty-row");
56818                     }
56819                     rp.cells = lcb;
56820                     if(this.getRowClass){
56821                         alt.push(this.getRowClass(r, rowIndex));
56822                     }
56823                     if (hasListener) {
56824                         rowcfg = {
56825                              
56826                             record: r,
56827                             rowIndex : rowIndex,
56828                             rowClass : ''
56829                         };
56830                         this.grid.fireEvent('rowclass', this, rowcfg);
56831                         alt.push(rowcfg.rowClass);
56832                     }
56833                     rp.alt = alt.join(" ");
56834                     lbuf+= rt.apply(rp);
56835                     rp.cells = cb;
56836                     buf+=  rt.apply(rp);
56837                 }
56838                 return [lbuf, buf];
56839             } :
56840             function(cs, rs, ds, startRow, colCount, stripe){
56841                 var ts = this.templates, ct = ts.cell, rt = ts.row;
56842                 // buffers
56843                 var buf = [], lbuf = [], cb, lcb, c, p = {}, rp = {}, r, rowIndex;
56844                 var hasListener = this.grid.hasListener('rowclass');
56845  
56846                 var rowcfg = {};
56847                 for(var j = 0, len = rs.length; j < len; j++){
56848                     r = rs[j]; cb = []; lcb = []; rowIndex = (j+startRow);
56849                     for(var i = 0; i < colCount; i++){
56850                         c = cs[i];
56851                         p.cellId = "x-grid-cell-" + rowIndex + "-" + i;
56852                         p.id = c.id;
56853                         p.css = p.attr = "";
56854                         p.value = c.renderer(r.data[c.name], p, r, rowIndex, i, ds);
56855                         if(p.value == undefined || p.value === "") {
56856                             p.value = "&#160;";
56857                         }
56858                         //Roo.log(c);
56859                          if(c.has_editor){
56860                             p.css += ' x-grid-editable-cell';
56861                         }
56862                         if(r.dirty && typeof r.modified[c.name] !== 'undefined'){
56863                             p.css += ' x-grid-dirty-cell' 
56864                         }
56865                         
56866                         var markup = ct.apply(p);
56867                         if(!c.locked){
56868                             cb[cb.length] = markup;
56869                         }else{
56870                             lcb[lcb.length] = markup;
56871                         }
56872                     }
56873                     var alt = [];
56874                     if(stripe && ((rowIndex+1) % 2 == 0)){
56875                         alt.push( "x-grid-row-alt");
56876                     }
56877                     if(r.dirty){
56878                         alt.push(" x-grid-dirty-row");
56879                     }
56880                     rp.cells = lcb;
56881                     if(this.getRowClass){
56882                         alt.push( this.getRowClass(r, rowIndex));
56883                     }
56884                     if (hasListener) {
56885                         rowcfg = {
56886                              
56887                             record: r,
56888                             rowIndex : rowIndex,
56889                             rowClass : ''
56890                         };
56891                         this.grid.fireEvent('rowclass', this, rowcfg);
56892                         alt.push(rowcfg.rowClass);
56893                     }
56894                     
56895                     rp.alt = alt.join(" ");
56896                     rp.cells = lcb.join("");
56897                     lbuf[lbuf.length] = rt.apply(rp);
56898                     rp.cells = cb.join("");
56899                     buf[buf.length] =  rt.apply(rp);
56900                 }
56901                 return [lbuf.join(""), buf.join("")];
56902             },
56903
56904     renderBody : function(){
56905         var markup = this.renderRows();
56906         var bt = this.templates.body;
56907         return [bt.apply({rows: markup[0]}), bt.apply({rows: markup[1]})];
56908     },
56909
56910     /**
56911      * Refreshes the grid
56912      * @param {Boolean} headersToo
56913      */
56914     refresh : function(headersToo){
56915         this.fireEvent("beforerefresh", this);
56916         this.grid.stopEditing();
56917         var result = this.renderBody();
56918         this.lockedBody.update(result[0]);
56919         this.mainBody.update(result[1]);
56920         if(headersToo === true){
56921             this.updateHeaders();
56922             this.updateColumns();
56923             this.updateSplitters();
56924             this.updateHeaderSortState();
56925         }
56926         this.syncRowHeights();
56927         this.layout();
56928         this.fireEvent("refresh", this);
56929     },
56930
56931     handleColumnMove : function(cm, oldIndex, newIndex){
56932         this.indexMap = null;
56933         var s = this.getScrollState();
56934         this.refresh(true);
56935         this.restoreScroll(s);
56936         this.afterMove(newIndex);
56937     },
56938
56939     afterMove : function(colIndex){
56940         if(this.enableMoveAnim && Roo.enableFx){
56941             this.fly(this.getHeaderCell(colIndex).firstChild).highlight(this.hlColor);
56942         }
56943         // if multisort - fix sortOrder, and reload..
56944         if (this.grid.dataSource.multiSort) {
56945             // the we can call sort again..
56946             var dm = this.grid.dataSource;
56947             var cm = this.grid.colModel;
56948             var so = [];
56949             for(var i = 0; i < cm.config.length; i++ ) {
56950                 
56951                 if ((typeof(dm.sortToggle[cm.config[i].dataIndex]) == 'undefined')) {
56952                     continue; // dont' bother, it's not in sort list or being set.
56953                 }
56954                 
56955                 so.push(cm.config[i].dataIndex);
56956             };
56957             dm.sortOrder = so;
56958             dm.load(dm.lastOptions);
56959             
56960             
56961         }
56962         
56963     },
56964
56965     updateCell : function(dm, rowIndex, dataIndex){
56966         var colIndex = this.getColumnIndexByDataIndex(dataIndex);
56967         if(typeof colIndex == "undefined"){ // not present in grid
56968             return;
56969         }
56970         var cm = this.grid.colModel;
56971         var cell = this.getCell(rowIndex, colIndex);
56972         var cellText = this.getCellText(rowIndex, colIndex);
56973
56974         var p = {
56975             cellId : "x-grid-cell-" + rowIndex + "-" + colIndex,
56976             id : cm.getColumnId(colIndex),
56977             css: colIndex == cm.getColumnCount()-1 ? "x-grid-col-last" : ""
56978         };
56979         var renderer = cm.getRenderer(colIndex);
56980         var val = renderer(dm.getValueAt(rowIndex, dataIndex), p, rowIndex, colIndex, dm);
56981         if(typeof val == "undefined" || val === "") {
56982             val = "&#160;";
56983         }
56984         cellText.innerHTML = val;
56985         cell.className = this.cellClass + " " + this.idToCssName(p.cellId) + " " + p.css;
56986         this.syncRowHeights(rowIndex, rowIndex);
56987     },
56988
56989     calcColumnWidth : function(colIndex, maxRowsToMeasure){
56990         var maxWidth = 0;
56991         if(this.grid.autoSizeHeaders){
56992             var h = this.getHeaderCellMeasure(colIndex);
56993             maxWidth = Math.max(maxWidth, h.scrollWidth);
56994         }
56995         var tb, index;
56996         if(this.cm.isLocked(colIndex)){
56997             tb = this.getLockedTable();
56998             index = colIndex;
56999         }else{
57000             tb = this.getBodyTable();
57001             index = colIndex - this.cm.getLockedCount();
57002         }
57003         if(tb && tb.rows){
57004             var rows = tb.rows;
57005             var stopIndex = Math.min(maxRowsToMeasure || rows.length, rows.length);
57006             for(var i = 0; i < stopIndex; i++){
57007                 var cell = rows[i].childNodes[index].firstChild;
57008                 maxWidth = Math.max(maxWidth, cell.scrollWidth);
57009             }
57010         }
57011         return maxWidth + /*margin for error in IE*/ 5;
57012     },
57013     /**
57014      * Autofit a column to its content.
57015      * @param {Number} colIndex
57016      * @param {Boolean} forceMinSize true to force the column to go smaller if possible
57017      */
57018      autoSizeColumn : function(colIndex, forceMinSize, suppressEvent){
57019          if(this.cm.isHidden(colIndex)){
57020              return; // can't calc a hidden column
57021          }
57022         if(forceMinSize){
57023             var cid = this.cm.getColumnId(colIndex);
57024             this.css.updateRule(this.colSelector +this.idToCssName( cid), "width", this.grid.minColumnWidth + "px");
57025            if(this.grid.autoSizeHeaders){
57026                this.css.updateRule(this.hdSelector + this.idToCssName(cid), "width", this.grid.minColumnWidth + "px");
57027            }
57028         }
57029         var newWidth = this.calcColumnWidth(colIndex);
57030         this.cm.setColumnWidth(colIndex,
57031             Math.max(this.grid.minColumnWidth, newWidth), suppressEvent);
57032         if(!suppressEvent){
57033             this.grid.fireEvent("columnresize", colIndex, newWidth);
57034         }
57035     },
57036
57037     /**
57038      * Autofits all columns to their content and then expands to fit any extra space in the grid
57039      */
57040      autoSizeColumns : function(){
57041         var cm = this.grid.colModel;
57042         var colCount = cm.getColumnCount();
57043         for(var i = 0; i < colCount; i++){
57044             this.autoSizeColumn(i, true, true);
57045         }
57046         if(cm.getTotalWidth() < this.scroller.dom.clientWidth){
57047             this.fitColumns();
57048         }else{
57049             this.updateColumns();
57050             this.layout();
57051         }
57052     },
57053
57054     /**
57055      * Autofits all columns to the grid's width proportionate with their current size
57056      * @param {Boolean} reserveScrollSpace Reserve space for a scrollbar
57057      */
57058     fitColumns : function(reserveScrollSpace){
57059         var cm = this.grid.colModel;
57060         var colCount = cm.getColumnCount();
57061         var cols = [];
57062         var width = 0;
57063         var i, w;
57064         for (i = 0; i < colCount; i++){
57065             if(!cm.isHidden(i) && !cm.isFixed(i)){
57066                 w = cm.getColumnWidth(i);
57067                 cols.push(i);
57068                 cols.push(w);
57069                 width += w;
57070             }
57071         }
57072         var avail = Math.min(this.scroller.dom.clientWidth, this.el.getWidth());
57073         if(reserveScrollSpace){
57074             avail -= 17;
57075         }
57076         var frac = (avail - cm.getTotalWidth())/width;
57077         while (cols.length){
57078             w = cols.pop();
57079             i = cols.pop();
57080             cm.setColumnWidth(i, Math.floor(w + w*frac), true);
57081         }
57082         this.updateColumns();
57083         this.layout();
57084     },
57085
57086     onRowSelect : function(rowIndex){
57087         var row = this.getRowComposite(rowIndex);
57088         row.addClass("x-grid-row-selected");
57089     },
57090
57091     onRowDeselect : function(rowIndex){
57092         var row = this.getRowComposite(rowIndex);
57093         row.removeClass("x-grid-row-selected");
57094     },
57095
57096     onCellSelect : function(row, col){
57097         var cell = this.getCell(row, col);
57098         if(cell){
57099             Roo.fly(cell).addClass("x-grid-cell-selected");
57100         }
57101     },
57102
57103     onCellDeselect : function(row, col){
57104         var cell = this.getCell(row, col);
57105         if(cell){
57106             Roo.fly(cell).removeClass("x-grid-cell-selected");
57107         }
57108     },
57109
57110     updateHeaderSortState : function(){
57111         
57112         // sort state can be single { field: xxx, direction : yyy}
57113         // or   { xxx=>ASC , yyy : DESC ..... }
57114         
57115         var mstate = {};
57116         if (!this.ds.multiSort) { 
57117             var state = this.ds.getSortState();
57118             if(!state){
57119                 return;
57120             }
57121             mstate[state.field] = state.direction;
57122             // FIXME... - this is not used here.. but might be elsewhere..
57123             this.sortState = state;
57124             
57125         } else {
57126             mstate = this.ds.sortToggle;
57127         }
57128         //remove existing sort classes..
57129         
57130         var sc = this.sortClasses;
57131         var hds = this.el.select(this.headerSelector).removeClass(sc);
57132         
57133         for(var f in mstate) {
57134         
57135             var sortColumn = this.cm.findColumnIndex(f);
57136             
57137             if(sortColumn != -1){
57138                 var sortDir = mstate[f];        
57139                 hds.item(sortColumn).addClass(sc[sortDir == "DESC" ? 1 : 0]);
57140             }
57141         }
57142         
57143          
57144         
57145     },
57146
57147
57148     handleHeaderClick : function(g, index,e){
57149         
57150         Roo.log("header click");
57151         
57152         if (Roo.isTouch) {
57153             // touch events on header are handled by context
57154             this.handleHdCtx(g,index,e);
57155             return;
57156         }
57157         
57158         
57159         if(this.headersDisabled){
57160             return;
57161         }
57162         var dm = g.dataSource, cm = g.colModel;
57163         if(!cm.isSortable(index)){
57164             return;
57165         }
57166         g.stopEditing();
57167         
57168         if (dm.multiSort) {
57169             // update the sortOrder
57170             var so = [];
57171             for(var i = 0; i < cm.config.length; i++ ) {
57172                 
57173                 if ((typeof(dm.sortToggle[cm.config[i].dataIndex]) == 'undefined') && (index != i)) {
57174                     continue; // dont' bother, it's not in sort list or being set.
57175                 }
57176                 
57177                 so.push(cm.config[i].dataIndex);
57178             };
57179             dm.sortOrder = so;
57180         }
57181         
57182         
57183         dm.sort(cm.getDataIndex(index));
57184     },
57185
57186
57187     destroy : function(){
57188         if(this.colMenu){
57189             this.colMenu.removeAll();
57190             Roo.menu.MenuMgr.unregister(this.colMenu);
57191             this.colMenu.getEl().remove();
57192             delete this.colMenu;
57193         }
57194         if(this.hmenu){
57195             this.hmenu.removeAll();
57196             Roo.menu.MenuMgr.unregister(this.hmenu);
57197             this.hmenu.getEl().remove();
57198             delete this.hmenu;
57199         }
57200         if(this.grid.enableColumnMove){
57201             var dds = Roo.dd.DDM.ids['gridHeader' + this.grid.getGridEl().id];
57202             if(dds){
57203                 for(var dd in dds){
57204                     if(!dds[dd].config.isTarget && dds[dd].dragElId){
57205                         var elid = dds[dd].dragElId;
57206                         dds[dd].unreg();
57207                         Roo.get(elid).remove();
57208                     } else if(dds[dd].config.isTarget){
57209                         dds[dd].proxyTop.remove();
57210                         dds[dd].proxyBottom.remove();
57211                         dds[dd].unreg();
57212                     }
57213                     if(Roo.dd.DDM.locationCache[dd]){
57214                         delete Roo.dd.DDM.locationCache[dd];
57215                     }
57216                 }
57217                 delete Roo.dd.DDM.ids['gridHeader' + this.grid.getGridEl().id];
57218             }
57219         }
57220         Roo.util.CSS.removeStyleSheet(this.idToCssName(this.grid.id) + '-cssrules');
57221         this.bind(null, null);
57222         Roo.EventManager.removeResizeListener(this.onWindowResize, this);
57223     },
57224
57225     handleLockChange : function(){
57226         this.refresh(true);
57227     },
57228
57229     onDenyColumnLock : function(){
57230
57231     },
57232
57233     onDenyColumnHide : function(){
57234
57235     },
57236
57237     handleHdMenuClick : function(item){
57238         var index = this.hdCtxIndex;
57239         var cm = this.cm, ds = this.ds;
57240         switch(item.id){
57241             case "asc":
57242                 ds.sort(cm.getDataIndex(index), "ASC");
57243                 break;
57244             case "desc":
57245                 ds.sort(cm.getDataIndex(index), "DESC");
57246                 break;
57247             case "lock":
57248                 var lc = cm.getLockedCount();
57249                 if(cm.getColumnCount(true) <= lc+1){
57250                     this.onDenyColumnLock();
57251                     return;
57252                 }
57253                 if(lc != index){
57254                     cm.setLocked(index, true, true);
57255                     cm.moveColumn(index, lc);
57256                     this.grid.fireEvent("columnmove", index, lc);
57257                 }else{
57258                     cm.setLocked(index, true);
57259                 }
57260             break;
57261             case "unlock":
57262                 var lc = cm.getLockedCount();
57263                 if((lc-1) != index){
57264                     cm.setLocked(index, false, true);
57265                     cm.moveColumn(index, lc-1);
57266                     this.grid.fireEvent("columnmove", index, lc-1);
57267                 }else{
57268                     cm.setLocked(index, false);
57269                 }
57270             break;
57271             case 'wider': // used to expand cols on touch..
57272             case 'narrow':
57273                 var cw = cm.getColumnWidth(index);
57274                 cw += (item.id == 'wider' ? 1 : -1) * 50;
57275                 cw = Math.max(0, cw);
57276                 cw = Math.min(cw,4000);
57277                 cm.setColumnWidth(index, cw);
57278                 break;
57279                 
57280             default:
57281                 index = cm.getIndexById(item.id.substr(4));
57282                 if(index != -1){
57283                     if(item.checked && cm.getColumnCount(true) <= 1){
57284                         this.onDenyColumnHide();
57285                         return false;
57286                     }
57287                     cm.setHidden(index, item.checked);
57288                 }
57289         }
57290         return true;
57291     },
57292
57293     beforeColMenuShow : function(){
57294         var cm = this.cm,  colCount = cm.getColumnCount();
57295         this.colMenu.removeAll();
57296         for(var i = 0; i < colCount; i++){
57297             this.colMenu.add(new Roo.menu.CheckItem({
57298                 id: "col-"+cm.getColumnId(i),
57299                 text: cm.getColumnHeader(i),
57300                 checked: !cm.isHidden(i),
57301                 hideOnClick:false
57302             }));
57303         }
57304     },
57305
57306     handleHdCtx : function(g, index, e){
57307         e.stopEvent();
57308         var hd = this.getHeaderCell(index);
57309         this.hdCtxIndex = index;
57310         var ms = this.hmenu.items, cm = this.cm;
57311         ms.get("asc").setDisabled(!cm.isSortable(index));
57312         ms.get("desc").setDisabled(!cm.isSortable(index));
57313         if(this.grid.enableColLock !== false){
57314             ms.get("lock").setDisabled(cm.isLocked(index));
57315             ms.get("unlock").setDisabled(!cm.isLocked(index));
57316         }
57317         this.hmenu.show(hd, "tl-bl");
57318     },
57319
57320     handleHdOver : function(e){
57321         var hd = this.findHeaderCell(e.getTarget());
57322         if(hd && !this.headersDisabled){
57323             if(this.grid.colModel.isSortable(this.getCellIndex(hd))){
57324                this.fly(hd).addClass("x-grid-hd-over");
57325             }
57326         }
57327     },
57328
57329     handleHdOut : function(e){
57330         var hd = this.findHeaderCell(e.getTarget());
57331         if(hd){
57332             this.fly(hd).removeClass("x-grid-hd-over");
57333         }
57334     },
57335
57336     handleSplitDblClick : function(e, t){
57337         var i = this.getCellIndex(t);
57338         if(this.grid.enableColumnResize !== false && this.cm.isResizable(i) && !this.cm.isFixed(i)){
57339             this.autoSizeColumn(i, true);
57340             this.layout();
57341         }
57342     },
57343
57344     render : function(){
57345
57346         var cm = this.cm;
57347         var colCount = cm.getColumnCount();
57348
57349         if(this.grid.monitorWindowResize === true){
57350             Roo.EventManager.onWindowResize(this.onWindowResize, this, true);
57351         }
57352         var header = this.renderHeaders();
57353         var body = this.templates.body.apply({rows:""});
57354         var html = this.templates.master.apply({
57355             lockedBody: body,
57356             body: body,
57357             lockedHeader: header[0],
57358             header: header[1]
57359         });
57360
57361         //this.updateColumns();
57362
57363         this.grid.getGridEl().dom.innerHTML = html;
57364
57365         this.initElements();
57366         
57367         // a kludge to fix the random scolling effect in webkit
57368         this.el.on("scroll", function() {
57369             this.el.dom.scrollTop=0; // hopefully not recursive..
57370         },this);
57371
57372         this.scroller.on("scroll", this.handleScroll, this);
57373         this.lockedBody.on("mousewheel", this.handleWheel, this);
57374         this.mainBody.on("mousewheel", this.handleWheel, this);
57375
57376         this.mainHd.on("mouseover", this.handleHdOver, this);
57377         this.mainHd.on("mouseout", this.handleHdOut, this);
57378         this.mainHd.on("dblclick", this.handleSplitDblClick, this,
57379                 {delegate: "."+this.splitClass});
57380
57381         this.lockedHd.on("mouseover", this.handleHdOver, this);
57382         this.lockedHd.on("mouseout", this.handleHdOut, this);
57383         this.lockedHd.on("dblclick", this.handleSplitDblClick, this,
57384                 {delegate: "."+this.splitClass});
57385
57386         if(this.grid.enableColumnResize !== false && Roo.grid.SplitDragZone){
57387             new Roo.grid.SplitDragZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57388         }
57389
57390         this.updateSplitters();
57391
57392         if(this.grid.enableColumnMove && Roo.grid.HeaderDragZone){
57393             new Roo.grid.HeaderDragZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57394             new Roo.grid.HeaderDropZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57395         }
57396
57397         if(this.grid.enableCtxMenu !== false && Roo.menu.Menu){
57398             this.hmenu = new Roo.menu.Menu({id: this.grid.id + "-hctx"});
57399             this.hmenu.add(
57400                 {id:"asc", text: this.sortAscText, cls: "xg-hmenu-sort-asc"},
57401                 {id:"desc", text: this.sortDescText, cls: "xg-hmenu-sort-desc"}
57402             );
57403             if(this.grid.enableColLock !== false){
57404                 this.hmenu.add('-',
57405                     {id:"lock", text: this.lockText, cls: "xg-hmenu-lock"},
57406                     {id:"unlock", text: this.unlockText, cls: "xg-hmenu-unlock"}
57407                 );
57408             }
57409             if (Roo.isTouch) {
57410                  this.hmenu.add('-',
57411                     {id:"wider", text: this.columnsWiderText},
57412                     {id:"narrow", text: this.columnsNarrowText }
57413                 );
57414                 
57415                  
57416             }
57417             
57418             if(this.grid.enableColumnHide !== false){
57419
57420                 this.colMenu = new Roo.menu.Menu({id:this.grid.id + "-hcols-menu"});
57421                 this.colMenu.on("beforeshow", this.beforeColMenuShow, this);
57422                 this.colMenu.on("itemclick", this.handleHdMenuClick, this);
57423
57424                 this.hmenu.add('-',
57425                     {id:"columns", text: this.columnsText, menu: this.colMenu}
57426                 );
57427             }
57428             this.hmenu.on("itemclick", this.handleHdMenuClick, this);
57429
57430             this.grid.on("headercontextmenu", this.handleHdCtx, this);
57431         }
57432
57433         if((this.grid.enableDragDrop || this.grid.enableDrag) && Roo.grid.GridDragZone){
57434             this.dd = new Roo.grid.GridDragZone(this.grid, {
57435                 ddGroup : this.grid.ddGroup || 'GridDD'
57436             });
57437             
57438         }
57439
57440         /*
57441         for(var i = 0; i < colCount; i++){
57442             if(cm.isHidden(i)){
57443                 this.hideColumn(i);
57444             }
57445             if(cm.config[i].align){
57446                 this.css.updateRule(this.colSelector + i, "textAlign", cm.config[i].align);
57447                 this.css.updateRule(this.hdSelector + i, "textAlign", cm.config[i].align);
57448             }
57449         }*/
57450         
57451         this.updateHeaderSortState();
57452
57453         this.beforeInitialResize();
57454         this.layout(true);
57455
57456         // two part rendering gives faster view to the user
57457         this.renderPhase2.defer(1, this);
57458     },
57459
57460     renderPhase2 : function(){
57461         // render the rows now
57462         this.refresh();
57463         if(this.grid.autoSizeColumns){
57464             this.autoSizeColumns();
57465         }
57466     },
57467
57468     beforeInitialResize : function(){
57469
57470     },
57471
57472     onColumnSplitterMoved : function(i, w){
57473         this.userResized = true;
57474         var cm = this.grid.colModel;
57475         cm.setColumnWidth(i, w, true);
57476         var cid = cm.getColumnId(i);
57477         this.css.updateRule(this.colSelector + this.idToCssName(cid), "width", (w-this.borderWidth) + "px");
57478         this.css.updateRule(this.hdSelector + this.idToCssName(cid), "width", (w-this.borderWidth) + "px");
57479         this.updateSplitters();
57480         this.layout();
57481         this.grid.fireEvent("columnresize", i, w);
57482     },
57483
57484     syncRowHeights : function(startIndex, endIndex){
57485         if(this.grid.enableRowHeightSync === true && this.cm.getLockedCount() > 0){
57486             startIndex = startIndex || 0;
57487             var mrows = this.getBodyTable().rows;
57488             var lrows = this.getLockedTable().rows;
57489             var len = mrows.length-1;
57490             endIndex = Math.min(endIndex || len, len);
57491             for(var i = startIndex; i <= endIndex; i++){
57492                 var m = mrows[i], l = lrows[i];
57493                 var h = Math.max(m.offsetHeight, l.offsetHeight);
57494                 m.style.height = l.style.height = h + "px";
57495             }
57496         }
57497     },
57498
57499     layout : function(initialRender, is2ndPass){
57500         var g = this.grid;
57501         var auto = g.autoHeight;
57502         var scrollOffset = 16;
57503         var c = g.getGridEl(), cm = this.cm,
57504                 expandCol = g.autoExpandColumn,
57505                 gv = this;
57506         //c.beginMeasure();
57507
57508         if(!c.dom.offsetWidth){ // display:none?
57509             if(initialRender){
57510                 this.lockedWrap.show();
57511                 this.mainWrap.show();
57512             }
57513             return;
57514         }
57515
57516         var hasLock = this.cm.isLocked(0);
57517
57518         var tbh = this.headerPanel.getHeight();
57519         var bbh = this.footerPanel.getHeight();
57520
57521         if(auto){
57522             var ch = this.getBodyTable().offsetHeight + tbh + bbh + this.mainHd.getHeight();
57523             var newHeight = ch + c.getBorderWidth("tb");
57524             if(g.maxHeight){
57525                 newHeight = Math.min(g.maxHeight, newHeight);
57526             }
57527             c.setHeight(newHeight);
57528         }
57529
57530         if(g.autoWidth){
57531             c.setWidth(cm.getTotalWidth()+c.getBorderWidth('lr'));
57532         }
57533
57534         var s = this.scroller;
57535
57536         var csize = c.getSize(true);
57537
57538         this.el.setSize(csize.width, csize.height);
57539
57540         this.headerPanel.setWidth(csize.width);
57541         this.footerPanel.setWidth(csize.width);
57542
57543         var hdHeight = this.mainHd.getHeight();
57544         var vw = csize.width;
57545         var vh = csize.height - (tbh + bbh);
57546
57547         s.setSize(vw, vh);
57548
57549         var bt = this.getBodyTable();
57550         
57551         if(cm.getLockedCount() == cm.config.length){
57552             bt = this.getLockedTable();
57553         }
57554         
57555         var ltWidth = hasLock ?
57556                       Math.max(this.getLockedTable().offsetWidth, this.lockedHd.dom.firstChild.offsetWidth) : 0;
57557
57558         var scrollHeight = bt.offsetHeight;
57559         var scrollWidth = ltWidth + bt.offsetWidth;
57560         var vscroll = false, hscroll = false;
57561
57562         this.scrollSizer.setSize(scrollWidth, scrollHeight+hdHeight);
57563
57564         var lw = this.lockedWrap, mw = this.mainWrap;
57565         var lb = this.lockedBody, mb = this.mainBody;
57566
57567         setTimeout(function(){
57568             var t = s.dom.offsetTop;
57569             var w = s.dom.clientWidth,
57570                 h = s.dom.clientHeight;
57571
57572             lw.setTop(t);
57573             lw.setSize(ltWidth, h);
57574
57575             mw.setLeftTop(ltWidth, t);
57576             mw.setSize(w-ltWidth, h);
57577
57578             lb.setHeight(h-hdHeight);
57579             mb.setHeight(h-hdHeight);
57580
57581             if(is2ndPass !== true && !gv.userResized && expandCol){
57582                 // high speed resize without full column calculation
57583                 
57584                 var ci = cm.getIndexById(expandCol);
57585                 if (ci < 0) {
57586                     ci = cm.findColumnIndex(expandCol);
57587                 }
57588                 ci = Math.max(0, ci); // make sure it's got at least the first col.
57589                 var expandId = cm.getColumnId(ci);
57590                 var  tw = cm.getTotalWidth(false);
57591                 var currentWidth = cm.getColumnWidth(ci);
57592                 var cw = Math.min(Math.max(((w-tw)+currentWidth-2)-/*scrollbar*/(w <= s.dom.offsetWidth ? 0 : 18), g.autoExpandMin), g.autoExpandMax);
57593                 if(currentWidth != cw){
57594                     cm.setColumnWidth(ci, cw, true);
57595                     gv.css.updateRule(gv.colSelector+gv.idToCssName(expandId), "width", (cw - gv.borderWidth) + "px");
57596                     gv.css.updateRule(gv.hdSelector+gv.idToCssName(expandId), "width", (cw - gv.borderWidth) + "px");
57597                     gv.updateSplitters();
57598                     gv.layout(false, true);
57599                 }
57600             }
57601
57602             if(initialRender){
57603                 lw.show();
57604                 mw.show();
57605             }
57606             //c.endMeasure();
57607         }, 10);
57608     },
57609
57610     onWindowResize : function(){
57611         if(!this.grid.monitorWindowResize || this.grid.autoHeight){
57612             return;
57613         }
57614         this.layout();
57615     },
57616
57617     appendFooter : function(parentEl){
57618         return null;
57619     },
57620
57621     sortAscText : "Sort Ascending",
57622     sortDescText : "Sort Descending",
57623     lockText : "Lock Column",
57624     unlockText : "Unlock Column",
57625     columnsText : "Columns",
57626  
57627     columnsWiderText : "Wider",
57628     columnsNarrowText : "Thinner"
57629 });
57630
57631
57632 Roo.grid.GridView.ColumnDragZone = function(grid, hd){
57633     Roo.grid.GridView.ColumnDragZone.superclass.constructor.call(this, grid, hd, null);
57634     this.proxy.el.addClass('x-grid3-col-dd');
57635 };
57636
57637 Roo.extend(Roo.grid.GridView.ColumnDragZone, Roo.grid.HeaderDragZone, {
57638     handleMouseDown : function(e){
57639
57640     },
57641
57642     callHandleMouseDown : function(e){
57643         Roo.grid.GridView.ColumnDragZone.superclass.handleMouseDown.call(this, e);
57644     }
57645 });
57646 /*
57647  * Based on:
57648  * Ext JS Library 1.1.1
57649  * Copyright(c) 2006-2007, Ext JS, LLC.
57650  *
57651  * Originally Released Under LGPL - original licence link has changed is not relivant.
57652  *
57653  * Fork - LGPL
57654  * <script type="text/javascript">
57655  */
57656  
57657 // private
57658 // This is a support class used internally by the Grid components
57659 Roo.grid.SplitDragZone = function(grid, hd, hd2){
57660     this.grid = grid;
57661     this.view = grid.getView();
57662     this.proxy = this.view.resizeProxy;
57663     Roo.grid.SplitDragZone.superclass.constructor.call(this, hd,
57664         "gridSplitters" + this.grid.getGridEl().id, {
57665         dragElId : Roo.id(this.proxy.dom), resizeFrame:false
57666     });
57667     this.setHandleElId(Roo.id(hd));
57668     this.setOuterHandleElId(Roo.id(hd2));
57669     this.scroll = false;
57670 };
57671 Roo.extend(Roo.grid.SplitDragZone, Roo.dd.DDProxy, {
57672     fly: Roo.Element.fly,
57673
57674     b4StartDrag : function(x, y){
57675         this.view.headersDisabled = true;
57676         this.proxy.setHeight(this.view.mainWrap.getHeight());
57677         var w = this.cm.getColumnWidth(this.cellIndex);
57678         var minw = Math.max(w-this.grid.minColumnWidth, 0);
57679         this.resetConstraints();
57680         this.setXConstraint(minw, 1000);
57681         this.setYConstraint(0, 0);
57682         this.minX = x - minw;
57683         this.maxX = x + 1000;
57684         this.startPos = x;
57685         Roo.dd.DDProxy.prototype.b4StartDrag.call(this, x, y);
57686     },
57687
57688
57689     handleMouseDown : function(e){
57690         ev = Roo.EventObject.setEvent(e);
57691         var t = this.fly(ev.getTarget());
57692         if(t.hasClass("x-grid-split")){
57693             this.cellIndex = this.view.getCellIndex(t.dom);
57694             this.split = t.dom;
57695             this.cm = this.grid.colModel;
57696             if(this.cm.isResizable(this.cellIndex) && !this.cm.isFixed(this.cellIndex)){
57697                 Roo.grid.SplitDragZone.superclass.handleMouseDown.apply(this, arguments);
57698             }
57699         }
57700     },
57701
57702     endDrag : function(e){
57703         this.view.headersDisabled = false;
57704         var endX = Math.max(this.minX, Roo.lib.Event.getPageX(e));
57705         var diff = endX - this.startPos;
57706         this.view.onColumnSplitterMoved(this.cellIndex, this.cm.getColumnWidth(this.cellIndex)+diff);
57707     },
57708
57709     autoOffset : function(){
57710         this.setDelta(0,0);
57711     }
57712 });/*
57713  * Based on:
57714  * Ext JS Library 1.1.1
57715  * Copyright(c) 2006-2007, Ext JS, LLC.
57716  *
57717  * Originally Released Under LGPL - original licence link has changed is not relivant.
57718  *
57719  * Fork - LGPL
57720  * <script type="text/javascript">
57721  */
57722  
57723 // private
57724 // This is a support class used internally by the Grid components
57725 Roo.grid.GridDragZone = function(grid, config){
57726     this.view = grid.getView();
57727     Roo.grid.GridDragZone.superclass.constructor.call(this, this.view.mainBody.dom, config);
57728     if(this.view.lockedBody){
57729         this.setHandleElId(Roo.id(this.view.mainBody.dom));
57730         this.setOuterHandleElId(Roo.id(this.view.lockedBody.dom));
57731     }
57732     this.scroll = false;
57733     this.grid = grid;
57734     this.ddel = document.createElement('div');
57735     this.ddel.className = 'x-grid-dd-wrap';
57736 };
57737
57738 Roo.extend(Roo.grid.GridDragZone, Roo.dd.DragZone, {
57739     ddGroup : "GridDD",
57740
57741     getDragData : function(e){
57742         var t = Roo.lib.Event.getTarget(e);
57743         var rowIndex = this.view.findRowIndex(t);
57744         var sm = this.grid.selModel;
57745             
57746         //Roo.log(rowIndex);
57747         
57748         if (sm.getSelectedCell) {
57749             // cell selection..
57750             if (!sm.getSelectedCell()) {
57751                 return false;
57752             }
57753             if (rowIndex != sm.getSelectedCell()[0]) {
57754                 return false;
57755             }
57756         
57757         }
57758         
57759         if(rowIndex !== false){
57760             
57761             // if editorgrid.. 
57762             
57763             
57764             //Roo.log([ sm.getSelectedCell() ? sm.getSelectedCell()[0] : 'NO' , rowIndex ]);
57765                
57766             //if(!sm.isSelected(rowIndex) || e.hasModifier()){
57767               //  
57768             //}
57769             if (e.hasModifier()){
57770                 sm.handleMouseDown(e, t); // non modifier buttons are handled by row select.
57771             }
57772             
57773             Roo.log("getDragData");
57774             
57775             return {
57776                 grid: this.grid,
57777                 ddel: this.ddel,
57778                 rowIndex: rowIndex,
57779                 selections:sm.getSelections ? sm.getSelections() : (
57780                     sm.getSelectedCell() ? [ this.grid.ds.getAt(sm.getSelectedCell()[0]) ] : []
57781                 )
57782             };
57783         }
57784         return false;
57785     },
57786
57787     onInitDrag : function(e){
57788         var data = this.dragData;
57789         this.ddel.innerHTML = this.grid.getDragDropText();
57790         this.proxy.update(this.ddel);
57791         // fire start drag?
57792     },
57793
57794     afterRepair : function(){
57795         this.dragging = false;
57796     },
57797
57798     getRepairXY : function(e, data){
57799         return false;
57800     },
57801
57802     onEndDrag : function(data, e){
57803         // fire end drag?
57804     },
57805
57806     onValidDrop : function(dd, e, id){
57807         // fire drag drop?
57808         this.hideProxy();
57809     },
57810
57811     beforeInvalidDrop : function(e, id){
57812
57813     }
57814 });/*
57815  * Based on:
57816  * Ext JS Library 1.1.1
57817  * Copyright(c) 2006-2007, Ext JS, LLC.
57818  *
57819  * Originally Released Under LGPL - original licence link has changed is not relivant.
57820  *
57821  * Fork - LGPL
57822  * <script type="text/javascript">
57823  */
57824  
57825
57826 /**
57827  * @class Roo.grid.ColumnModel
57828  * @extends Roo.util.Observable
57829  * This is the default implementation of a ColumnModel used by the Grid. It defines
57830  * the columns in the grid.
57831  * <br>Usage:<br>
57832  <pre><code>
57833  var colModel = new Roo.grid.ColumnModel([
57834         {header: "Ticker", width: 60, sortable: true, locked: true},
57835         {header: "Company Name", width: 150, sortable: true},
57836         {header: "Market Cap.", width: 100, sortable: true},
57837         {header: "$ Sales", width: 100, sortable: true, renderer: money},
57838         {header: "Employees", width: 100, sortable: true, resizable: false}
57839  ]);
57840  </code></pre>
57841  * <p>
57842  
57843  * The config options listed for this class are options which may appear in each
57844  * individual column definition.
57845  * <br/>RooJS Fix - column id's are not sequential but use Roo.id() - fixes bugs with layouts.
57846  * @constructor
57847  * @param {Object} config An Array of column config objects. See this class's
57848  * config objects for details.
57849 */
57850 Roo.grid.ColumnModel = function(config){
57851         /**
57852      * The config passed into the constructor
57853      */
57854     this.config = config;
57855     this.lookup = {};
57856
57857     // if no id, create one
57858     // if the column does not have a dataIndex mapping,
57859     // map it to the order it is in the config
57860     for(var i = 0, len = config.length; i < len; i++){
57861         var c = config[i];
57862         if(typeof c.dataIndex == "undefined"){
57863             c.dataIndex = i;
57864         }
57865         if(typeof c.renderer == "string"){
57866             c.renderer = Roo.util.Format[c.renderer];
57867         }
57868         if(typeof c.id == "undefined"){
57869             c.id = Roo.id();
57870         }
57871         if(c.editor && c.editor.xtype){
57872             c.editor  = Roo.factory(c.editor, Roo.grid);
57873         }
57874         if(c.editor && c.editor.isFormField){
57875             c.editor = new Roo.grid.GridEditor(c.editor);
57876         }
57877         this.lookup[c.id] = c;
57878     }
57879
57880     /**
57881      * The width of columns which have no width specified (defaults to 100)
57882      * @type Number
57883      */
57884     this.defaultWidth = 100;
57885
57886     /**
57887      * Default sortable of columns which have no sortable specified (defaults to false)
57888      * @type Boolean
57889      */
57890     this.defaultSortable = false;
57891
57892     this.addEvents({
57893         /**
57894              * @event widthchange
57895              * Fires when the width of a column changes.
57896              * @param {ColumnModel} this
57897              * @param {Number} columnIndex The column index
57898              * @param {Number} newWidth The new width
57899              */
57900             "widthchange": true,
57901         /**
57902              * @event headerchange
57903              * Fires when the text of a header changes.
57904              * @param {ColumnModel} this
57905              * @param {Number} columnIndex The column index
57906              * @param {Number} newText The new header text
57907              */
57908             "headerchange": true,
57909         /**
57910              * @event hiddenchange
57911              * Fires when a column is hidden or "unhidden".
57912              * @param {ColumnModel} this
57913              * @param {Number} columnIndex The column index
57914              * @param {Boolean} hidden true if hidden, false otherwise
57915              */
57916             "hiddenchange": true,
57917             /**
57918          * @event columnmoved
57919          * Fires when a column is moved.
57920          * @param {ColumnModel} this
57921          * @param {Number} oldIndex
57922          * @param {Number} newIndex
57923          */
57924         "columnmoved" : true,
57925         /**
57926          * @event columlockchange
57927          * Fires when a column's locked state is changed
57928          * @param {ColumnModel} this
57929          * @param {Number} colIndex
57930          * @param {Boolean} locked true if locked
57931          */
57932         "columnlockchange" : true
57933     });
57934     Roo.grid.ColumnModel.superclass.constructor.call(this);
57935 };
57936 Roo.extend(Roo.grid.ColumnModel, Roo.util.Observable, {
57937     /**
57938      * @cfg {String} header The header text to display in the Grid view.
57939      */
57940     /**
57941      * @cfg {String} dataIndex (Optional) The name of the field in the grid's {@link Roo.data.Store}'s
57942      * {@link Roo.data.Record} definition from which to draw the column's value. If not
57943      * specified, the column's index is used as an index into the Record's data Array.
57944      */
57945     /**
57946      * @cfg {Number} width (Optional) The initial width in pixels of the column. Using this
57947      * instead of {@link Roo.grid.Grid#autoSizeColumns} is more efficient.
57948      */
57949     /**
57950      * @cfg {Boolean} sortable (Optional) True if sorting is to be allowed on this column.
57951      * Defaults to the value of the {@link #defaultSortable} property.
57952      * Whether local/remote sorting is used is specified in {@link Roo.data.Store#remoteSort}.
57953      */
57954     /**
57955      * @cfg {Boolean} locked (Optional) True to lock the column in place while scrolling the Grid.  Defaults to false.
57956      */
57957     /**
57958      * @cfg {Boolean} fixed (Optional) True if the column width cannot be changed.  Defaults to false.
57959      */
57960     /**
57961      * @cfg {Boolean} resizable (Optional) False to disable column resizing. Defaults to true.
57962      */
57963     /**
57964      * @cfg {Boolean} hidden (Optional) True to hide the column. Defaults to false.
57965      */
57966     /**
57967      * @cfg {Function} renderer (Optional) A function used to generate HTML markup for a cell
57968      * given the cell's data value. See {@link #setRenderer}. If not specified, the
57969      * default renderer returns the escaped data value. If an object is returned (bootstrap only)
57970      * then it is treated as a Roo Component object instance, and it is rendered after the initial row is rendered
57971      */
57972        /**
57973      * @cfg {Roo.grid.GridEditor} editor (Optional) For grid editors - returns the grid editor 
57974      */
57975     /**
57976      * @cfg {String} align (Optional) Set the CSS text-align property of the column.  Defaults to undefined.
57977      */
57978     /**
57979      * @cfg {String} valign (Optional) Set the CSS vertical-align property of the column (eg. middle, top, bottom etc).  Defaults to undefined.
57980      */
57981     /**
57982      * @cfg {String} cursor (Optional)
57983      */
57984     /**
57985      * @cfg {String} tooltip (Optional)
57986      */
57987     /**
57988      * @cfg {Number} xs (Optional)
57989      */
57990     /**
57991      * @cfg {Number} sm (Optional)
57992      */
57993     /**
57994      * @cfg {Number} md (Optional)
57995      */
57996     /**
57997      * @cfg {Number} lg (Optional)
57998      */
57999     /**
58000      * Returns the id of the column at the specified index.
58001      * @param {Number} index The column index
58002      * @return {String} the id
58003      */
58004     getColumnId : function(index){
58005         return this.config[index].id;
58006     },
58007
58008     /**
58009      * Returns the column for a specified id.
58010      * @param {String} id The column id
58011      * @return {Object} the column
58012      */
58013     getColumnById : function(id){
58014         return this.lookup[id];
58015     },
58016
58017     
58018     /**
58019      * Returns the column for a specified dataIndex.
58020      * @param {String} dataIndex The column dataIndex
58021      * @return {Object|Boolean} the column or false if not found
58022      */
58023     getColumnByDataIndex: function(dataIndex){
58024         var index = this.findColumnIndex(dataIndex);
58025         return index > -1 ? this.config[index] : false;
58026     },
58027     
58028     /**
58029      * Returns the index for a specified column id.
58030      * @param {String} id The column id
58031      * @return {Number} the index, or -1 if not found
58032      */
58033     getIndexById : function(id){
58034         for(var i = 0, len = this.config.length; i < len; i++){
58035             if(this.config[i].id == id){
58036                 return i;
58037             }
58038         }
58039         return -1;
58040     },
58041     
58042     /**
58043      * Returns the index for a specified column dataIndex.
58044      * @param {String} dataIndex The column dataIndex
58045      * @return {Number} the index, or -1 if not found
58046      */
58047     
58048     findColumnIndex : function(dataIndex){
58049         for(var i = 0, len = this.config.length; i < len; i++){
58050             if(this.config[i].dataIndex == dataIndex){
58051                 return i;
58052             }
58053         }
58054         return -1;
58055     },
58056     
58057     
58058     moveColumn : function(oldIndex, newIndex){
58059         var c = this.config[oldIndex];
58060         this.config.splice(oldIndex, 1);
58061         this.config.splice(newIndex, 0, c);
58062         this.dataMap = null;
58063         this.fireEvent("columnmoved", this, oldIndex, newIndex);
58064     },
58065
58066     isLocked : function(colIndex){
58067         return this.config[colIndex].locked === true;
58068     },
58069
58070     setLocked : function(colIndex, value, suppressEvent){
58071         if(this.isLocked(colIndex) == value){
58072             return;
58073         }
58074         this.config[colIndex].locked = value;
58075         if(!suppressEvent){
58076             this.fireEvent("columnlockchange", this, colIndex, value);
58077         }
58078     },
58079
58080     getTotalLockedWidth : function(){
58081         var totalWidth = 0;
58082         for(var i = 0; i < this.config.length; i++){
58083             if(this.isLocked(i) && !this.isHidden(i)){
58084                 this.totalWidth += this.getColumnWidth(i);
58085             }
58086         }
58087         return totalWidth;
58088     },
58089
58090     getLockedCount : function(){
58091         for(var i = 0, len = this.config.length; i < len; i++){
58092             if(!this.isLocked(i)){
58093                 return i;
58094             }
58095         }
58096         
58097         return this.config.length;
58098     },
58099
58100     /**
58101      * Returns the number of columns.
58102      * @return {Number}
58103      */
58104     getColumnCount : function(visibleOnly){
58105         if(visibleOnly === true){
58106             var c = 0;
58107             for(var i = 0, len = this.config.length; i < len; i++){
58108                 if(!this.isHidden(i)){
58109                     c++;
58110                 }
58111             }
58112             return c;
58113         }
58114         return this.config.length;
58115     },
58116
58117     /**
58118      * Returns the column configs that return true by the passed function that is called with (columnConfig, index)
58119      * @param {Function} fn
58120      * @param {Object} scope (optional)
58121      * @return {Array} result
58122      */
58123     getColumnsBy : function(fn, scope){
58124         var r = [];
58125         for(var i = 0, len = this.config.length; i < len; i++){
58126             var c = this.config[i];
58127             if(fn.call(scope||this, c, i) === true){
58128                 r[r.length] = c;
58129             }
58130         }
58131         return r;
58132     },
58133
58134     /**
58135      * Returns true if the specified column is sortable.
58136      * @param {Number} col The column index
58137      * @return {Boolean}
58138      */
58139     isSortable : function(col){
58140         if(typeof this.config[col].sortable == "undefined"){
58141             return this.defaultSortable;
58142         }
58143         return this.config[col].sortable;
58144     },
58145
58146     /**
58147      * Returns the rendering (formatting) function defined for the column.
58148      * @param {Number} col The column index.
58149      * @return {Function} The function used to render the cell. See {@link #setRenderer}.
58150      */
58151     getRenderer : function(col){
58152         if(!this.config[col].renderer){
58153             return Roo.grid.ColumnModel.defaultRenderer;
58154         }
58155         return this.config[col].renderer;
58156     },
58157
58158     /**
58159      * Sets the rendering (formatting) function for a column.
58160      * @param {Number} col The column index
58161      * @param {Function} fn The function to use to process the cell's raw data
58162      * to return HTML markup for the grid view. The render function is called with
58163      * the following parameters:<ul>
58164      * <li>Data value.</li>
58165      * <li>Cell metadata. An object in which you may set the following attributes:<ul>
58166      * <li>css A CSS style string to apply to the table cell.</li>
58167      * <li>attr An HTML attribute definition string to apply to the data container element <i>within</i> the table cell.</li></ul>
58168      * <li>The {@link Roo.data.Record} from which the data was extracted.</li>
58169      * <li>Row index</li>
58170      * <li>Column index</li>
58171      * <li>The {@link Roo.data.Store} object from which the Record was extracted</li></ul>
58172      */
58173     setRenderer : function(col, fn){
58174         this.config[col].renderer = fn;
58175     },
58176
58177     /**
58178      * Returns the width for the specified column.
58179      * @param {Number} col The column index
58180      * @return {Number}
58181      */
58182     getColumnWidth : function(col){
58183         return this.config[col].width * 1 || this.defaultWidth;
58184     },
58185
58186     /**
58187      * Sets the width for a column.
58188      * @param {Number} col The column index
58189      * @param {Number} width The new width
58190      */
58191     setColumnWidth : function(col, width, suppressEvent){
58192         this.config[col].width = width;
58193         this.totalWidth = null;
58194         if(!suppressEvent){
58195              this.fireEvent("widthchange", this, col, width);
58196         }
58197     },
58198
58199     /**
58200      * Returns the total width of all columns.
58201      * @param {Boolean} includeHidden True to include hidden column widths
58202      * @return {Number}
58203      */
58204     getTotalWidth : function(includeHidden){
58205         if(!this.totalWidth){
58206             this.totalWidth = 0;
58207             for(var i = 0, len = this.config.length; i < len; i++){
58208                 if(includeHidden || !this.isHidden(i)){
58209                     this.totalWidth += this.getColumnWidth(i);
58210                 }
58211             }
58212         }
58213         return this.totalWidth;
58214     },
58215
58216     /**
58217      * Returns the header for the specified column.
58218      * @param {Number} col The column index
58219      * @return {String}
58220      */
58221     getColumnHeader : function(col){
58222         return this.config[col].header;
58223     },
58224
58225     /**
58226      * Sets the header for a column.
58227      * @param {Number} col The column index
58228      * @param {String} header The new header
58229      */
58230     setColumnHeader : function(col, header){
58231         this.config[col].header = header;
58232         this.fireEvent("headerchange", this, col, header);
58233     },
58234
58235     /**
58236      * Returns the tooltip for the specified column.
58237      * @param {Number} col The column index
58238      * @return {String}
58239      */
58240     getColumnTooltip : function(col){
58241             return this.config[col].tooltip;
58242     },
58243     /**
58244      * Sets the tooltip for a column.
58245      * @param {Number} col The column index
58246      * @param {String} tooltip The new tooltip
58247      */
58248     setColumnTooltip : function(col, tooltip){
58249             this.config[col].tooltip = tooltip;
58250     },
58251
58252     /**
58253      * Returns the dataIndex for the specified column.
58254      * @param {Number} col The column index
58255      * @return {Number}
58256      */
58257     getDataIndex : function(col){
58258         return this.config[col].dataIndex;
58259     },
58260
58261     /**
58262      * Sets the dataIndex for a column.
58263      * @param {Number} col The column index
58264      * @param {Number} dataIndex The new dataIndex
58265      */
58266     setDataIndex : function(col, dataIndex){
58267         this.config[col].dataIndex = dataIndex;
58268     },
58269
58270     
58271     
58272     /**
58273      * Returns true if the cell is editable.
58274      * @param {Number} colIndex The column index
58275      * @param {Number} rowIndex The row index - this is nto actually used..?
58276      * @return {Boolean}
58277      */
58278     isCellEditable : function(colIndex, rowIndex){
58279         return (this.config[colIndex].editable || (typeof this.config[colIndex].editable == "undefined" && this.config[colIndex].editor)) ? true : false;
58280     },
58281
58282     /**
58283      * Returns the editor defined for the cell/column.
58284      * return false or null to disable editing.
58285      * @param {Number} colIndex The column index
58286      * @param {Number} rowIndex The row index
58287      * @return {Object}
58288      */
58289     getCellEditor : function(colIndex, rowIndex){
58290         return this.config[colIndex].editor;
58291     },
58292
58293     /**
58294      * Sets if a column is editable.
58295      * @param {Number} col The column index
58296      * @param {Boolean} editable True if the column is editable
58297      */
58298     setEditable : function(col, editable){
58299         this.config[col].editable = editable;
58300     },
58301
58302
58303     /**
58304      * Returns true if the column is hidden.
58305      * @param {Number} colIndex The column index
58306      * @return {Boolean}
58307      */
58308     isHidden : function(colIndex){
58309         return this.config[colIndex].hidden;
58310     },
58311
58312
58313     /**
58314      * Returns true if the column width cannot be changed
58315      */
58316     isFixed : function(colIndex){
58317         return this.config[colIndex].fixed;
58318     },
58319
58320     /**
58321      * Returns true if the column can be resized
58322      * @return {Boolean}
58323      */
58324     isResizable : function(colIndex){
58325         return colIndex >= 0 && this.config[colIndex].resizable !== false && this.config[colIndex].fixed !== true;
58326     },
58327     /**
58328      * Sets if a column is hidden.
58329      * @param {Number} colIndex The column index
58330      * @param {Boolean} hidden True if the column is hidden
58331      */
58332     setHidden : function(colIndex, hidden){
58333         this.config[colIndex].hidden = hidden;
58334         this.totalWidth = null;
58335         this.fireEvent("hiddenchange", this, colIndex, hidden);
58336     },
58337
58338     /**
58339      * Sets the editor for a column.
58340      * @param {Number} col The column index
58341      * @param {Object} editor The editor object
58342      */
58343     setEditor : function(col, editor){
58344         this.config[col].editor = editor;
58345     }
58346 });
58347
58348 Roo.grid.ColumnModel.defaultRenderer = function(value)
58349 {
58350     if(typeof value == "object") {
58351         return value;
58352     }
58353         if(typeof value == "string" && value.length < 1){
58354             return "&#160;";
58355         }
58356     
58357         return String.format("{0}", value);
58358 };
58359
58360 // Alias for backwards compatibility
58361 Roo.grid.DefaultColumnModel = Roo.grid.ColumnModel;
58362 /*
58363  * Based on:
58364  * Ext JS Library 1.1.1
58365  * Copyright(c) 2006-2007, Ext JS, LLC.
58366  *
58367  * Originally Released Under LGPL - original licence link has changed is not relivant.
58368  *
58369  * Fork - LGPL
58370  * <script type="text/javascript">
58371  */
58372
58373 /**
58374  * @class Roo.grid.AbstractSelectionModel
58375  * @extends Roo.util.Observable
58376  * Abstract base class for grid SelectionModels.  It provides the interface that should be
58377  * implemented by descendant classes.  This class should not be directly instantiated.
58378  * @constructor
58379  */
58380 Roo.grid.AbstractSelectionModel = function(){
58381     this.locked = false;
58382     Roo.grid.AbstractSelectionModel.superclass.constructor.call(this);
58383 };
58384
58385 Roo.extend(Roo.grid.AbstractSelectionModel, Roo.util.Observable,  {
58386     /** @ignore Called by the grid automatically. Do not call directly. */
58387     init : function(grid){
58388         this.grid = grid;
58389         this.initEvents();
58390     },
58391
58392     /**
58393      * Locks the selections.
58394      */
58395     lock : function(){
58396         this.locked = true;
58397     },
58398
58399     /**
58400      * Unlocks the selections.
58401      */
58402     unlock : function(){
58403         this.locked = false;
58404     },
58405
58406     /**
58407      * Returns true if the selections are locked.
58408      * @return {Boolean}
58409      */
58410     isLocked : function(){
58411         return this.locked;
58412     }
58413 });/*
58414  * Based on:
58415  * Ext JS Library 1.1.1
58416  * Copyright(c) 2006-2007, Ext JS, LLC.
58417  *
58418  * Originally Released Under LGPL - original licence link has changed is not relivant.
58419  *
58420  * Fork - LGPL
58421  * <script type="text/javascript">
58422  */
58423 /**
58424  * @extends Roo.grid.AbstractSelectionModel
58425  * @class Roo.grid.RowSelectionModel
58426  * The default SelectionModel used by {@link Roo.grid.Grid}.
58427  * It supports multiple selections and keyboard selection/navigation. 
58428  * @constructor
58429  * @param {Object} config
58430  */
58431 Roo.grid.RowSelectionModel = function(config){
58432     Roo.apply(this, config);
58433     this.selections = new Roo.util.MixedCollection(false, function(o){
58434         return o.id;
58435     });
58436
58437     this.last = false;
58438     this.lastActive = false;
58439
58440     this.addEvents({
58441         /**
58442              * @event selectionchange
58443              * Fires when the selection changes
58444              * @param {SelectionModel} this
58445              */
58446             "selectionchange" : true,
58447         /**
58448              * @event afterselectionchange
58449              * Fires after the selection changes (eg. by key press or clicking)
58450              * @param {SelectionModel} this
58451              */
58452             "afterselectionchange" : true,
58453         /**
58454              * @event beforerowselect
58455              * Fires when a row is selected being selected, return false to cancel.
58456              * @param {SelectionModel} this
58457              * @param {Number} rowIndex The selected index
58458              * @param {Boolean} keepExisting False if other selections will be cleared
58459              */
58460             "beforerowselect" : true,
58461         /**
58462              * @event rowselect
58463              * Fires when a row is selected.
58464              * @param {SelectionModel} this
58465              * @param {Number} rowIndex The selected index
58466              * @param {Roo.data.Record} r The record
58467              */
58468             "rowselect" : true,
58469         /**
58470              * @event rowdeselect
58471              * Fires when a row is deselected.
58472              * @param {SelectionModel} this
58473              * @param {Number} rowIndex The selected index
58474              */
58475         "rowdeselect" : true
58476     });
58477     Roo.grid.RowSelectionModel.superclass.constructor.call(this);
58478     this.locked = false;
58479 };
58480
58481 Roo.extend(Roo.grid.RowSelectionModel, Roo.grid.AbstractSelectionModel,  {
58482     /**
58483      * @cfg {Boolean} singleSelect
58484      * True to allow selection of only one row at a time (defaults to false)
58485      */
58486     singleSelect : false,
58487
58488     // private
58489     initEvents : function(){
58490
58491         if(!this.grid.enableDragDrop && !this.grid.enableDrag){
58492             this.grid.on("mousedown", this.handleMouseDown, this);
58493         }else{ // allow click to work like normal
58494             this.grid.on("rowclick", this.handleDragableRowClick, this);
58495         }
58496
58497         this.rowNav = new Roo.KeyNav(this.grid.getGridEl(), {
58498             "up" : function(e){
58499                 if(!e.shiftKey){
58500                     this.selectPrevious(e.shiftKey);
58501                 }else if(this.last !== false && this.lastActive !== false){
58502                     var last = this.last;
58503                     this.selectRange(this.last,  this.lastActive-1);
58504                     this.grid.getView().focusRow(this.lastActive);
58505                     if(last !== false){
58506                         this.last = last;
58507                     }
58508                 }else{
58509                     this.selectFirstRow();
58510                 }
58511                 this.fireEvent("afterselectionchange", this);
58512             },
58513             "down" : function(e){
58514                 if(!e.shiftKey){
58515                     this.selectNext(e.shiftKey);
58516                 }else if(this.last !== false && this.lastActive !== false){
58517                     var last = this.last;
58518                     this.selectRange(this.last,  this.lastActive+1);
58519                     this.grid.getView().focusRow(this.lastActive);
58520                     if(last !== false){
58521                         this.last = last;
58522                     }
58523                 }else{
58524                     this.selectFirstRow();
58525                 }
58526                 this.fireEvent("afterselectionchange", this);
58527             },
58528             scope: this
58529         });
58530
58531         var view = this.grid.view;
58532         view.on("refresh", this.onRefresh, this);
58533         view.on("rowupdated", this.onRowUpdated, this);
58534         view.on("rowremoved", this.onRemove, this);
58535     },
58536
58537     // private
58538     onRefresh : function(){
58539         var ds = this.grid.dataSource, i, v = this.grid.view;
58540         var s = this.selections;
58541         s.each(function(r){
58542             if((i = ds.indexOfId(r.id)) != -1){
58543                 v.onRowSelect(i);
58544                 s.add(ds.getAt(i)); // updating the selection relate data
58545             }else{
58546                 s.remove(r);
58547             }
58548         });
58549     },
58550
58551     // private
58552     onRemove : function(v, index, r){
58553         this.selections.remove(r);
58554     },
58555
58556     // private
58557     onRowUpdated : function(v, index, r){
58558         if(this.isSelected(r)){
58559             v.onRowSelect(index);
58560         }
58561     },
58562
58563     /**
58564      * Select records.
58565      * @param {Array} records The records to select
58566      * @param {Boolean} keepExisting (optional) True to keep existing selections
58567      */
58568     selectRecords : function(records, keepExisting){
58569         if(!keepExisting){
58570             this.clearSelections();
58571         }
58572         var ds = this.grid.dataSource;
58573         for(var i = 0, len = records.length; i < len; i++){
58574             this.selectRow(ds.indexOf(records[i]), true);
58575         }
58576     },
58577
58578     /**
58579      * Gets the number of selected rows.
58580      * @return {Number}
58581      */
58582     getCount : function(){
58583         return this.selections.length;
58584     },
58585
58586     /**
58587      * Selects the first row in the grid.
58588      */
58589     selectFirstRow : function(){
58590         this.selectRow(0);
58591     },
58592
58593     /**
58594      * Select the last row.
58595      * @param {Boolean} keepExisting (optional) True to keep existing selections
58596      */
58597     selectLastRow : function(keepExisting){
58598         this.selectRow(this.grid.dataSource.getCount() - 1, keepExisting);
58599     },
58600
58601     /**
58602      * Selects the row immediately following the last selected row.
58603      * @param {Boolean} keepExisting (optional) True to keep existing selections
58604      */
58605     selectNext : function(keepExisting){
58606         if(this.last !== false && (this.last+1) < this.grid.dataSource.getCount()){
58607             this.selectRow(this.last+1, keepExisting);
58608             this.grid.getView().focusRow(this.last);
58609         }
58610     },
58611
58612     /**
58613      * Selects the row that precedes the last selected row.
58614      * @param {Boolean} keepExisting (optional) True to keep existing selections
58615      */
58616     selectPrevious : function(keepExisting){
58617         if(this.last){
58618             this.selectRow(this.last-1, keepExisting);
58619             this.grid.getView().focusRow(this.last);
58620         }
58621     },
58622
58623     /**
58624      * Returns the selected records
58625      * @return {Array} Array of selected records
58626      */
58627     getSelections : function(){
58628         return [].concat(this.selections.items);
58629     },
58630
58631     /**
58632      * Returns the first selected record.
58633      * @return {Record}
58634      */
58635     getSelected : function(){
58636         return this.selections.itemAt(0);
58637     },
58638
58639
58640     /**
58641      * Clears all selections.
58642      */
58643     clearSelections : function(fast){
58644         if(this.locked) {
58645             return;
58646         }
58647         if(fast !== true){
58648             var ds = this.grid.dataSource;
58649             var s = this.selections;
58650             s.each(function(r){
58651                 this.deselectRow(ds.indexOfId(r.id));
58652             }, this);
58653             s.clear();
58654         }else{
58655             this.selections.clear();
58656         }
58657         this.last = false;
58658     },
58659
58660
58661     /**
58662      * Selects all rows.
58663      */
58664     selectAll : function(){
58665         if(this.locked) {
58666             return;
58667         }
58668         this.selections.clear();
58669         for(var i = 0, len = this.grid.dataSource.getCount(); i < len; i++){
58670             this.selectRow(i, true);
58671         }
58672     },
58673
58674     /**
58675      * Returns True if there is a selection.
58676      * @return {Boolean}
58677      */
58678     hasSelection : function(){
58679         return this.selections.length > 0;
58680     },
58681
58682     /**
58683      * Returns True if the specified row is selected.
58684      * @param {Number/Record} record The record or index of the record to check
58685      * @return {Boolean}
58686      */
58687     isSelected : function(index){
58688         var r = typeof index == "number" ? this.grid.dataSource.getAt(index) : index;
58689         return (r && this.selections.key(r.id) ? true : false);
58690     },
58691
58692     /**
58693      * Returns True if the specified record id is selected.
58694      * @param {String} id The id of record to check
58695      * @return {Boolean}
58696      */
58697     isIdSelected : function(id){
58698         return (this.selections.key(id) ? true : false);
58699     },
58700
58701     // private
58702     handleMouseDown : function(e, t){
58703         var view = this.grid.getView(), rowIndex;
58704         if(this.isLocked() || (rowIndex = view.findRowIndex(t)) === false){
58705             return;
58706         };
58707         if(e.shiftKey && this.last !== false){
58708             var last = this.last;
58709             this.selectRange(last, rowIndex, e.ctrlKey);
58710             this.last = last; // reset the last
58711             view.focusRow(rowIndex);
58712         }else{
58713             var isSelected = this.isSelected(rowIndex);
58714             if(e.button !== 0 && isSelected){
58715                 view.focusRow(rowIndex);
58716             }else if(e.ctrlKey && isSelected){
58717                 this.deselectRow(rowIndex);
58718             }else if(!isSelected){
58719                 this.selectRow(rowIndex, e.button === 0 && (e.ctrlKey || e.shiftKey));
58720                 view.focusRow(rowIndex);
58721             }
58722         }
58723         this.fireEvent("afterselectionchange", this);
58724     },
58725     // private
58726     handleDragableRowClick :  function(grid, rowIndex, e) 
58727     {
58728         if(e.button === 0 && !e.shiftKey && !e.ctrlKey) {
58729             this.selectRow(rowIndex, false);
58730             grid.view.focusRow(rowIndex);
58731              this.fireEvent("afterselectionchange", this);
58732         }
58733     },
58734     
58735     /**
58736      * Selects multiple rows.
58737      * @param {Array} rows Array of the indexes of the row to select
58738      * @param {Boolean} keepExisting (optional) True to keep existing selections
58739      */
58740     selectRows : function(rows, keepExisting){
58741         if(!keepExisting){
58742             this.clearSelections();
58743         }
58744         for(var i = 0, len = rows.length; i < len; i++){
58745             this.selectRow(rows[i], true);
58746         }
58747     },
58748
58749     /**
58750      * Selects a range of rows. All rows in between startRow and endRow are also selected.
58751      * @param {Number} startRow The index of the first row in the range
58752      * @param {Number} endRow The index of the last row in the range
58753      * @param {Boolean} keepExisting (optional) True to retain existing selections
58754      */
58755     selectRange : function(startRow, endRow, keepExisting){
58756         if(this.locked) {
58757             return;
58758         }
58759         if(!keepExisting){
58760             this.clearSelections();
58761         }
58762         if(startRow <= endRow){
58763             for(var i = startRow; i <= endRow; i++){
58764                 this.selectRow(i, true);
58765             }
58766         }else{
58767             for(var i = startRow; i >= endRow; i--){
58768                 this.selectRow(i, true);
58769             }
58770         }
58771     },
58772
58773     /**
58774      * Deselects a range of rows. All rows in between startRow and endRow are also deselected.
58775      * @param {Number} startRow The index of the first row in the range
58776      * @param {Number} endRow The index of the last row in the range
58777      */
58778     deselectRange : function(startRow, endRow, preventViewNotify){
58779         if(this.locked) {
58780             return;
58781         }
58782         for(var i = startRow; i <= endRow; i++){
58783             this.deselectRow(i, preventViewNotify);
58784         }
58785     },
58786
58787     /**
58788      * Selects a row.
58789      * @param {Number} row The index of the row to select
58790      * @param {Boolean} keepExisting (optional) True to keep existing selections
58791      */
58792     selectRow : function(index, keepExisting, preventViewNotify){
58793         if(this.locked || (index < 0 || index >= this.grid.dataSource.getCount())) {
58794             return;
58795         }
58796         if(this.fireEvent("beforerowselect", this, index, keepExisting) !== false){
58797             if(!keepExisting || this.singleSelect){
58798                 this.clearSelections();
58799             }
58800             var r = this.grid.dataSource.getAt(index);
58801             this.selections.add(r);
58802             this.last = this.lastActive = index;
58803             if(!preventViewNotify){
58804                 this.grid.getView().onRowSelect(index);
58805             }
58806             this.fireEvent("rowselect", this, index, r);
58807             this.fireEvent("selectionchange", this);
58808         }
58809     },
58810
58811     /**
58812      * Deselects a row.
58813      * @param {Number} row The index of the row to deselect
58814      */
58815     deselectRow : function(index, preventViewNotify){
58816         if(this.locked) {
58817             return;
58818         }
58819         if(this.last == index){
58820             this.last = false;
58821         }
58822         if(this.lastActive == index){
58823             this.lastActive = false;
58824         }
58825         var r = this.grid.dataSource.getAt(index);
58826         this.selections.remove(r);
58827         if(!preventViewNotify){
58828             this.grid.getView().onRowDeselect(index);
58829         }
58830         this.fireEvent("rowdeselect", this, index);
58831         this.fireEvent("selectionchange", this);
58832     },
58833
58834     // private
58835     restoreLast : function(){
58836         if(this._last){
58837             this.last = this._last;
58838         }
58839     },
58840
58841     // private
58842     acceptsNav : function(row, col, cm){
58843         return !cm.isHidden(col) && cm.isCellEditable(col, row);
58844     },
58845
58846     // private
58847     onEditorKey : function(field, e){
58848         var k = e.getKey(), newCell, g = this.grid, ed = g.activeEditor;
58849         if(k == e.TAB){
58850             e.stopEvent();
58851             ed.completeEdit();
58852             if(e.shiftKey){
58853                 newCell = g.walkCells(ed.row, ed.col-1, -1, this.acceptsNav, this);
58854             }else{
58855                 newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
58856             }
58857         }else if(k == e.ENTER && !e.ctrlKey){
58858             e.stopEvent();
58859             ed.completeEdit();
58860             if(e.shiftKey){
58861                 newCell = g.walkCells(ed.row-1, ed.col, -1, this.acceptsNav, this);
58862             }else{
58863                 newCell = g.walkCells(ed.row+1, ed.col, 1, this.acceptsNav, this);
58864             }
58865         }else if(k == e.ESC){
58866             ed.cancelEdit();
58867         }
58868         if(newCell){
58869             g.startEditing(newCell[0], newCell[1]);
58870         }
58871     }
58872 });/*
58873  * Based on:
58874  * Ext JS Library 1.1.1
58875  * Copyright(c) 2006-2007, Ext JS, LLC.
58876  *
58877  * Originally Released Under LGPL - original licence link has changed is not relivant.
58878  *
58879  * Fork - LGPL
58880  * <script type="text/javascript">
58881  */
58882 /**
58883  * @class Roo.grid.CellSelectionModel
58884  * @extends Roo.grid.AbstractSelectionModel
58885  * This class provides the basic implementation for cell selection in a grid.
58886  * @constructor
58887  * @param {Object} config The object containing the configuration of this model.
58888  * @cfg {Boolean} enter_is_tab Enter behaves the same as tab. (eg. goes to next cell) default: false
58889  */
58890 Roo.grid.CellSelectionModel = function(config){
58891     Roo.apply(this, config);
58892
58893     this.selection = null;
58894
58895     this.addEvents({
58896         /**
58897              * @event beforerowselect
58898              * Fires before a cell is selected.
58899              * @param {SelectionModel} this
58900              * @param {Number} rowIndex The selected row index
58901              * @param {Number} colIndex The selected cell index
58902              */
58903             "beforecellselect" : true,
58904         /**
58905              * @event cellselect
58906              * Fires when a cell is selected.
58907              * @param {SelectionModel} this
58908              * @param {Number} rowIndex The selected row index
58909              * @param {Number} colIndex The selected cell index
58910              */
58911             "cellselect" : true,
58912         /**
58913              * @event selectionchange
58914              * Fires when the active selection changes.
58915              * @param {SelectionModel} this
58916              * @param {Object} selection null for no selection or an object (o) with two properties
58917                 <ul>
58918                 <li>o.record: the record object for the row the selection is in</li>
58919                 <li>o.cell: An array of [rowIndex, columnIndex]</li>
58920                 </ul>
58921              */
58922             "selectionchange" : true,
58923         /**
58924              * @event tabend
58925              * Fires when the tab (or enter) was pressed on the last editable cell
58926              * You can use this to trigger add new row.
58927              * @param {SelectionModel} this
58928              */
58929             "tabend" : true,
58930          /**
58931              * @event beforeeditnext
58932              * Fires before the next editable sell is made active
58933              * You can use this to skip to another cell or fire the tabend
58934              *    if you set cell to false
58935              * @param {Object} eventdata object : { cell : [ row, col ] } 
58936              */
58937             "beforeeditnext" : true
58938     });
58939     Roo.grid.CellSelectionModel.superclass.constructor.call(this);
58940 };
58941
58942 Roo.extend(Roo.grid.CellSelectionModel, Roo.grid.AbstractSelectionModel,  {
58943     
58944     enter_is_tab: false,
58945
58946     /** @ignore */
58947     initEvents : function(){
58948         this.grid.on("mousedown", this.handleMouseDown, this);
58949         this.grid.getGridEl().on(Roo.isIE ? "keydown" : "keypress", this.handleKeyDown, this);
58950         var view = this.grid.view;
58951         view.on("refresh", this.onViewChange, this);
58952         view.on("rowupdated", this.onRowUpdated, this);
58953         view.on("beforerowremoved", this.clearSelections, this);
58954         view.on("beforerowsinserted", this.clearSelections, this);
58955         if(this.grid.isEditor){
58956             this.grid.on("beforeedit", this.beforeEdit,  this);
58957         }
58958     },
58959
58960         //private
58961     beforeEdit : function(e){
58962         this.select(e.row, e.column, false, true, e.record);
58963     },
58964
58965         //private
58966     onRowUpdated : function(v, index, r){
58967         if(this.selection && this.selection.record == r){
58968             v.onCellSelect(index, this.selection.cell[1]);
58969         }
58970     },
58971
58972         //private
58973     onViewChange : function(){
58974         this.clearSelections(true);
58975     },
58976
58977         /**
58978          * Returns the currently selected cell,.
58979          * @return {Array} The selected cell (row, column) or null if none selected.
58980          */
58981     getSelectedCell : function(){
58982         return this.selection ? this.selection.cell : null;
58983     },
58984
58985     /**
58986      * Clears all selections.
58987      * @param {Boolean} true to prevent the gridview from being notified about the change.
58988      */
58989     clearSelections : function(preventNotify){
58990         var s = this.selection;
58991         if(s){
58992             if(preventNotify !== true){
58993                 this.grid.view.onCellDeselect(s.cell[0], s.cell[1]);
58994             }
58995             this.selection = null;
58996             this.fireEvent("selectionchange", this, null);
58997         }
58998     },
58999
59000     /**
59001      * Returns true if there is a selection.
59002      * @return {Boolean}
59003      */
59004     hasSelection : function(){
59005         return this.selection ? true : false;
59006     },
59007
59008     /** @ignore */
59009     handleMouseDown : function(e, t){
59010         var v = this.grid.getView();
59011         if(this.isLocked()){
59012             return;
59013         };
59014         var row = v.findRowIndex(t);
59015         var cell = v.findCellIndex(t);
59016         if(row !== false && cell !== false){
59017             this.select(row, cell);
59018         }
59019     },
59020
59021     /**
59022      * Selects a cell.
59023      * @param {Number} rowIndex
59024      * @param {Number} collIndex
59025      */
59026     select : function(rowIndex, colIndex, preventViewNotify, preventFocus, /*internal*/ r){
59027         if(this.fireEvent("beforecellselect", this, rowIndex, colIndex) !== false){
59028             this.clearSelections();
59029             r = r || this.grid.dataSource.getAt(rowIndex);
59030             this.selection = {
59031                 record : r,
59032                 cell : [rowIndex, colIndex]
59033             };
59034             if(!preventViewNotify){
59035                 var v = this.grid.getView();
59036                 v.onCellSelect(rowIndex, colIndex);
59037                 if(preventFocus !== true){
59038                     v.focusCell(rowIndex, colIndex);
59039                 }
59040             }
59041             this.fireEvent("cellselect", this, rowIndex, colIndex);
59042             this.fireEvent("selectionchange", this, this.selection);
59043         }
59044     },
59045
59046         //private
59047     isSelectable : function(rowIndex, colIndex, cm){
59048         return !cm.isHidden(colIndex);
59049     },
59050
59051     /** @ignore */
59052     handleKeyDown : function(e){
59053         //Roo.log('Cell Sel Model handleKeyDown');
59054         if(!e.isNavKeyPress()){
59055             return;
59056         }
59057         var g = this.grid, s = this.selection;
59058         if(!s){
59059             e.stopEvent();
59060             var cell = g.walkCells(0, 0, 1, this.isSelectable,  this);
59061             if(cell){
59062                 this.select(cell[0], cell[1]);
59063             }
59064             return;
59065         }
59066         var sm = this;
59067         var walk = function(row, col, step){
59068             return g.walkCells(row, col, step, sm.isSelectable,  sm);
59069         };
59070         var k = e.getKey(), r = s.cell[0], c = s.cell[1];
59071         var newCell;
59072
59073       
59074
59075         switch(k){
59076             case e.TAB:
59077                 // handled by onEditorKey
59078                 if (g.isEditor && g.editing) {
59079                     return;
59080                 }
59081                 if(e.shiftKey) {
59082                     newCell = walk(r, c-1, -1);
59083                 } else {
59084                     newCell = walk(r, c+1, 1);
59085                 }
59086                 break;
59087             
59088             case e.DOWN:
59089                newCell = walk(r+1, c, 1);
59090                 break;
59091             
59092             case e.UP:
59093                 newCell = walk(r-1, c, -1);
59094                 break;
59095             
59096             case e.RIGHT:
59097                 newCell = walk(r, c+1, 1);
59098                 break;
59099             
59100             case e.LEFT:
59101                 newCell = walk(r, c-1, -1);
59102                 break;
59103             
59104             case e.ENTER:
59105                 
59106                 if(g.isEditor && !g.editing){
59107                    g.startEditing(r, c);
59108                    e.stopEvent();
59109                    return;
59110                 }
59111                 
59112                 
59113              break;
59114         };
59115         if(newCell){
59116             this.select(newCell[0], newCell[1]);
59117             e.stopEvent();
59118             
59119         }
59120     },
59121
59122     acceptsNav : function(row, col, cm){
59123         return !cm.isHidden(col) && cm.isCellEditable(col, row);
59124     },
59125     /**
59126      * Selects a cell.
59127      * @param {Number} field (not used) - as it's normally used as a listener
59128      * @param {Number} e - event - fake it by using
59129      *
59130      * var e = Roo.EventObjectImpl.prototype;
59131      * e.keyCode = e.TAB
59132      *
59133      * 
59134      */
59135     onEditorKey : function(field, e){
59136         
59137         var k = e.getKey(),
59138             newCell,
59139             g = this.grid,
59140             ed = g.activeEditor,
59141             forward = false;
59142         ///Roo.log('onEditorKey' + k);
59143         
59144         
59145         if (this.enter_is_tab && k == e.ENTER) {
59146             k = e.TAB;
59147         }
59148         
59149         if(k == e.TAB){
59150             if(e.shiftKey){
59151                 newCell = g.walkCells(ed.row, ed.col-1, -1, this.acceptsNav, this);
59152             }else{
59153                 newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
59154                 forward = true;
59155             }
59156             
59157             e.stopEvent();
59158             
59159         } else if(k == e.ENTER &&  !e.ctrlKey){
59160             ed.completeEdit();
59161             e.stopEvent();
59162             newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
59163         
59164                 } else if(k == e.ESC){
59165             ed.cancelEdit();
59166         }
59167                 
59168         if (newCell) {
59169             var ecall = { cell : newCell, forward : forward };
59170             this.fireEvent('beforeeditnext', ecall );
59171             newCell = ecall.cell;
59172                         forward = ecall.forward;
59173         }
59174                 
59175         if(newCell){
59176             //Roo.log('next cell after edit');
59177             g.startEditing.defer(100, g, [newCell[0], newCell[1]]);
59178         } else if (forward) {
59179             // tabbed past last
59180             this.fireEvent.defer(100, this, ['tabend',this]);
59181         }
59182     }
59183 });/*
59184  * Based on:
59185  * Ext JS Library 1.1.1
59186  * Copyright(c) 2006-2007, Ext JS, LLC.
59187  *
59188  * Originally Released Under LGPL - original licence link has changed is not relivant.
59189  *
59190  * Fork - LGPL
59191  * <script type="text/javascript">
59192  */
59193  
59194 /**
59195  * @class Roo.grid.EditorGrid
59196  * @extends Roo.grid.Grid
59197  * Class for creating and editable grid.
59198  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered - 
59199  * The container MUST have some type of size defined for the grid to fill. The container will be 
59200  * automatically set to position relative if it isn't already.
59201  * @param {Object} dataSource The data model to bind to
59202  * @param {Object} colModel The column model with info about this grid's columns
59203  */
59204 Roo.grid.EditorGrid = function(container, config){
59205     Roo.grid.EditorGrid.superclass.constructor.call(this, container, config);
59206     this.getGridEl().addClass("xedit-grid");
59207
59208     if(!this.selModel){
59209         this.selModel = new Roo.grid.CellSelectionModel();
59210     }
59211
59212     this.activeEditor = null;
59213
59214         this.addEvents({
59215             /**
59216              * @event beforeedit
59217              * Fires before cell editing is triggered. The edit event object has the following properties <br />
59218              * <ul style="padding:5px;padding-left:16px;">
59219              * <li>grid - This grid</li>
59220              * <li>record - The record being edited</li>
59221              * <li>field - The field name being edited</li>
59222              * <li>value - The value for the field being edited.</li>
59223              * <li>row - The grid row index</li>
59224              * <li>column - The grid column index</li>
59225              * <li>cancel - Set this to true to cancel the edit or return false from your handler.</li>
59226              * </ul>
59227              * @param {Object} e An edit event (see above for description)
59228              */
59229             "beforeedit" : true,
59230             /**
59231              * @event afteredit
59232              * Fires after a cell is edited. <br />
59233              * <ul style="padding:5px;padding-left:16px;">
59234              * <li>grid - This grid</li>
59235              * <li>record - The record being edited</li>
59236              * <li>field - The field name being edited</li>
59237              * <li>value - The value being set</li>
59238              * <li>originalValue - The original value for the field, before the edit.</li>
59239              * <li>row - The grid row index</li>
59240              * <li>column - The grid column index</li>
59241              * </ul>
59242              * @param {Object} e An edit event (see above for description)
59243              */
59244             "afteredit" : true,
59245             /**
59246              * @event validateedit
59247              * Fires after a cell is edited, but before the value is set in the record. 
59248          * You can use this to modify the value being set in the field, Return false
59249              * to cancel the change. The edit event object has the following properties <br />
59250              * <ul style="padding:5px;padding-left:16px;">
59251          * <li>editor - This editor</li>
59252              * <li>grid - This grid</li>
59253              * <li>record - The record being edited</li>
59254              * <li>field - The field name being edited</li>
59255              * <li>value - The value being set</li>
59256              * <li>originalValue - The original value for the field, before the edit.</li>
59257              * <li>row - The grid row index</li>
59258              * <li>column - The grid column index</li>
59259              * <li>cancel - Set this to true to cancel the edit or return false from your handler.</li>
59260              * </ul>
59261              * @param {Object} e An edit event (see above for description)
59262              */
59263             "validateedit" : true
59264         });
59265     this.on("bodyscroll", this.stopEditing,  this);
59266     this.on(this.clicksToEdit == 1 ? "cellclick" : "celldblclick", this.onCellDblClick,  this);
59267 };
59268
59269 Roo.extend(Roo.grid.EditorGrid, Roo.grid.Grid, {
59270     /**
59271      * @cfg {Number} clicksToEdit
59272      * The number of clicks on a cell required to display the cell's editor (defaults to 2)
59273      */
59274     clicksToEdit: 2,
59275
59276     // private
59277     isEditor : true,
59278     // private
59279     trackMouseOver: false, // causes very odd FF errors
59280
59281     onCellDblClick : function(g, row, col){
59282         this.startEditing(row, col);
59283     },
59284
59285     onEditComplete : function(ed, value, startValue){
59286         this.editing = false;
59287         this.activeEditor = null;
59288         ed.un("specialkey", this.selModel.onEditorKey, this.selModel);
59289         var r = ed.record;
59290         var field = this.colModel.getDataIndex(ed.col);
59291         var e = {
59292             grid: this,
59293             record: r,
59294             field: field,
59295             originalValue: startValue,
59296             value: value,
59297             row: ed.row,
59298             column: ed.col,
59299             cancel:false,
59300             editor: ed
59301         };
59302         var cell = Roo.get(this.view.getCell(ed.row,ed.col));
59303         cell.show();
59304           
59305         if(String(value) !== String(startValue)){
59306             
59307             if(this.fireEvent("validateedit", e) !== false && !e.cancel){
59308                 r.set(field, e.value);
59309                 // if we are dealing with a combo box..
59310                 // then we also set the 'name' colum to be the displayField
59311                 if (ed.field.displayField && ed.field.name) {
59312                     r.set(ed.field.name, ed.field.el.dom.value);
59313                 }
59314                 
59315                 delete e.cancel; //?? why!!!
59316                 this.fireEvent("afteredit", e);
59317             }
59318         } else {
59319             this.fireEvent("afteredit", e); // always fire it!
59320         }
59321         this.view.focusCell(ed.row, ed.col);
59322     },
59323
59324     /**
59325      * Starts editing the specified for the specified row/column
59326      * @param {Number} rowIndex
59327      * @param {Number} colIndex
59328      */
59329     startEditing : function(row, col){
59330         this.stopEditing();
59331         if(this.colModel.isCellEditable(col, row)){
59332             this.view.ensureVisible(row, col, true);
59333           
59334             var r = this.dataSource.getAt(row);
59335             var field = this.colModel.getDataIndex(col);
59336             var cell = Roo.get(this.view.getCell(row,col));
59337             var e = {
59338                 grid: this,
59339                 record: r,
59340                 field: field,
59341                 value: r.data[field],
59342                 row: row,
59343                 column: col,
59344                 cancel:false 
59345             };
59346             if(this.fireEvent("beforeedit", e) !== false && !e.cancel){
59347                 this.editing = true;
59348                 var ed = this.colModel.getCellEditor(col, row);
59349                 
59350                 if (!ed) {
59351                     return;
59352                 }
59353                 if(!ed.rendered){
59354                     ed.render(ed.parentEl || document.body);
59355                 }
59356                 ed.field.reset();
59357                
59358                 cell.hide();
59359                 
59360                 (function(){ // complex but required for focus issues in safari, ie and opera
59361                     ed.row = row;
59362                     ed.col = col;
59363                     ed.record = r;
59364                     ed.on("complete",   this.onEditComplete,        this,       {single: true});
59365                     ed.on("specialkey", this.selModel.onEditorKey,  this.selModel);
59366                     this.activeEditor = ed;
59367                     var v = r.data[field];
59368                     ed.startEdit(this.view.getCell(row, col), v);
59369                     // combo's with 'displayField and name set
59370                     if (ed.field.displayField && ed.field.name) {
59371                         ed.field.el.dom.value = r.data[ed.field.name];
59372                     }
59373                     
59374                     
59375                 }).defer(50, this);
59376             }
59377         }
59378     },
59379         
59380     /**
59381      * Stops any active editing
59382      */
59383     stopEditing : function(){
59384         if(this.activeEditor){
59385             this.activeEditor.completeEdit();
59386         }
59387         this.activeEditor = null;
59388     },
59389         
59390          /**
59391      * Called to get grid's drag proxy text, by default returns this.ddText.
59392      * @return {String}
59393      */
59394     getDragDropText : function(){
59395         var count = this.selModel.getSelectedCell() ? 1 : 0;
59396         return String.format(this.ddText, count, count == 1 ? '' : 's');
59397     }
59398         
59399 });/*
59400  * Based on:
59401  * Ext JS Library 1.1.1
59402  * Copyright(c) 2006-2007, Ext JS, LLC.
59403  *
59404  * Originally Released Under LGPL - original licence link has changed is not relivant.
59405  *
59406  * Fork - LGPL
59407  * <script type="text/javascript">
59408  */
59409
59410 // private - not really -- you end up using it !
59411 // This is a support class used internally by the Grid components
59412
59413 /**
59414  * @class Roo.grid.GridEditor
59415  * @extends Roo.Editor
59416  * Class for creating and editable grid elements.
59417  * @param {Object} config any settings (must include field)
59418  */
59419 Roo.grid.GridEditor = function(field, config){
59420     if (!config && field.field) {
59421         config = field;
59422         field = Roo.factory(config.field, Roo.form);
59423     }
59424     Roo.grid.GridEditor.superclass.constructor.call(this, field, config);
59425     field.monitorTab = false;
59426 };
59427
59428 Roo.extend(Roo.grid.GridEditor, Roo.Editor, {
59429     
59430     /**
59431      * @cfg {Roo.form.Field} field Field to wrap (or xtyped)
59432      */
59433     
59434     alignment: "tl-tl",
59435     autoSize: "width",
59436     hideEl : false,
59437     cls: "x-small-editor x-grid-editor",
59438     shim:false,
59439     shadow:"frame"
59440 });/*
59441  * Based on:
59442  * Ext JS Library 1.1.1
59443  * Copyright(c) 2006-2007, Ext JS, LLC.
59444  *
59445  * Originally Released Under LGPL - original licence link has changed is not relivant.
59446  *
59447  * Fork - LGPL
59448  * <script type="text/javascript">
59449  */
59450   
59451
59452   
59453 Roo.grid.PropertyRecord = Roo.data.Record.create([
59454     {name:'name',type:'string'},  'value'
59455 ]);
59456
59457
59458 Roo.grid.PropertyStore = function(grid, source){
59459     this.grid = grid;
59460     this.store = new Roo.data.Store({
59461         recordType : Roo.grid.PropertyRecord
59462     });
59463     this.store.on('update', this.onUpdate,  this);
59464     if(source){
59465         this.setSource(source);
59466     }
59467     Roo.grid.PropertyStore.superclass.constructor.call(this);
59468 };
59469
59470
59471
59472 Roo.extend(Roo.grid.PropertyStore, Roo.util.Observable, {
59473     setSource : function(o){
59474         this.source = o;
59475         this.store.removeAll();
59476         var data = [];
59477         for(var k in o){
59478             if(this.isEditableValue(o[k])){
59479                 data.push(new Roo.grid.PropertyRecord({name: k, value: o[k]}, k));
59480             }
59481         }
59482         this.store.loadRecords({records: data}, {}, true);
59483     },
59484
59485     onUpdate : function(ds, record, type){
59486         if(type == Roo.data.Record.EDIT){
59487             var v = record.data['value'];
59488             var oldValue = record.modified['value'];
59489             if(this.grid.fireEvent('beforepropertychange', this.source, record.id, v, oldValue) !== false){
59490                 this.source[record.id] = v;
59491                 record.commit();
59492                 this.grid.fireEvent('propertychange', this.source, record.id, v, oldValue);
59493             }else{
59494                 record.reject();
59495             }
59496         }
59497     },
59498
59499     getProperty : function(row){
59500        return this.store.getAt(row);
59501     },
59502
59503     isEditableValue: function(val){
59504         if(val && val instanceof Date){
59505             return true;
59506         }else if(typeof val == 'object' || typeof val == 'function'){
59507             return false;
59508         }
59509         return true;
59510     },
59511
59512     setValue : function(prop, value){
59513         this.source[prop] = value;
59514         this.store.getById(prop).set('value', value);
59515     },
59516
59517     getSource : function(){
59518         return this.source;
59519     }
59520 });
59521
59522 Roo.grid.PropertyColumnModel = function(grid, store){
59523     this.grid = grid;
59524     var g = Roo.grid;
59525     g.PropertyColumnModel.superclass.constructor.call(this, [
59526         {header: this.nameText, sortable: true, dataIndex:'name', id: 'name'},
59527         {header: this.valueText, resizable:false, dataIndex: 'value', id: 'value'}
59528     ]);
59529     this.store = store;
59530     this.bselect = Roo.DomHelper.append(document.body, {
59531         tag: 'select', style:'display:none', cls: 'x-grid-editor', children: [
59532             {tag: 'option', value: 'true', html: 'true'},
59533             {tag: 'option', value: 'false', html: 'false'}
59534         ]
59535     });
59536     Roo.id(this.bselect);
59537     var f = Roo.form;
59538     this.editors = {
59539         'date' : new g.GridEditor(new f.DateField({selectOnFocus:true})),
59540         'string' : new g.GridEditor(new f.TextField({selectOnFocus:true})),
59541         'number' : new g.GridEditor(new f.NumberField({selectOnFocus:true, style:'text-align:left;'})),
59542         'int' : new g.GridEditor(new f.NumberField({selectOnFocus:true, allowDecimals:false, style:'text-align:left;'})),
59543         'boolean' : new g.GridEditor(new f.Field({el:this.bselect,selectOnFocus:true}))
59544     };
59545     this.renderCellDelegate = this.renderCell.createDelegate(this);
59546     this.renderPropDelegate = this.renderProp.createDelegate(this);
59547 };
59548
59549 Roo.extend(Roo.grid.PropertyColumnModel, Roo.grid.ColumnModel, {
59550     
59551     
59552     nameText : 'Name',
59553     valueText : 'Value',
59554     
59555     dateFormat : 'm/j/Y',
59556     
59557     
59558     renderDate : function(dateVal){
59559         return dateVal.dateFormat(this.dateFormat);
59560     },
59561
59562     renderBool : function(bVal){
59563         return bVal ? 'true' : 'false';
59564     },
59565
59566     isCellEditable : function(colIndex, rowIndex){
59567         return colIndex == 1;
59568     },
59569
59570     getRenderer : function(col){
59571         return col == 1 ?
59572             this.renderCellDelegate : this.renderPropDelegate;
59573     },
59574
59575     renderProp : function(v){
59576         return this.getPropertyName(v);
59577     },
59578
59579     renderCell : function(val){
59580         var rv = val;
59581         if(val instanceof Date){
59582             rv = this.renderDate(val);
59583         }else if(typeof val == 'boolean'){
59584             rv = this.renderBool(val);
59585         }
59586         return Roo.util.Format.htmlEncode(rv);
59587     },
59588
59589     getPropertyName : function(name){
59590         var pn = this.grid.propertyNames;
59591         return pn && pn[name] ? pn[name] : name;
59592     },
59593
59594     getCellEditor : function(colIndex, rowIndex){
59595         var p = this.store.getProperty(rowIndex);
59596         var n = p.data['name'], val = p.data['value'];
59597         
59598         if(typeof(this.grid.customEditors[n]) == 'string'){
59599             return this.editors[this.grid.customEditors[n]];
59600         }
59601         if(typeof(this.grid.customEditors[n]) != 'undefined'){
59602             return this.grid.customEditors[n];
59603         }
59604         if(val instanceof Date){
59605             return this.editors['date'];
59606         }else if(typeof val == 'number'){
59607             return this.editors['number'];
59608         }else if(typeof val == 'boolean'){
59609             return this.editors['boolean'];
59610         }else{
59611             return this.editors['string'];
59612         }
59613     }
59614 });
59615
59616 /**
59617  * @class Roo.grid.PropertyGrid
59618  * @extends Roo.grid.EditorGrid
59619  * This class represents the  interface of a component based property grid control.
59620  * <br><br>Usage:<pre><code>
59621  var grid = new Roo.grid.PropertyGrid("my-container-id", {
59622       
59623  });
59624  // set any options
59625  grid.render();
59626  * </code></pre>
59627   
59628  * @constructor
59629  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
59630  * The container MUST have some type of size defined for the grid to fill. The container will be
59631  * automatically set to position relative if it isn't already.
59632  * @param {Object} config A config object that sets properties on this grid.
59633  */
59634 Roo.grid.PropertyGrid = function(container, config){
59635     config = config || {};
59636     var store = new Roo.grid.PropertyStore(this);
59637     this.store = store;
59638     var cm = new Roo.grid.PropertyColumnModel(this, store);
59639     store.store.sort('name', 'ASC');
59640     Roo.grid.PropertyGrid.superclass.constructor.call(this, container, Roo.apply({
59641         ds: store.store,
59642         cm: cm,
59643         enableColLock:false,
59644         enableColumnMove:false,
59645         stripeRows:false,
59646         trackMouseOver: false,
59647         clicksToEdit:1
59648     }, config));
59649     this.getGridEl().addClass('x-props-grid');
59650     this.lastEditRow = null;
59651     this.on('columnresize', this.onColumnResize, this);
59652     this.addEvents({
59653          /**
59654              * @event beforepropertychange
59655              * Fires before a property changes (return false to stop?)
59656              * @param {Roo.grid.PropertyGrid} grid property grid? (check could be store)
59657              * @param {String} id Record Id
59658              * @param {String} newval New Value
59659          * @param {String} oldval Old Value
59660              */
59661         "beforepropertychange": true,
59662         /**
59663              * @event propertychange
59664              * Fires after a property changes
59665              * @param {Roo.grid.PropertyGrid} grid property grid? (check could be store)
59666              * @param {String} id Record Id
59667              * @param {String} newval New Value
59668          * @param {String} oldval Old Value
59669              */
59670         "propertychange": true
59671     });
59672     this.customEditors = this.customEditors || {};
59673 };
59674 Roo.extend(Roo.grid.PropertyGrid, Roo.grid.EditorGrid, {
59675     
59676      /**
59677      * @cfg {Object} customEditors map of colnames=> custom editors.
59678      * the custom editor can be one of the standard ones (date|string|number|int|boolean), or a
59679      * grid editor eg. Roo.grid.GridEditor(new Roo.form.TextArea({selectOnFocus:true})),
59680      * false disables editing of the field.
59681          */
59682     
59683       /**
59684      * @cfg {Object} propertyNames map of property Names to their displayed value
59685          */
59686     
59687     render : function(){
59688         Roo.grid.PropertyGrid.superclass.render.call(this);
59689         this.autoSize.defer(100, this);
59690     },
59691
59692     autoSize : function(){
59693         Roo.grid.PropertyGrid.superclass.autoSize.call(this);
59694         if(this.view){
59695             this.view.fitColumns();
59696         }
59697     },
59698
59699     onColumnResize : function(){
59700         this.colModel.setColumnWidth(1, this.container.getWidth(true)-this.colModel.getColumnWidth(0));
59701         this.autoSize();
59702     },
59703     /**
59704      * Sets the data for the Grid
59705      * accepts a Key => Value object of all the elements avaiable.
59706      * @param {Object} data  to appear in grid.
59707      */
59708     setSource : function(source){
59709         this.store.setSource(source);
59710         //this.autoSize();
59711     },
59712     /**
59713      * Gets all the data from the grid.
59714      * @return {Object} data  data stored in grid
59715      */
59716     getSource : function(){
59717         return this.store.getSource();
59718     }
59719 });/*
59720   
59721  * Licence LGPL
59722  
59723  */
59724  
59725 /**
59726  * @class Roo.grid.Calendar
59727  * @extends Roo.util.Grid
59728  * This class extends the Grid to provide a calendar widget
59729  * <br><br>Usage:<pre><code>
59730  var grid = new Roo.grid.Calendar("my-container-id", {
59731      ds: myDataStore,
59732      cm: myColModel,
59733      selModel: mySelectionModel,
59734      autoSizeColumns: true,
59735      monitorWindowResize: false,
59736      trackMouseOver: true
59737      eventstore : real data store..
59738  });
59739  // set any options
59740  grid.render();
59741   
59742   * @constructor
59743  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
59744  * The container MUST have some type of size defined for the grid to fill. The container will be
59745  * automatically set to position relative if it isn't already.
59746  * @param {Object} config A config object that sets properties on this grid.
59747  */
59748 Roo.grid.Calendar = function(container, config){
59749         // initialize the container
59750         this.container = Roo.get(container);
59751         this.container.update("");
59752         this.container.setStyle("overflow", "hidden");
59753     this.container.addClass('x-grid-container');
59754
59755     this.id = this.container.id;
59756
59757     Roo.apply(this, config);
59758     // check and correct shorthanded configs
59759     
59760     var rows = [];
59761     var d =1;
59762     for (var r = 0;r < 6;r++) {
59763         
59764         rows[r]=[];
59765         for (var c =0;c < 7;c++) {
59766             rows[r][c]= '';
59767         }
59768     }
59769     if (this.eventStore) {
59770         this.eventStore= Roo.factory(this.eventStore, Roo.data);
59771         this.eventStore.on('load',this.onLoad, this);
59772         this.eventStore.on('beforeload',this.clearEvents, this);
59773          
59774     }
59775     
59776     this.dataSource = new Roo.data.Store({
59777             proxy: new Roo.data.MemoryProxy(rows),
59778             reader: new Roo.data.ArrayReader({}, [
59779                    'weekday0', 'weekday1', 'weekday2', 'weekday3', 'weekday4', 'weekday5', 'weekday6' ])
59780     });
59781
59782     this.dataSource.load();
59783     this.ds = this.dataSource;
59784     this.ds.xmodule = this.xmodule || false;
59785     
59786     
59787     var cellRender = function(v,x,r)
59788     {
59789         return String.format(
59790             '<div class="fc-day  fc-widget-content"><div>' +
59791                 '<div class="fc-event-container"></div>' +
59792                 '<div class="fc-day-number">{0}</div>'+
59793                 
59794                 '<div class="fc-day-content"><div style="position:relative"></div></div>' +
59795             '</div></div>', v);
59796     
59797     }
59798     
59799     
59800     this.colModel = new Roo.grid.ColumnModel( [
59801         {
59802             xtype: 'ColumnModel',
59803             xns: Roo.grid,
59804             dataIndex : 'weekday0',
59805             header : 'Sunday',
59806             renderer : cellRender
59807         },
59808         {
59809             xtype: 'ColumnModel',
59810             xns: Roo.grid,
59811             dataIndex : 'weekday1',
59812             header : 'Monday',
59813             renderer : cellRender
59814         },
59815         {
59816             xtype: 'ColumnModel',
59817             xns: Roo.grid,
59818             dataIndex : 'weekday2',
59819             header : 'Tuesday',
59820             renderer : cellRender
59821         },
59822         {
59823             xtype: 'ColumnModel',
59824             xns: Roo.grid,
59825             dataIndex : 'weekday3',
59826             header : 'Wednesday',
59827             renderer : cellRender
59828         },
59829         {
59830             xtype: 'ColumnModel',
59831             xns: Roo.grid,
59832             dataIndex : 'weekday4',
59833             header : 'Thursday',
59834             renderer : cellRender
59835         },
59836         {
59837             xtype: 'ColumnModel',
59838             xns: Roo.grid,
59839             dataIndex : 'weekday5',
59840             header : 'Friday',
59841             renderer : cellRender
59842         },
59843         {
59844             xtype: 'ColumnModel',
59845             xns: Roo.grid,
59846             dataIndex : 'weekday6',
59847             header : 'Saturday',
59848             renderer : cellRender
59849         }
59850     ]);
59851     this.cm = this.colModel;
59852     this.cm.xmodule = this.xmodule || false;
59853  
59854         
59855           
59856     //this.selModel = new Roo.grid.CellSelectionModel();
59857     //this.sm = this.selModel;
59858     //this.selModel.init(this);
59859     
59860     
59861     if(this.width){
59862         this.container.setWidth(this.width);
59863     }
59864
59865     if(this.height){
59866         this.container.setHeight(this.height);
59867     }
59868     /** @private */
59869         this.addEvents({
59870         // raw events
59871         /**
59872          * @event click
59873          * The raw click event for the entire grid.
59874          * @param {Roo.EventObject} e
59875          */
59876         "click" : true,
59877         /**
59878          * @event dblclick
59879          * The raw dblclick event for the entire grid.
59880          * @param {Roo.EventObject} e
59881          */
59882         "dblclick" : true,
59883         /**
59884          * @event contextmenu
59885          * The raw contextmenu event for the entire grid.
59886          * @param {Roo.EventObject} e
59887          */
59888         "contextmenu" : true,
59889         /**
59890          * @event mousedown
59891          * The raw mousedown event for the entire grid.
59892          * @param {Roo.EventObject} e
59893          */
59894         "mousedown" : true,
59895         /**
59896          * @event mouseup
59897          * The raw mouseup event for the entire grid.
59898          * @param {Roo.EventObject} e
59899          */
59900         "mouseup" : true,
59901         /**
59902          * @event mouseover
59903          * The raw mouseover event for the entire grid.
59904          * @param {Roo.EventObject} e
59905          */
59906         "mouseover" : true,
59907         /**
59908          * @event mouseout
59909          * The raw mouseout event for the entire grid.
59910          * @param {Roo.EventObject} e
59911          */
59912         "mouseout" : true,
59913         /**
59914          * @event keypress
59915          * The raw keypress event for the entire grid.
59916          * @param {Roo.EventObject} e
59917          */
59918         "keypress" : true,
59919         /**
59920          * @event keydown
59921          * The raw keydown event for the entire grid.
59922          * @param {Roo.EventObject} e
59923          */
59924         "keydown" : true,
59925
59926         // custom events
59927
59928         /**
59929          * @event cellclick
59930          * Fires when a cell is clicked
59931          * @param {Grid} this
59932          * @param {Number} rowIndex
59933          * @param {Number} columnIndex
59934          * @param {Roo.EventObject} e
59935          */
59936         "cellclick" : true,
59937         /**
59938          * @event celldblclick
59939          * Fires when a cell is double clicked
59940          * @param {Grid} this
59941          * @param {Number} rowIndex
59942          * @param {Number} columnIndex
59943          * @param {Roo.EventObject} e
59944          */
59945         "celldblclick" : true,
59946         /**
59947          * @event rowclick
59948          * Fires when a row is clicked
59949          * @param {Grid} this
59950          * @param {Number} rowIndex
59951          * @param {Roo.EventObject} e
59952          */
59953         "rowclick" : true,
59954         /**
59955          * @event rowdblclick
59956          * Fires when a row is double clicked
59957          * @param {Grid} this
59958          * @param {Number} rowIndex
59959          * @param {Roo.EventObject} e
59960          */
59961         "rowdblclick" : true,
59962         /**
59963          * @event headerclick
59964          * Fires when a header is clicked
59965          * @param {Grid} this
59966          * @param {Number} columnIndex
59967          * @param {Roo.EventObject} e
59968          */
59969         "headerclick" : true,
59970         /**
59971          * @event headerdblclick
59972          * Fires when a header cell is double clicked
59973          * @param {Grid} this
59974          * @param {Number} columnIndex
59975          * @param {Roo.EventObject} e
59976          */
59977         "headerdblclick" : true,
59978         /**
59979          * @event rowcontextmenu
59980          * Fires when a row is right clicked
59981          * @param {Grid} this
59982          * @param {Number} rowIndex
59983          * @param {Roo.EventObject} e
59984          */
59985         "rowcontextmenu" : true,
59986         /**
59987          * @event cellcontextmenu
59988          * Fires when a cell is right clicked
59989          * @param {Grid} this
59990          * @param {Number} rowIndex
59991          * @param {Number} cellIndex
59992          * @param {Roo.EventObject} e
59993          */
59994          "cellcontextmenu" : true,
59995         /**
59996          * @event headercontextmenu
59997          * Fires when a header is right clicked
59998          * @param {Grid} this
59999          * @param {Number} columnIndex
60000          * @param {Roo.EventObject} e
60001          */
60002         "headercontextmenu" : true,
60003         /**
60004          * @event bodyscroll
60005          * Fires when the body element is scrolled
60006          * @param {Number} scrollLeft
60007          * @param {Number} scrollTop
60008          */
60009         "bodyscroll" : true,
60010         /**
60011          * @event columnresize
60012          * Fires when the user resizes a column
60013          * @param {Number} columnIndex
60014          * @param {Number} newSize
60015          */
60016         "columnresize" : true,
60017         /**
60018          * @event columnmove
60019          * Fires when the user moves a column
60020          * @param {Number} oldIndex
60021          * @param {Number} newIndex
60022          */
60023         "columnmove" : true,
60024         /**
60025          * @event startdrag
60026          * Fires when row(s) start being dragged
60027          * @param {Grid} this
60028          * @param {Roo.GridDD} dd The drag drop object
60029          * @param {event} e The raw browser event
60030          */
60031         "startdrag" : true,
60032         /**
60033          * @event enddrag
60034          * Fires when a drag operation is complete
60035          * @param {Grid} this
60036          * @param {Roo.GridDD} dd The drag drop object
60037          * @param {event} e The raw browser event
60038          */
60039         "enddrag" : true,
60040         /**
60041          * @event dragdrop
60042          * Fires when dragged row(s) are dropped on a valid DD target
60043          * @param {Grid} this
60044          * @param {Roo.GridDD} dd The drag drop object
60045          * @param {String} targetId The target drag drop object
60046          * @param {event} e The raw browser event
60047          */
60048         "dragdrop" : true,
60049         /**
60050          * @event dragover
60051          * Fires while row(s) are being dragged. "targetId" is the id of the Yahoo.util.DD object the selected rows are being dragged over.
60052          * @param {Grid} this
60053          * @param {Roo.GridDD} dd The drag drop object
60054          * @param {String} targetId The target drag drop object
60055          * @param {event} e The raw browser event
60056          */
60057         "dragover" : true,
60058         /**
60059          * @event dragenter
60060          *  Fires when the dragged row(s) first cross another DD target while being dragged
60061          * @param {Grid} this
60062          * @param {Roo.GridDD} dd The drag drop object
60063          * @param {String} targetId The target drag drop object
60064          * @param {event} e The raw browser event
60065          */
60066         "dragenter" : true,
60067         /**
60068          * @event dragout
60069          * Fires when the dragged row(s) leave another DD target while being dragged
60070          * @param {Grid} this
60071          * @param {Roo.GridDD} dd The drag drop object
60072          * @param {String} targetId The target drag drop object
60073          * @param {event} e The raw browser event
60074          */
60075         "dragout" : true,
60076         /**
60077          * @event rowclass
60078          * Fires when a row is rendered, so you can change add a style to it.
60079          * @param {GridView} gridview   The grid view
60080          * @param {Object} rowcfg   contains record  rowIndex and rowClass - set rowClass to add a style.
60081          */
60082         'rowclass' : true,
60083
60084         /**
60085          * @event render
60086          * Fires when the grid is rendered
60087          * @param {Grid} grid
60088          */
60089         'render' : true,
60090             /**
60091              * @event select
60092              * Fires when a date is selected
60093              * @param {DatePicker} this
60094              * @param {Date} date The selected date
60095              */
60096         'select': true,
60097         /**
60098              * @event monthchange
60099              * Fires when the displayed month changes 
60100              * @param {DatePicker} this
60101              * @param {Date} date The selected month
60102              */
60103         'monthchange': true,
60104         /**
60105              * @event evententer
60106              * Fires when mouse over an event
60107              * @param {Calendar} this
60108              * @param {event} Event
60109              */
60110         'evententer': true,
60111         /**
60112              * @event eventleave
60113              * Fires when the mouse leaves an
60114              * @param {Calendar} this
60115              * @param {event}
60116              */
60117         'eventleave': true,
60118         /**
60119              * @event eventclick
60120              * Fires when the mouse click an
60121              * @param {Calendar} this
60122              * @param {event}
60123              */
60124         'eventclick': true,
60125         /**
60126              * @event eventrender
60127              * Fires before each cell is rendered, so you can modify the contents, like cls / title / qtip
60128              * @param {Calendar} this
60129              * @param {data} data to be modified
60130              */
60131         'eventrender': true
60132         
60133     });
60134
60135     Roo.grid.Grid.superclass.constructor.call(this);
60136     this.on('render', function() {
60137         this.view.el.addClass('x-grid-cal'); 
60138         
60139         (function() { this.setDate(new Date()); }).defer(100,this); //default today..
60140
60141     },this);
60142     
60143     if (!Roo.grid.Calendar.style) {
60144         Roo.grid.Calendar.style = Roo.util.CSS.createStyleSheet({
60145             
60146             
60147             '.x-grid-cal .x-grid-col' :  {
60148                 height: 'auto !important',
60149                 'vertical-align': 'top'
60150             },
60151             '.x-grid-cal  .fc-event-hori' : {
60152                 height: '14px'
60153             }
60154              
60155             
60156         }, Roo.id());
60157     }
60158
60159     
60160     
60161 };
60162 Roo.extend(Roo.grid.Calendar, Roo.grid.Grid, {
60163     /**
60164      * @cfg {Store} eventStore The store that loads events.
60165      */
60166     eventStore : 25,
60167
60168      
60169     activeDate : false,
60170     startDay : 0,
60171     autoWidth : true,
60172     monitorWindowResize : false,
60173
60174     
60175     resizeColumns : function() {
60176         var col = (this.view.el.getWidth() / 7) - 3;
60177         // loop through cols, and setWidth
60178         for(var i =0 ; i < 7 ; i++){
60179             this.cm.setColumnWidth(i, col);
60180         }
60181     },
60182      setDate :function(date) {
60183         
60184         Roo.log('setDate?');
60185         
60186         this.resizeColumns();
60187         var vd = this.activeDate;
60188         this.activeDate = date;
60189 //        if(vd && this.el){
60190 //            var t = date.getTime();
60191 //            if(vd.getMonth() == date.getMonth() && vd.getFullYear() == date.getFullYear()){
60192 //                Roo.log('using add remove');
60193 //                
60194 //                this.fireEvent('monthchange', this, date);
60195 //                
60196 //                this.cells.removeClass("fc-state-highlight");
60197 //                this.cells.each(function(c){
60198 //                   if(c.dateValue == t){
60199 //                       c.addClass("fc-state-highlight");
60200 //                       setTimeout(function(){
60201 //                            try{c.dom.firstChild.focus();}catch(e){}
60202 //                       }, 50);
60203 //                       return false;
60204 //                   }
60205 //                   return true;
60206 //                });
60207 //                return;
60208 //            }
60209 //        }
60210         
60211         var days = date.getDaysInMonth();
60212         
60213         var firstOfMonth = date.getFirstDateOfMonth();
60214         var startingPos = firstOfMonth.getDay()-this.startDay;
60215         
60216         if(startingPos < this.startDay){
60217             startingPos += 7;
60218         }
60219         
60220         var pm = date.add(Date.MONTH, -1);
60221         var prevStart = pm.getDaysInMonth()-startingPos;
60222 //        
60223         
60224         
60225         this.cells = this.view.el.select('.x-grid-row .x-grid-col',true);
60226         
60227         this.textNodes = this.view.el.query('.x-grid-row .x-grid-col .x-grid-cell-text');
60228         //this.cells.addClassOnOver('fc-state-hover');
60229         
60230         var cells = this.cells.elements;
60231         var textEls = this.textNodes;
60232         
60233         //Roo.each(cells, function(cell){
60234         //    cell.removeClass([ 'fc-past', 'fc-other-month', 'fc-future', 'fc-state-highlight', 'fc-state-disabled']);
60235         //});
60236         
60237         days += startingPos;
60238
60239         // convert everything to numbers so it's fast
60240         var day = 86400000;
60241         var d = (new Date(pm.getFullYear(), pm.getMonth(), prevStart)).clearTime();
60242         //Roo.log(d);
60243         //Roo.log(pm);
60244         //Roo.log(prevStart);
60245         
60246         var today = new Date().clearTime().getTime();
60247         var sel = date.clearTime().getTime();
60248         var min = this.minDate ? this.minDate.clearTime() : Number.NEGATIVE_INFINITY;
60249         var max = this.maxDate ? this.maxDate.clearTime() : Number.POSITIVE_INFINITY;
60250         var ddMatch = this.disabledDatesRE;
60251         var ddText = this.disabledDatesText;
60252         var ddays = this.disabledDays ? this.disabledDays.join("") : false;
60253         var ddaysText = this.disabledDaysText;
60254         var format = this.format;
60255         
60256         var setCellClass = function(cal, cell){
60257             
60258             //Roo.log('set Cell Class');
60259             cell.title = "";
60260             var t = d.getTime();
60261             
60262             //Roo.log(d);
60263             
60264             
60265             cell.dateValue = t;
60266             if(t == today){
60267                 cell.className += " fc-today";
60268                 cell.className += " fc-state-highlight";
60269                 cell.title = cal.todayText;
60270             }
60271             if(t == sel){
60272                 // disable highlight in other month..
60273                 cell.className += " fc-state-highlight";
60274                 
60275             }
60276             // disabling
60277             if(t < min) {
60278                 //cell.className = " fc-state-disabled";
60279                 cell.title = cal.minText;
60280                 return;
60281             }
60282             if(t > max) {
60283                 //cell.className = " fc-state-disabled";
60284                 cell.title = cal.maxText;
60285                 return;
60286             }
60287             if(ddays){
60288                 if(ddays.indexOf(d.getDay()) != -1){
60289                     // cell.title = ddaysText;
60290                    // cell.className = " fc-state-disabled";
60291                 }
60292             }
60293             if(ddMatch && format){
60294                 var fvalue = d.dateFormat(format);
60295                 if(ddMatch.test(fvalue)){
60296                     cell.title = ddText.replace("%0", fvalue);
60297                    cell.className = " fc-state-disabled";
60298                 }
60299             }
60300             
60301             if (!cell.initialClassName) {
60302                 cell.initialClassName = cell.dom.className;
60303             }
60304             
60305             cell.dom.className = cell.initialClassName  + ' ' +  cell.className;
60306         };
60307
60308         var i = 0;
60309         
60310         for(; i < startingPos; i++) {
60311             cells[i].dayName =  (++prevStart);
60312             Roo.log(textEls[i]);
60313             d.setDate(d.getDate()+1);
60314             
60315             //cells[i].className = "fc-past fc-other-month";
60316             setCellClass(this, cells[i]);
60317         }
60318         
60319         var intDay = 0;
60320         
60321         for(; i < days; i++){
60322             intDay = i - startingPos + 1;
60323             cells[i].dayName =  (intDay);
60324             d.setDate(d.getDate()+1);
60325             
60326             cells[i].className = ''; // "x-date-active";
60327             setCellClass(this, cells[i]);
60328         }
60329         var extraDays = 0;
60330         
60331         for(; i < 42; i++) {
60332             //textEls[i].innerHTML = (++extraDays);
60333             
60334             d.setDate(d.getDate()+1);
60335             cells[i].dayName = (++extraDays);
60336             cells[i].className = "fc-future fc-other-month";
60337             setCellClass(this, cells[i]);
60338         }
60339         
60340         //this.el.select('.fc-header-title h2',true).update(Date.monthNames[date.getMonth()] + " " + date.getFullYear());
60341         
60342         var totalRows = Math.ceil((date.getDaysInMonth() + date.getFirstDateOfMonth().getDay()) / 7);
60343         
60344         // this will cause all the cells to mis
60345         var rows= [];
60346         var i =0;
60347         for (var r = 0;r < 6;r++) {
60348             for (var c =0;c < 7;c++) {
60349                 this.ds.getAt(r).set('weekday' + c ,cells[i++].dayName );
60350             }    
60351         }
60352         
60353         this.cells = this.view.el.select('.x-grid-row .x-grid-col',true);
60354         for(i=0;i<cells.length;i++) {
60355             
60356             this.cells.elements[i].dayName = cells[i].dayName ;
60357             this.cells.elements[i].className = cells[i].className;
60358             this.cells.elements[i].initialClassName = cells[i].initialClassName ;
60359             this.cells.elements[i].title = cells[i].title ;
60360             this.cells.elements[i].dateValue = cells[i].dateValue ;
60361         }
60362         
60363         
60364         
60365         
60366         //this.el.select('tr.fc-week.fc-prev-last',true).removeClass('fc-last');
60367         //this.el.select('tr.fc-week.fc-next-last',true).addClass('fc-last').show();
60368         
60369         ////if(totalRows != 6){
60370             //this.el.select('tr.fc-week.fc-last',true).removeClass('fc-last').addClass('fc-next-last').hide();
60371            // this.el.select('tr.fc-week.fc-prev-last',true).addClass('fc-last');
60372        // }
60373         
60374         this.fireEvent('monthchange', this, date);
60375         
60376         
60377     },
60378  /**
60379      * Returns the grid's SelectionModel.
60380      * @return {SelectionModel}
60381      */
60382     getSelectionModel : function(){
60383         if(!this.selModel){
60384             this.selModel = new Roo.grid.CellSelectionModel();
60385         }
60386         return this.selModel;
60387     },
60388
60389     load: function() {
60390         this.eventStore.load()
60391         
60392         
60393         
60394     },
60395     
60396     findCell : function(dt) {
60397         dt = dt.clearTime().getTime();
60398         var ret = false;
60399         this.cells.each(function(c){
60400             //Roo.log("check " +c.dateValue + '?=' + dt);
60401             if(c.dateValue == dt){
60402                 ret = c;
60403                 return false;
60404             }
60405             return true;
60406         });
60407         
60408         return ret;
60409     },
60410     
60411     findCells : function(rec) {
60412         var s = rec.data.start_dt.clone().clearTime().getTime();
60413        // Roo.log(s);
60414         var e= rec.data.end_dt.clone().clearTime().getTime();
60415        // Roo.log(e);
60416         var ret = [];
60417         this.cells.each(function(c){
60418              ////Roo.log("check " +c.dateValue + '<' + e + ' > ' + s);
60419             
60420             if(c.dateValue > e){
60421                 return ;
60422             }
60423             if(c.dateValue < s){
60424                 return ;
60425             }
60426             ret.push(c);
60427         });
60428         
60429         return ret;    
60430     },
60431     
60432     findBestRow: function(cells)
60433     {
60434         var ret = 0;
60435         
60436         for (var i =0 ; i < cells.length;i++) {
60437             ret  = Math.max(cells[i].rows || 0,ret);
60438         }
60439         return ret;
60440         
60441     },
60442     
60443     
60444     addItem : function(rec)
60445     {
60446         // look for vertical location slot in
60447         var cells = this.findCells(rec);
60448         
60449         rec.row = this.findBestRow(cells);
60450         
60451         // work out the location.
60452         
60453         var crow = false;
60454         var rows = [];
60455         for(var i =0; i < cells.length; i++) {
60456             if (!crow) {
60457                 crow = {
60458                     start : cells[i],
60459                     end :  cells[i]
60460                 };
60461                 continue;
60462             }
60463             if (crow.start.getY() == cells[i].getY()) {
60464                 // on same row.
60465                 crow.end = cells[i];
60466                 continue;
60467             }
60468             // different row.
60469             rows.push(crow);
60470             crow = {
60471                 start: cells[i],
60472                 end : cells[i]
60473             };
60474             
60475         }
60476         
60477         rows.push(crow);
60478         rec.els = [];
60479         rec.rows = rows;
60480         rec.cells = cells;
60481         for (var i = 0; i < cells.length;i++) {
60482             cells[i].rows = Math.max(cells[i].rows || 0 , rec.row + 1 );
60483             
60484         }
60485         
60486         
60487     },
60488     
60489     clearEvents: function() {
60490         
60491         if (!this.eventStore.getCount()) {
60492             return;
60493         }
60494         // reset number of rows in cells.
60495         Roo.each(this.cells.elements, function(c){
60496             c.rows = 0;
60497         });
60498         
60499         this.eventStore.each(function(e) {
60500             this.clearEvent(e);
60501         },this);
60502         
60503     },
60504     
60505     clearEvent : function(ev)
60506     {
60507         if (ev.els) {
60508             Roo.each(ev.els, function(el) {
60509                 el.un('mouseenter' ,this.onEventEnter, this);
60510                 el.un('mouseleave' ,this.onEventLeave, this);
60511                 el.remove();
60512             },this);
60513             ev.els = [];
60514         }
60515     },
60516     
60517     
60518     renderEvent : function(ev,ctr) {
60519         if (!ctr) {
60520              ctr = this.view.el.select('.fc-event-container',true).first();
60521         }
60522         
60523          
60524         this.clearEvent(ev);
60525             //code
60526        
60527         
60528         
60529         ev.els = [];
60530         var cells = ev.cells;
60531         var rows = ev.rows;
60532         this.fireEvent('eventrender', this, ev);
60533         
60534         for(var i =0; i < rows.length; i++) {
60535             
60536             cls = '';
60537             if (i == 0) {
60538                 cls += ' fc-event-start';
60539             }
60540             if ((i+1) == rows.length) {
60541                 cls += ' fc-event-end';
60542             }
60543             
60544             //Roo.log(ev.data);
60545             // how many rows should it span..
60546             var cg = this.eventTmpl.append(ctr,Roo.apply({
60547                 fccls : cls
60548                 
60549             }, ev.data) , true);
60550             
60551             
60552             cg.on('mouseenter' ,this.onEventEnter, this, ev);
60553             cg.on('mouseleave' ,this.onEventLeave, this, ev);
60554             cg.on('click', this.onEventClick, this, ev);
60555             
60556             ev.els.push(cg);
60557             
60558             var sbox = rows[i].start.select('.fc-day-content',true).first().getBox();
60559             var ebox = rows[i].end.select('.fc-day-content',true).first().getBox();
60560             //Roo.log(cg);
60561              
60562             cg.setXY([sbox.x +2, sbox.y +(ev.row * 20)]);    
60563             cg.setWidth(ebox.right - sbox.x -2);
60564         }
60565     },
60566     
60567     renderEvents: function()
60568     {   
60569         // first make sure there is enough space..
60570         
60571         if (!this.eventTmpl) {
60572             this.eventTmpl = new Roo.Template(
60573                 '<div class="roo-dynamic fc-event fc-event-hori fc-event-draggable ui-draggable {fccls} {cls}"  style="position: absolute" unselectable="on">' +
60574                     '<div class="fc-event-inner">' +
60575                         '<span class="fc-event-time">{time}</span>' +
60576                         '<span class="fc-event-title" qtip="{qtip}">{title}</span>' +
60577                     '</div>' +
60578                     '<div class="ui-resizable-heandle ui-resizable-e">&nbsp;&nbsp;&nbsp;</div>' +
60579                 '</div>'
60580             );
60581                 
60582         }
60583                
60584         
60585         
60586         this.cells.each(function(c) {
60587             //Roo.log(c.select('.fc-day-content div',true).first());
60588             c.select('.fc-day-content div',true).first().setHeight(Math.max(34, (c.rows || 1) * 20));
60589         });
60590         
60591         var ctr = this.view.el.select('.fc-event-container',true).first();
60592         
60593         var cls;
60594         this.eventStore.each(function(ev){
60595             
60596             this.renderEvent(ev);
60597              
60598              
60599         }, this);
60600         this.view.layout();
60601         
60602     },
60603     
60604     onEventEnter: function (e, el,event,d) {
60605         this.fireEvent('evententer', this, el, event);
60606     },
60607     
60608     onEventLeave: function (e, el,event,d) {
60609         this.fireEvent('eventleave', this, el, event);
60610     },
60611     
60612     onEventClick: function (e, el,event,d) {
60613         this.fireEvent('eventclick', this, el, event);
60614     },
60615     
60616     onMonthChange: function () {
60617         this.store.load();
60618     },
60619     
60620     onLoad: function () {
60621         
60622         //Roo.log('calendar onload');
60623 //         
60624         if(this.eventStore.getCount() > 0){
60625             
60626            
60627             
60628             this.eventStore.each(function(d){
60629                 
60630                 
60631                 // FIXME..
60632                 var add =   d.data;
60633                 if (typeof(add.end_dt) == 'undefined')  {
60634                     Roo.log("Missing End time in calendar data: ");
60635                     Roo.log(d);
60636                     return;
60637                 }
60638                 if (typeof(add.start_dt) == 'undefined')  {
60639                     Roo.log("Missing Start time in calendar data: ");
60640                     Roo.log(d);
60641                     return;
60642                 }
60643                 add.start_dt = typeof(add.start_dt) == 'string' ? Date.parseDate(add.start_dt,'Y-m-d H:i:s') : add.start_dt,
60644                 add.end_dt = typeof(add.end_dt) == 'string' ? Date.parseDate(add.end_dt,'Y-m-d H:i:s') : add.end_dt,
60645                 add.id = add.id || d.id;
60646                 add.title = add.title || '??';
60647                 
60648                 this.addItem(d);
60649                 
60650              
60651             },this);
60652         }
60653         
60654         this.renderEvents();
60655     }
60656     
60657
60658 });
60659 /*
60660  grid : {
60661                 xtype: 'Grid',
60662                 xns: Roo.grid,
60663                 listeners : {
60664                     render : function ()
60665                     {
60666                         _this.grid = this;
60667                         
60668                         if (!this.view.el.hasClass('course-timesheet')) {
60669                             this.view.el.addClass('course-timesheet');
60670                         }
60671                         if (this.tsStyle) {
60672                             this.ds.load({});
60673                             return; 
60674                         }
60675                         Roo.log('width');
60676                         Roo.log(_this.grid.view.el.getWidth());
60677                         
60678                         
60679                         this.tsStyle =  Roo.util.CSS.createStyleSheet({
60680                             '.course-timesheet .x-grid-row' : {
60681                                 height: '80px'
60682                             },
60683                             '.x-grid-row td' : {
60684                                 'vertical-align' : 0
60685                             },
60686                             '.course-edit-link' : {
60687                                 'color' : 'blue',
60688                                 'text-overflow' : 'ellipsis',
60689                                 'overflow' : 'hidden',
60690                                 'white-space' : 'nowrap',
60691                                 'cursor' : 'pointer'
60692                             },
60693                             '.sub-link' : {
60694                                 'color' : 'green'
60695                             },
60696                             '.de-act-sup-link' : {
60697                                 'color' : 'purple',
60698                                 'text-decoration' : 'line-through'
60699                             },
60700                             '.de-act-link' : {
60701                                 'color' : 'red',
60702                                 'text-decoration' : 'line-through'
60703                             },
60704                             '.course-timesheet .course-highlight' : {
60705                                 'border-top-style': 'dashed !important',
60706                                 'border-bottom-bottom': 'dashed !important'
60707                             },
60708                             '.course-timesheet .course-item' : {
60709                                 'font-family'   : 'tahoma, arial, helvetica',
60710                                 'font-size'     : '11px',
60711                                 'overflow'      : 'hidden',
60712                                 'padding-left'  : '10px',
60713                                 'padding-right' : '10px',
60714                                 'padding-top' : '10px' 
60715                             }
60716                             
60717                         }, Roo.id());
60718                                 this.ds.load({});
60719                     }
60720                 },
60721                 autoWidth : true,
60722                 monitorWindowResize : false,
60723                 cellrenderer : function(v,x,r)
60724                 {
60725                     return v;
60726                 },
60727                 sm : {
60728                     xtype: 'CellSelectionModel',
60729                     xns: Roo.grid
60730                 },
60731                 dataSource : {
60732                     xtype: 'Store',
60733                     xns: Roo.data,
60734                     listeners : {
60735                         beforeload : function (_self, options)
60736                         {
60737                             options.params = options.params || {};
60738                             options.params._month = _this.monthField.getValue();
60739                             options.params.limit = 9999;
60740                             options.params['sort'] = 'when_dt';    
60741                             options.params['dir'] = 'ASC';    
60742                             this.proxy.loadResponse = this.loadResponse;
60743                             Roo.log("load?");
60744                             //this.addColumns();
60745                         },
60746                         load : function (_self, records, options)
60747                         {
60748                             _this.grid.view.el.select('.course-edit-link', true).on('click', function() {
60749                                 // if you click on the translation.. you can edit it...
60750                                 var el = Roo.get(this);
60751                                 var id = el.dom.getAttribute('data-id');
60752                                 var d = el.dom.getAttribute('data-date');
60753                                 var t = el.dom.getAttribute('data-time');
60754                                 //var id = this.child('span').dom.textContent;
60755                                 
60756                                 //Roo.log(this);
60757                                 Pman.Dialog.CourseCalendar.show({
60758                                     id : id,
60759                                     when_d : d,
60760                                     when_t : t,
60761                                     productitem_active : id ? 1 : 0
60762                                 }, function() {
60763                                     _this.grid.ds.load({});
60764                                 });
60765                            
60766                            });
60767                            
60768                            _this.panel.fireEvent('resize', [ '', '' ]);
60769                         }
60770                     },
60771                     loadResponse : function(o, success, response){
60772                             // this is overridden on before load..
60773                             
60774                             Roo.log("our code?");       
60775                             //Roo.log(success);
60776                             //Roo.log(response)
60777                             delete this.activeRequest;
60778                             if(!success){
60779                                 this.fireEvent("loadexception", this, o, response);
60780                                 o.request.callback.call(o.request.scope, null, o.request.arg, false);
60781                                 return;
60782                             }
60783                             var result;
60784                             try {
60785                                 result = o.reader.read(response);
60786                             }catch(e){
60787                                 Roo.log("load exception?");
60788                                 this.fireEvent("loadexception", this, o, response, e);
60789                                 o.request.callback.call(o.request.scope, null, o.request.arg, false);
60790                                 return;
60791                             }
60792                             Roo.log("ready...");        
60793                             // loop through result.records;
60794                             // and set this.tdate[date] = [] << array of records..
60795                             _this.tdata  = {};
60796                             Roo.each(result.records, function(r){
60797                                 //Roo.log(r.data);
60798                                 if(typeof(_this.tdata[r.data.when_dt.format('j')]) == 'undefined'){
60799                                     _this.tdata[r.data.when_dt.format('j')] = [];
60800                                 }
60801                                 _this.tdata[r.data.when_dt.format('j')].push(r.data);
60802                             });
60803                             
60804                             //Roo.log(_this.tdata);
60805                             
60806                             result.records = [];
60807                             result.totalRecords = 6;
60808                     
60809                             // let's generate some duumy records for the rows.
60810                             //var st = _this.dateField.getValue();
60811                             
60812                             // work out monday..
60813                             //st = st.add(Date.DAY, -1 * st.format('w'));
60814                             
60815                             var date = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60816                             
60817                             var firstOfMonth = date.getFirstDayOfMonth();
60818                             var days = date.getDaysInMonth();
60819                             var d = 1;
60820                             var firstAdded = false;
60821                             for (var i = 0; i < result.totalRecords ; i++) {
60822                                 //var d= st.add(Date.DAY, i);
60823                                 var row = {};
60824                                 var added = 0;
60825                                 for(var w = 0 ; w < 7 ; w++){
60826                                     if(!firstAdded && firstOfMonth != w){
60827                                         continue;
60828                                     }
60829                                     if(d > days){
60830                                         continue;
60831                                     }
60832                                     firstAdded = true;
60833                                     var dd = (d > 0 && d < 10) ? "0"+d : d;
60834                                     row['weekday'+w] = String.format(
60835                                                     '<span style="font-size: 16px;"><b>{0}</b></span>'+
60836                                                     '<span class="course-edit-link" style="color:blue;" data-id="0" data-date="{1}"> Add New</span>',
60837                                                     d,
60838                                                     date.format('Y-m-')+dd
60839                                                 );
60840                                     added++;
60841                                     if(typeof(_this.tdata[d]) != 'undefined'){
60842                                         Roo.each(_this.tdata[d], function(r){
60843                                             var is_sub = '';
60844                                             var deactive = '';
60845                                             var id = r.id;
60846                                             var desc = (r.productitem_id_descrip) ? r.productitem_id_descrip : '';
60847                                             if(r.parent_id*1>0){
60848                                                 is_sub = (r.productitem_id_visible*1 < 1) ? 'de-act-sup-link' :'sub-link';
60849                                                 id = r.parent_id;
60850                                             }
60851                                             if(r.productitem_id_visible*1 < 1 && r.parent_id*1 < 1){
60852                                                 deactive = 'de-act-link';
60853                                             }
60854                                             
60855                                             row['weekday'+w] += String.format(
60856                                                     '<br /><span class="course-edit-link {3} {4}" qtip="{5}" data-id="{0}">{2} - {1}</span>',
60857                                                     id, //0
60858                                                     r.product_id_name, //1
60859                                                     r.when_dt.format('h:ia'), //2
60860                                                     is_sub, //3
60861                                                     deactive, //4
60862                                                     desc // 5
60863                                             );
60864                                         });
60865                                     }
60866                                     d++;
60867                                 }
60868                                 
60869                                 // only do this if something added..
60870                                 if(added > 0){ 
60871                                     result.records.push(_this.grid.dataSource.reader.newRow(row));
60872                                 }
60873                                 
60874                                 
60875                                 // push it twice. (second one with an hour..
60876                                 
60877                             }
60878                             //Roo.log(result);
60879                             this.fireEvent("load", this, o, o.request.arg);
60880                             o.request.callback.call(o.request.scope, result, o.request.arg, true);
60881                         },
60882                     sortInfo : {field: 'when_dt', direction : 'ASC' },
60883                     proxy : {
60884                         xtype: 'HttpProxy',
60885                         xns: Roo.data,
60886                         method : 'GET',
60887                         url : baseURL + '/Roo/Shop_course.php'
60888                     },
60889                     reader : {
60890                         xtype: 'JsonReader',
60891                         xns: Roo.data,
60892                         id : 'id',
60893                         fields : [
60894                             {
60895                                 'name': 'id',
60896                                 'type': 'int'
60897                             },
60898                             {
60899                                 'name': 'when_dt',
60900                                 'type': 'string'
60901                             },
60902                             {
60903                                 'name': 'end_dt',
60904                                 'type': 'string'
60905                             },
60906                             {
60907                                 'name': 'parent_id',
60908                                 'type': 'int'
60909                             },
60910                             {
60911                                 'name': 'product_id',
60912                                 'type': 'int'
60913                             },
60914                             {
60915                                 'name': 'productitem_id',
60916                                 'type': 'int'
60917                             },
60918                             {
60919                                 'name': 'guid',
60920                                 'type': 'int'
60921                             }
60922                         ]
60923                     }
60924                 },
60925                 toolbar : {
60926                     xtype: 'Toolbar',
60927                     xns: Roo,
60928                     items : [
60929                         {
60930                             xtype: 'Button',
60931                             xns: Roo.Toolbar,
60932                             listeners : {
60933                                 click : function (_self, e)
60934                                 {
60935                                     var sd = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60936                                     sd.setMonth(sd.getMonth()-1);
60937                                     _this.monthField.setValue(sd.format('Y-m-d'));
60938                                     _this.grid.ds.load({});
60939                                 }
60940                             },
60941                             text : "Back"
60942                         },
60943                         {
60944                             xtype: 'Separator',
60945                             xns: Roo.Toolbar
60946                         },
60947                         {
60948                             xtype: 'MonthField',
60949                             xns: Roo.form,
60950                             listeners : {
60951                                 render : function (_self)
60952                                 {
60953                                     _this.monthField = _self;
60954                                    // _this.monthField.set  today
60955                                 },
60956                                 select : function (combo, date)
60957                                 {
60958                                     _this.grid.ds.load({});
60959                                 }
60960                             },
60961                             value : (function() { return new Date(); })()
60962                         },
60963                         {
60964                             xtype: 'Separator',
60965                             xns: Roo.Toolbar
60966                         },
60967                         {
60968                             xtype: 'TextItem',
60969                             xns: Roo.Toolbar,
60970                             text : "Blue: in-active, green: in-active sup-event, red: de-active, purple: de-active sup-event"
60971                         },
60972                         {
60973                             xtype: 'Fill',
60974                             xns: Roo.Toolbar
60975                         },
60976                         {
60977                             xtype: 'Button',
60978                             xns: Roo.Toolbar,
60979                             listeners : {
60980                                 click : function (_self, e)
60981                                 {
60982                                     var sd = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60983                                     sd.setMonth(sd.getMonth()+1);
60984                                     _this.monthField.setValue(sd.format('Y-m-d'));
60985                                     _this.grid.ds.load({});
60986                                 }
60987                             },
60988                             text : "Next"
60989                         }
60990                     ]
60991                 },
60992                  
60993             }
60994         };
60995         
60996         *//*
60997  * Based on:
60998  * Ext JS Library 1.1.1
60999  * Copyright(c) 2006-2007, Ext JS, LLC.
61000  *
61001  * Originally Released Under LGPL - original licence link has changed is not relivant.
61002  *
61003  * Fork - LGPL
61004  * <script type="text/javascript">
61005  */
61006  
61007 /**
61008  * @class Roo.LoadMask
61009  * A simple utility class for generically masking elements while loading data.  If the element being masked has
61010  * an underlying {@link Roo.data.Store}, the masking will be automatically synchronized with the store's loading
61011  * process and the mask element will be cached for reuse.  For all other elements, this mask will replace the
61012  * element's UpdateManager load indicator and will be destroyed after the initial load.
61013  * @constructor
61014  * Create a new LoadMask
61015  * @param {String/HTMLElement/Roo.Element} el The element or DOM node, or its id
61016  * @param {Object} config The config object
61017  */
61018 Roo.LoadMask = function(el, config){
61019     this.el = Roo.get(el);
61020     Roo.apply(this, config);
61021     if(this.store){
61022         this.store.on('beforeload', this.onBeforeLoad, this);
61023         this.store.on('load', this.onLoad, this);
61024         this.store.on('loadexception', this.onLoadException, this);
61025         this.removeMask = false;
61026     }else{
61027         var um = this.el.getUpdateManager();
61028         um.showLoadIndicator = false; // disable the default indicator
61029         um.on('beforeupdate', this.onBeforeLoad, this);
61030         um.on('update', this.onLoad, this);
61031         um.on('failure', this.onLoad, this);
61032         this.removeMask = true;
61033     }
61034 };
61035
61036 Roo.LoadMask.prototype = {
61037     /**
61038      * @cfg {Boolean} removeMask
61039      * True to create a single-use mask that is automatically destroyed after loading (useful for page loads),
61040      * False to persist the mask element reference for multiple uses (e.g., for paged data widgets).  Defaults to false.
61041      */
61042     /**
61043      * @cfg {String} msg
61044      * The text to display in a centered loading message box (defaults to 'Loading...')
61045      */
61046     msg : 'Loading...',
61047     /**
61048      * @cfg {String} msgCls
61049      * The CSS class to apply to the loading message element (defaults to "x-mask-loading")
61050      */
61051     msgCls : 'x-mask-loading',
61052
61053     /**
61054      * Read-only. True if the mask is currently disabled so that it will not be displayed (defaults to false)
61055      * @type Boolean
61056      */
61057     disabled: false,
61058
61059     /**
61060      * Disables the mask to prevent it from being displayed
61061      */
61062     disable : function(){
61063        this.disabled = true;
61064     },
61065
61066     /**
61067      * Enables the mask so that it can be displayed
61068      */
61069     enable : function(){
61070         this.disabled = false;
61071     },
61072     
61073     onLoadException : function()
61074     {
61075         Roo.log(arguments);
61076         
61077         if (typeof(arguments[3]) != 'undefined') {
61078             Roo.MessageBox.alert("Error loading",arguments[3]);
61079         } 
61080         /*
61081         try {
61082             if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
61083                 Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
61084             }   
61085         } catch(e) {
61086             
61087         }
61088         */
61089     
61090         (function() { this.el.unmask(this.removeMask); }).defer(50, this);
61091     },
61092     // private
61093     onLoad : function()
61094     {
61095         (function() { this.el.unmask(this.removeMask); }).defer(50, this);
61096     },
61097
61098     // private
61099     onBeforeLoad : function(){
61100         if(!this.disabled){
61101             (function() { this.el.mask(this.msg, this.msgCls); }).defer(50, this);
61102         }
61103     },
61104
61105     // private
61106     destroy : function(){
61107         if(this.store){
61108             this.store.un('beforeload', this.onBeforeLoad, this);
61109             this.store.un('load', this.onLoad, this);
61110             this.store.un('loadexception', this.onLoadException, this);
61111         }else{
61112             var um = this.el.getUpdateManager();
61113             um.un('beforeupdate', this.onBeforeLoad, this);
61114             um.un('update', this.onLoad, this);
61115             um.un('failure', this.onLoad, this);
61116         }
61117     }
61118 };/*
61119  * Based on:
61120  * Ext JS Library 1.1.1
61121  * Copyright(c) 2006-2007, Ext JS, LLC.
61122  *
61123  * Originally Released Under LGPL - original licence link has changed is not relivant.
61124  *
61125  * Fork - LGPL
61126  * <script type="text/javascript">
61127  */
61128
61129
61130 /**
61131  * @class Roo.XTemplate
61132  * @extends Roo.Template
61133  * Provides a template that can have nested templates for loops or conditionals. The syntax is:
61134 <pre><code>
61135 var t = new Roo.XTemplate(
61136         '&lt;select name="{name}"&gt;',
61137                 '&lt;tpl for="options"&gt;&lt;option value="{value:trim}"&gt;{text:ellipsis(10)}&lt;/option&gt;&lt;/tpl&gt;',
61138         '&lt;/select&gt;'
61139 );
61140  
61141 // then append, applying the master template values
61142  </code></pre>
61143  *
61144  * Supported features:
61145  *
61146  *  Tags:
61147
61148 <pre><code>
61149       {a_variable} - output encoded.
61150       {a_variable.format:("Y-m-d")} - call a method on the variable
61151       {a_variable:raw} - unencoded output
61152       {a_variable:toFixed(1,2)} - Roo.util.Format."toFixed"
61153       {a_variable:this.method_on_template(...)} - call a method on the template object.
61154  
61155 </code></pre>
61156  *  The tpl tag:
61157 <pre><code>
61158         &lt;tpl for="a_variable or condition.."&gt;&lt;/tpl&gt;
61159         &lt;tpl if="a_variable or condition"&gt;&lt;/tpl&gt;
61160         &lt;tpl exec="some javascript"&gt;&lt;/tpl&gt;
61161         &lt;tpl name="named_template"&gt;&lt;/tpl&gt; (experimental)
61162   
61163         &lt;tpl for="."&gt;&lt;/tpl&gt; - just iterate the property..
61164         &lt;tpl for=".."&gt;&lt;/tpl&gt; - iterates with the parent (probably the template) 
61165 </code></pre>
61166  *      
61167  */
61168 Roo.XTemplate = function()
61169 {
61170     Roo.XTemplate.superclass.constructor.apply(this, arguments);
61171     if (this.html) {
61172         this.compile();
61173     }
61174 };
61175
61176
61177 Roo.extend(Roo.XTemplate, Roo.Template, {
61178
61179     /**
61180      * The various sub templates
61181      */
61182     tpls : false,
61183     /**
61184      *
61185      * basic tag replacing syntax
61186      * WORD:WORD()
61187      *
61188      * // you can fake an object call by doing this
61189      *  x.t:(test,tesT) 
61190      * 
61191      */
61192     re : /\{([\w-\.]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
61193
61194     /**
61195      * compile the template
61196      *
61197      * This is not recursive, so I'm not sure how nested templates are really going to be handled..
61198      *
61199      */
61200     compile: function()
61201     {
61202         var s = this.html;
61203      
61204         s = ['<tpl>', s, '</tpl>'].join('');
61205     
61206         var re     = /<tpl\b[^>]*>((?:(?=([^<]+))\2|<(?!tpl\b[^>]*>))*?)<\/tpl>/,
61207             nameRe = /^<tpl\b[^>]*?for="(.*?)"/,
61208             ifRe   = /^<tpl\b[^>]*?if="(.*?)"/,
61209             execRe = /^<tpl\b[^>]*?exec="(.*?)"/,
61210             namedRe = /^<tpl\b[^>]*?name="(\w+)"/,  // named templates..
61211             m,
61212             id     = 0,
61213             tpls   = [];
61214     
61215         while(true == !!(m = s.match(re))){
61216             var forMatch   = m[0].match(nameRe),
61217                 ifMatch   = m[0].match(ifRe),
61218                 execMatch   = m[0].match(execRe),
61219                 namedMatch   = m[0].match(namedRe),
61220                 
61221                 exp  = null, 
61222                 fn   = null,
61223                 exec = null,
61224                 name = forMatch && forMatch[1] ? forMatch[1] : '';
61225                 
61226             if (ifMatch) {
61227                 // if - puts fn into test..
61228                 exp = ifMatch && ifMatch[1] ? ifMatch[1] : null;
61229                 if(exp){
61230                    fn = new Function('values', 'parent', 'with(values){ return '+(Roo.util.Format.htmlDecode(exp))+'; }');
61231                 }
61232             }
61233             
61234             if (execMatch) {
61235                 // exec - calls a function... returns empty if true is  returned.
61236                 exp = execMatch && execMatch[1] ? execMatch[1] : null;
61237                 if(exp){
61238                    exec = new Function('values', 'parent', 'with(values){ '+(Roo.util.Format.htmlDecode(exp))+'; }');
61239                 }
61240             }
61241             
61242             
61243             if (name) {
61244                 // for = 
61245                 switch(name){
61246                     case '.':  name = new Function('values', 'parent', 'with(values){ return values; }'); break;
61247                     case '..': name = new Function('values', 'parent', 'with(values){ return parent; }'); break;
61248                     default:   name = new Function('values', 'parent', 'with(values){ return '+name+'; }');
61249                 }
61250             }
61251             var uid = namedMatch ? namedMatch[1] : id;
61252             
61253             
61254             tpls.push({
61255                 id:     namedMatch ? namedMatch[1] : id,
61256                 target: name,
61257                 exec:   exec,
61258                 test:   fn,
61259                 body:   m[1] || ''
61260             });
61261             if (namedMatch) {
61262                 s = s.replace(m[0], '');
61263             } else { 
61264                 s = s.replace(m[0], '{xtpl'+ id + '}');
61265             }
61266             ++id;
61267         }
61268         this.tpls = [];
61269         for(var i = tpls.length-1; i >= 0; --i){
61270             this.compileTpl(tpls[i]);
61271             this.tpls[tpls[i].id] = tpls[i];
61272         }
61273         this.master = tpls[tpls.length-1];
61274         return this;
61275     },
61276     /**
61277      * same as applyTemplate, except it's done to one of the subTemplates
61278      * when using named templates, you can do:
61279      *
61280      * var str = pl.applySubTemplate('your-name', values);
61281      *
61282      * 
61283      * @param {Number} id of the template
61284      * @param {Object} values to apply to template
61285      * @param {Object} parent (normaly the instance of this object)
61286      */
61287     applySubTemplate : function(id, values, parent)
61288     {
61289         
61290         
61291         var t = this.tpls[id];
61292         
61293         
61294         try { 
61295             if(t.test && !t.test.call(this, values, parent)){
61296                 return '';
61297             }
61298         } catch(e) {
61299             Roo.log("Xtemplate.applySubTemplate 'test': Exception thrown");
61300             Roo.log(e.toString());
61301             Roo.log(t.test);
61302             return ''
61303         }
61304         try { 
61305             
61306             if(t.exec && t.exec.call(this, values, parent)){
61307                 return '';
61308             }
61309         } catch(e) {
61310             Roo.log("Xtemplate.applySubTemplate 'exec': Exception thrown");
61311             Roo.log(e.toString());
61312             Roo.log(t.exec);
61313             return ''
61314         }
61315         try {
61316             var vs = t.target ? t.target.call(this, values, parent) : values;
61317             parent = t.target ? values : parent;
61318             if(t.target && vs instanceof Array){
61319                 var buf = [];
61320                 for(var i = 0, len = vs.length; i < len; i++){
61321                     buf[buf.length] = t.compiled.call(this, vs[i], parent);
61322                 }
61323                 return buf.join('');
61324             }
61325             return t.compiled.call(this, vs, parent);
61326         } catch (e) {
61327             Roo.log("Xtemplate.applySubTemplate : Exception thrown");
61328             Roo.log(e.toString());
61329             Roo.log(t.compiled);
61330             return '';
61331         }
61332     },
61333
61334     compileTpl : function(tpl)
61335     {
61336         var fm = Roo.util.Format;
61337         var useF = this.disableFormats !== true;
61338         var sep = Roo.isGecko ? "+" : ",";
61339         var undef = function(str) {
61340             Roo.log("Property not found :"  + str);
61341             return '';
61342         };
61343         
61344         var fn = function(m, name, format, args)
61345         {
61346             //Roo.log(arguments);
61347             args = args ? args.replace(/\\'/g,"'") : args;
61348             //["{TEST:(a,b,c)}", "TEST", "", "a,b,c", 0, "{TEST:(a,b,c)}"]
61349             if (typeof(format) == 'undefined') {
61350                 format= 'htmlEncode';
61351             }
61352             if (format == 'raw' ) {
61353                 format = false;
61354             }
61355             
61356             if(name.substr(0, 4) == 'xtpl'){
61357                 return "'"+ sep +'this.applySubTemplate('+name.substr(4)+', values, parent)'+sep+"'";
61358             }
61359             
61360             // build an array of options to determine if value is undefined..
61361             
61362             // basically get 'xxxx.yyyy' then do
61363             // (typeof(xxxx) == 'undefined' || typeof(xxx.yyyy) == 'undefined') ?
61364             //    (function () { Roo.log("Property not found"); return ''; })() :
61365             //    ......
61366             
61367             var udef_ar = [];
61368             var lookfor = '';
61369             Roo.each(name.split('.'), function(st) {
61370                 lookfor += (lookfor.length ? '.': '') + st;
61371                 udef_ar.push(  "(typeof(" + lookfor + ") == 'undefined')"  );
61372             });
61373             
61374             var udef_st = '((' + udef_ar.join(" || ") +") ? undef('" + name + "') : "; // .. needs )
61375             
61376             
61377             if(format && useF){
61378                 
61379                 args = args ? ',' + args : "";
61380                  
61381                 if(format.substr(0, 5) != "this."){
61382                     format = "fm." + format + '(';
61383                 }else{
61384                     format = 'this.call("'+ format.substr(5) + '", ';
61385                     args = ", values";
61386                 }
61387                 
61388                 return "'"+ sep +   udef_st   +    format + name + args + "))"+sep+"'";
61389             }
61390              
61391             if (args.length) {
61392                 // called with xxyx.yuu:(test,test)
61393                 // change to ()
61394                 return "'"+ sep + udef_st  + name + '(' +  args + "))"+sep+"'";
61395             }
61396             // raw.. - :raw modifier..
61397             return "'"+ sep + udef_st  + name + ")"+sep+"'";
61398             
61399         };
61400         var body;
61401         // branched to use + in gecko and [].join() in others
61402         if(Roo.isGecko){
61403             body = "tpl.compiled = function(values, parent){  with(values) { return '" +
61404                    tpl.body.replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
61405                     "';};};";
61406         }else{
61407             body = ["tpl.compiled = function(values, parent){  with (values) { return ['"];
61408             body.push(tpl.body.replace(/(\r\n|\n)/g,
61409                             '\\n').replace(/'/g, "\\'").replace(this.re, fn));
61410             body.push("'].join('');};};");
61411             body = body.join('');
61412         }
61413         
61414         Roo.debug && Roo.log(body.replace(/\\n/,'\n'));
61415        
61416         /** eval:var:tpl eval:var:fm eval:var:useF eval:var:undef  */
61417         eval(body);
61418         
61419         return this;
61420     },
61421
61422     applyTemplate : function(values){
61423         return this.master.compiled.call(this, values, {});
61424         //var s = this.subs;
61425     },
61426
61427     apply : function(){
61428         return this.applyTemplate.apply(this, arguments);
61429     }
61430
61431  });
61432
61433 Roo.XTemplate.from = function(el){
61434     el = Roo.getDom(el);
61435     return new Roo.XTemplate(el.value || el.innerHTML);
61436 };